2 * Unit tests for window message handling
4 * Copyright 1999 Ove Kaaven
5 * Copyright 2003 Dimitrie O. Paun
6 * Copyright 2004, 2005 Dmitry Timoshkov
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
27 #define _WIN32_WINNT 0x0501 /* For WM_CHANGEUISTATE,QS_RAWINPUT */
35 #include "wine/test.h"
37 #define MDI_FIRST_CHILD_ID 2004
39 /* undocumented SWP flags - from SDK 3.1 */
40 #define SWP_NOCLIENTSIZE 0x0800
41 #define SWP_NOCLIENTMOVE 0x1000
43 #define SW_NORMALNA 0xCC /* undoc. flag in MinMaximize */
45 #define WND_PARENT_ID 1
46 #define WND_POPUP_ID 2
47 #define WND_CHILD_ID 3
49 static BOOL test_DestroyWindow_flag;
50 static HWINEVENTHOOK hEvent_hook;
52 static HWND (WINAPI *pGetAncestor)(HWND,UINT);
55 FIXME: add tests for these
56 Window Edge Styles (Win31/Win95/98 look), in order of precedence:
57 WS_EX_DLGMODALFRAME: double border, WS_CAPTION allowed
58 WS_THICKFRAME: thick border
59 WS_DLGFRAME: double border, WS_CAPTION not allowed (but possibly shown anyway)
60 WS_BORDER (default for overlapped windows): single black border
61 none (default for child (and popup?) windows): no border
78 UINT message; /* the WM_* code */
79 msg_flags_t flags; /* message props */
80 WPARAM wParam; /* expected value of wParam */
81 LPARAM lParam; /* expected value of lParam */
84 /* Empty message sequence */
85 static const struct message WmEmptySeq[] =
89 /* CreateWindow (for overlapped window, not initially visible) (16/32) */
90 static const struct message WmCreateOverlappedSeq[] = {
91 { HCBT_CREATEWND, hook },
92 { WM_GETMINMAXINFO, sent },
93 { WM_NCCREATE, sent },
94 { WM_NCCALCSIZE, sent|wparam, 0 },
95 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
97 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
100 /* SetWindowPos(SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE)
101 * for a not visible overlapped window.
103 static const struct message WmSWP_ShowOverlappedSeq[] = {
104 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
105 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
106 { WM_NCPAINT, sent|wparam|optional, 1 },
107 { WM_GETTEXT, sent|defwinproc|optional },
108 { WM_ERASEBKGND, sent|optional },
109 { HCBT_ACTIVATE, hook },
110 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
111 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
112 { WM_WINDOWPOSCHANGING, sent|wparam|optional, SWP_NOSIZE|SWP_NOMOVE }, /* Win9x: SWP_NOSENDCHANGING */
113 { WM_ACTIVATEAPP, sent|wparam, 1 },
114 { WM_NCACTIVATE, sent|wparam, 1 },
115 { WM_GETTEXT, sent|defwinproc|optional },
116 { WM_ACTIVATE, sent|wparam, 1 },
117 { HCBT_SETFOCUS, hook },
118 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
119 { WM_IME_NOTIFY, sent|defwinproc|optional },
120 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
121 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
122 { WM_NCPAINT, sent|wparam|optional, 1 },
123 { WM_GETTEXT, sent|defwinproc|optional },
124 { WM_ERASEBKGND, sent|optional },
125 /* Win9x adds SWP_NOZORDER below */
126 { WM_WINDOWPOSCHANGED, sent, /*|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE*/ },
127 { WM_NCCALCSIZE, sent|wparam|optional, 1 },
128 { WM_NCPAINT, sent|wparam|optional, 1 },
129 { WM_ERASEBKGND, sent|optional },
130 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
133 /* SetWindowPos(SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE)
134 * for a visible overlapped window.
136 static const struct message WmSWP_HideOverlappedSeq[] = {
137 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE },
138 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
139 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
143 /* SetWindowPos(SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOMOVE)
144 * for a visible overlapped window.
146 static const struct message WmSWP_ResizeSeq[] = {
147 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOMOVE },
148 { WM_GETMINMAXINFO, sent|defwinproc },
149 { WM_NCCALCSIZE, sent|wparam, TRUE },
150 { WM_NCPAINT, sent|optional },
151 { WM_GETTEXT, sent|defwinproc|optional },
152 { WM_ERASEBKGND, sent|optional },
153 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
154 { WM_SIZE, sent|defwinproc|wparam, SIZE_RESTORED },
155 { WM_NCCALCSIZE, sent|wparam|optional, TRUE },
156 { WM_NCPAINT, sent|optional },
157 { WM_GETTEXT, sent|defwinproc|optional },
158 { WM_ERASEBKGND, sent|optional },
159 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
160 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam|optional, 0, 0 }, /* XP sends a duplicate */
164 /* SetWindowPos(SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOMOVE)
165 * for a visible popup window.
167 static const struct message WmSWP_ResizePopupSeq[] = {
168 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOMOVE },
169 { WM_NCCALCSIZE, sent|wparam, TRUE },
170 { WM_NCPAINT, sent|optional },
171 { WM_GETTEXT, sent|defwinproc|optional },
172 { WM_ERASEBKGND, sent|optional },
173 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
174 { WM_SIZE, sent|defwinproc|wparam, SIZE_RESTORED },
175 { WM_NCCALCSIZE, sent|wparam|optional, TRUE },
176 { WM_NCPAINT, sent|optional },
177 { WM_GETTEXT, sent|defwinproc|optional },
178 { WM_ERASEBKGND, sent|optional },
179 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
183 /* SetWindowPos(SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOSIZE)
184 * for a visible overlapped window.
186 static const struct message WmSWP_MoveSeq[] = {
187 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE },
188 { WM_NCPAINT, sent|optional },
189 { WM_GETTEXT, sent|defwinproc|optional },
190 { WM_ERASEBKGND, sent|optional },
191 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOZORDER|SWP_NOCLIENTSIZE },
192 { WM_MOVE, sent|defwinproc|wparam, 0 },
193 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
197 /* SetWindowPos(SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|
198 SWP_NOZORDER|SWP_FRAMECHANGED)
199 * for a visible overlapped window with WS_CLIPCHILDREN style set.
201 static const struct message WmSWP_FrameChanged_clip[] = {
202 { WM_WINDOWPOSCHANGING, sent|wparam|parent, SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_FRAMECHANGED },
203 { WM_NCCALCSIZE, sent|wparam|parent, 1 },
204 { WM_NCPAINT, sent|parent }, /* wparam != 1 */
205 { WM_GETTEXT, sent|parent|defwinproc|optional },
206 { WM_ERASEBKGND, sent|parent|optional }, /* FIXME: remove optional once Wine is fixed */
207 { WM_NCPAINT, sent }, /* wparam != 1 */
208 { WM_ERASEBKGND, sent },
209 { WM_WINDOWPOSCHANGED, sent|wparam|parent, SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_FRAMECHANGED|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
210 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
214 /* SetWindowPos(SWP_NOSIZE|SWP_NOMOVE|SWP_DEFERERASE|SWP_NOACTIVATE|
215 SWP_NOZORDER|SWP_FRAMECHANGED)
216 * for a visible overlapped window.
218 static const struct message WmSWP_FrameChangedDeferErase[] = {
219 { WM_WINDOWPOSCHANGING, sent|wparam|parent, SWP_NOSIZE|SWP_NOMOVE|SWP_DEFERERASE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_FRAMECHANGED },
220 { WM_NCCALCSIZE, sent|wparam|parent, 1 },
221 { WM_WINDOWPOSCHANGED, sent|wparam|parent, SWP_NOSIZE|SWP_NOMOVE|SWP_DEFERERASE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_FRAMECHANGED|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
222 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
223 { WM_PAINT, sent|parent },
224 { WM_NCPAINT, sent|beginpaint|parent }, /* wparam != 1 */
225 { WM_GETTEXT, sent|beginpaint|parent|defwinproc|optional },
227 { WM_NCPAINT, sent|beginpaint }, /* wparam != 1 */
228 { WM_ERASEBKGND, sent|beginpaint },
232 /* SetWindowPos(SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|
233 SWP_NOZORDER|SWP_FRAMECHANGED)
234 * for a visible overlapped window without WS_CLIPCHILDREN style set.
236 static const struct message WmSWP_FrameChanged_noclip[] = {
237 { WM_WINDOWPOSCHANGING, sent|wparam|parent, SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_FRAMECHANGED },
238 { WM_NCCALCSIZE, sent|wparam|parent, 1 },
239 { WM_NCPAINT, sent|parent }, /* wparam != 1 */
240 { WM_GETTEXT, sent|parent|defwinproc|optional },
241 { WM_ERASEBKGND, sent|parent|optional }, /* FIXME: remove optional once Wine is fixed */
242 { WM_WINDOWPOSCHANGED, sent|wparam|parent, SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_FRAMECHANGED|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
243 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
245 { WM_NCPAINT, sent|beginpaint }, /* wparam != 1 */
246 { WM_ERASEBKGND, sent|beginpaint },
250 /* ShowWindow(SW_SHOW) for a not visible overlapped window */
251 static const struct message WmShowOverlappedSeq[] = {
252 { WM_SHOWWINDOW, sent|wparam, 1 },
253 { WM_NCPAINT, sent|wparam|optional, 1 },
254 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
255 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
256 { WM_NCPAINT, sent|wparam|optional, 1 },
257 { WM_GETTEXT, sent|defwinproc|optional },
258 { WM_ERASEBKGND, sent|optional },
259 { HCBT_ACTIVATE, hook },
260 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
261 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
262 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOMOVE },
263 { WM_NCPAINT, sent|wparam|optional, 1 },
264 { WM_ACTIVATEAPP, sent|wparam, 1 },
265 { WM_NCACTIVATE, sent|wparam, 1 },
266 { WM_GETTEXT, sent|defwinproc|optional },
267 { WM_ACTIVATE, sent|wparam, 1 },
268 { HCBT_SETFOCUS, hook },
269 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
270 { WM_IME_NOTIFY, sent|defwinproc|optional },
271 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
272 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
273 { WM_NCPAINT, sent|wparam|optional, 1 },
274 { WM_GETTEXT, sent|defwinproc|optional },
275 { WM_ERASEBKGND, sent|optional },
276 /* Win9x adds SWP_NOZORDER below */
277 { WM_WINDOWPOSCHANGED, sent, /*|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE*/ },
278 { WM_NCCALCSIZE, sent|optional },
279 { WM_NCPAINT, sent|optional },
280 { WM_ERASEBKGND, sent|optional },
281 #if 0 /* CreateWindow/ShowWindow(SW_SHOW) also generates WM_SIZE/WM_MOVE
282 * messages. Does that mean that CreateWindow doesn't set initial
283 * window dimensions for overlapped windows?
290 /* ShowWindow(SW_SHOWMAXIMIZED) for a not visible overlapped window */
291 static const struct message WmShowMaxOverlappedSeq[] = {
292 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
293 { WM_GETMINMAXINFO, sent },
294 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|0x8000 },
295 { WM_GETMINMAXINFO, sent|defwinproc },
296 { WM_NCCALCSIZE, sent|wparam, TRUE },
297 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
298 { HCBT_ACTIVATE, hook },
299 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
300 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
301 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOMOVE },
302 { WM_ACTIVATEAPP, sent|wparam, 1 },
303 { WM_NCACTIVATE, sent|wparam, 1 },
304 { WM_GETTEXT, sent|defwinproc|optional },
305 { WM_ACTIVATE, sent|wparam, 1 },
306 { HCBT_SETFOCUS, hook },
307 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
308 { WM_IME_NOTIFY, sent|defwinproc|optional },
309 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
310 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
311 { WM_NCPAINT, sent|wparam|optional, 1 },
312 { WM_GETTEXT, sent|defwinproc|optional },
313 { WM_ERASEBKGND, sent|optional },
314 /* Win9x adds SWP_NOZORDER below */
315 { WM_WINDOWPOSCHANGED, sent, /*|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE*/ },
316 { WM_MOVE, sent|defwinproc },
317 { WM_SIZE, sent|defwinproc },
318 { WM_NCCALCSIZE, sent|optional },
319 { WM_NCPAINT, sent|optional },
320 { WM_ERASEBKGND, sent|optional },
321 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
322 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
325 /* ShowWindow(SW_HIDE) for a visible overlapped window */
326 static const struct message WmHideOverlappedSeq[] = {
327 { WM_SHOWWINDOW, sent|wparam, 0 },
328 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE },
329 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
330 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
331 { WM_SIZE, sent|optional }, /* XP doesn't send it */
332 { WM_MOVE, sent|optional }, /* XP doesn't send it */
333 { WM_NCACTIVATE, sent|wparam, 0 },
334 { WM_ACTIVATE, sent|wparam, 0 },
335 { WM_ACTIVATEAPP, sent|wparam, 0 },
336 { WM_KILLFOCUS, sent|wparam, 0 },
337 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
338 { WM_IME_NOTIFY, sent|optional|defwinproc },
341 /* DestroyWindow for a visible overlapped window */
342 static const struct message WmDestroyOverlappedSeq[] = {
343 { HCBT_DESTROYWND, hook },
344 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
345 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
346 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
347 { WM_NCACTIVATE, sent|wparam, 0 },
348 { WM_ACTIVATE, sent|wparam, 0 },
349 { WM_ACTIVATEAPP, sent|wparam, 0 },
350 { WM_KILLFOCUS, sent|wparam, 0 },
351 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
352 { WM_IME_NOTIFY, sent|optional|defwinproc },
353 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
354 { WM_DESTROY, sent },
355 { WM_NCDESTROY, sent },
358 /* CreateWindow(WS_MAXIMIZE|WS_VISIBLE) for popup window */
359 static const struct message WmCreateMaxPopupSeq[] = {
360 { HCBT_CREATEWND, hook },
361 { WM_NCCREATE, sent },
362 { WM_NCCALCSIZE, sent|wparam, 0 },
366 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
367 { WM_GETMINMAXINFO, sent },
368 { WM_WINDOWPOSCHANGING, sent /*|wparam, SWP_NOACTIVATE|SWP_FRAMECHANGED|0x8000*/ },
369 { WM_NCCALCSIZE, sent|wparam, TRUE },
370 { WM_WINDOWPOSCHANGED, sent /*|wparam, SWP_NOACTIVATE|SWP_FRAMECHANGED|SWP_NOREDRAW|SWP_NOZORDER|0x8000*/ },
371 { WM_MOVE, sent|defwinproc },
372 { WM_SIZE, sent|defwinproc },
373 { WM_SHOWWINDOW, sent|wparam, 1 },
374 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
375 { HCBT_ACTIVATE, hook },
376 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
377 { WM_WINDOWPOSCHANGING, sent|wparam|optional, SWP_NOSIZE|SWP_NOMOVE },
378 { WM_ACTIVATEAPP, sent|wparam, 1 },
379 { WM_NCACTIVATE, sent|wparam, 1 },
380 { WM_ACTIVATE, sent|wparam, 1 },
381 { HCBT_SETFOCUS, hook },
382 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
383 { WM_SYNCPAINT, sent|wparam|optional, 4 },
384 { WM_NCPAINT, sent|wparam|optional, 1 },
385 { WM_ERASEBKGND, sent|optional },
386 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOCLIENTMOVE|SWP_NOCLIENTSIZE|SWP_SHOWWINDOW|SWP_NOMOVE|SWP_NOZORDER|SWP_NOSIZE },
389 /* CreateWindow(WS_MAXIMIZE) for popup window, not initially visible */
390 static const struct message WmCreateInvisibleMaxPopupSeq[] = {
391 { HCBT_CREATEWND, hook },
392 { WM_NCCREATE, sent },
393 { WM_NCCALCSIZE, sent|wparam, 0 },
397 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
398 { WM_GETMINMAXINFO, sent },
399 { WM_WINDOWPOSCHANGING, sent /*|wparam, SWP_NOACTIVATE|SWP_FRAMECHANGED|0x8000*/ },
400 { WM_NCCALCSIZE, sent|wparam, TRUE },
401 { WM_WINDOWPOSCHANGED, sent /*|wparam, SWP_NOACTIVATE|SWP_FRAMECHANGED|SWP_NOREDRAW|SWP_NOZORDER|0x8000*/ },
402 { WM_MOVE, sent|defwinproc },
403 { WM_SIZE, sent|defwinproc },
406 /* ShowWindow(SW_SHOWMAXIMIZED) for a resized not visible popup window */
407 static const struct message WmShowMaxPopupResizedSeq[] = {
408 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
409 { WM_GETMINMAXINFO, sent },
410 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED },
411 { WM_NCCALCSIZE, sent|wparam, TRUE },
412 { HCBT_ACTIVATE, hook },
413 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
414 { WM_WINDOWPOSCHANGING, sent|wparam|optional, SWP_NOSIZE|SWP_NOMOVE },
415 { WM_ACTIVATEAPP, sent|wparam, 1 },
416 { WM_NCACTIVATE, sent|wparam, 1 },
417 { WM_ACTIVATE, sent|wparam, 1 },
418 { HCBT_SETFOCUS, hook },
419 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
420 { WM_NCPAINT, sent|wparam|optional, 1 },
421 { WM_ERASEBKGND, sent|optional },
422 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOCLIENTMOVE|SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOMOVE|SWP_NOZORDER },
423 /* WinNT4.0 sends WM_MOVE */
424 { WM_MOVE, sent|defwinproc|optional },
425 { WM_SIZE, sent|defwinproc },
428 /* ShowWindow(SW_SHOWMAXIMIZED) for a not visible popup window */
429 static const struct message WmShowMaxPopupSeq[] = {
430 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
431 { WM_GETMINMAXINFO, sent },
432 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED },
433 { WM_NCCALCSIZE, sent|wparam, TRUE },
434 { HCBT_ACTIVATE, hook },
435 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
436 { WM_WINDOWPOSCHANGING, sent|wparam|optional, SWP_NOSIZE|SWP_NOMOVE },
437 { WM_ACTIVATEAPP, sent|wparam, 1 },
438 { WM_NCACTIVATE, sent|wparam, 1 },
439 { WM_ACTIVATE, sent|wparam, 1 },
440 { HCBT_SETFOCUS, hook },
441 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
442 { WM_SYNCPAINT, sent|wparam|optional, 4 },
443 { WM_NCPAINT, sent|wparam|optional, 1 },
444 { WM_ERASEBKGND, sent|optional },
445 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER },
448 /* CreateWindow(WS_VISIBLE) for popup window */
449 static const struct message WmCreatePopupSeq[] = {
450 { HCBT_CREATEWND, hook },
451 { WM_NCCREATE, sent },
452 { WM_NCCALCSIZE, sent|wparam, 0 },
456 { WM_SHOWWINDOW, sent|wparam, 1 },
457 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
458 { HCBT_ACTIVATE, hook },
459 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
460 { WM_WINDOWPOSCHANGING, sent|wparam|optional, SWP_NOSIZE|SWP_NOMOVE },
461 { WM_NCPAINT, sent|wparam|optional, 1 },
462 { WM_ERASEBKGND, sent|optional },
463 { WM_ACTIVATEAPP, sent|wparam, 1 },
464 { WM_NCACTIVATE, sent|wparam, 1 },
465 { WM_ACTIVATE, sent|wparam, 1 },
466 { HCBT_SETFOCUS, hook },
467 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
468 { WM_SYNCPAINT, sent|wparam|optional, 4 },
469 { WM_NCPAINT, sent|wparam|optional, 1 },
470 { WM_ERASEBKGND, sent|optional },
471 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOCLIENTMOVE|SWP_NOCLIENTSIZE|SWP_SHOWWINDOW|SWP_NOMOVE|SWP_NOZORDER|SWP_NOSIZE },
474 /* ShowWindow(SW_SHOWMAXIMIZED) for a visible popup window */
475 static const struct message WmShowVisMaxPopupSeq[] = {
476 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
477 { WM_GETMINMAXINFO, sent },
478 { WM_GETTEXT, sent|optional },
479 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|0x8000 },
480 { WM_NCCALCSIZE, sent|wparam, TRUE },
481 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
482 { WM_NCPAINT, sent|wparam|optional, 1 },
483 { WM_ERASEBKGND, sent|optional },
484 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOZORDER|0x8000 },
485 { WM_MOVE, sent|defwinproc },
486 { WM_SIZE, sent|defwinproc },
489 /* CreateWindow (for a child popup window, not initially visible) */
490 static const struct message WmCreateChildPopupSeq[] = {
491 { HCBT_CREATEWND, hook },
492 { WM_NCCREATE, sent },
493 { WM_NCCALCSIZE, sent|wparam, 0 },
495 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
500 /* CreateWindow (for a popup window, not initially visible,
501 * which sets WS_VISIBLE in WM_CREATE handler)
503 static const struct message WmCreateInvisiblePopupSeq[] = {
504 { HCBT_CREATEWND, hook },
505 { WM_NCCREATE, sent },
506 { WM_NCCALCSIZE, sent|wparam, 0 },
508 { WM_STYLECHANGING, sent },
509 { WM_STYLECHANGED, sent },
510 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
515 /* SetWindowPos(SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER)
516 * for a popup window with WS_VISIBLE style set
518 static const struct message WmShowVisiblePopupSeq_2[] = {
519 { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
522 /* SetWindowPos(SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE)
523 * for a popup window with WS_VISIBLE style set
525 static const struct message WmShowVisiblePopupSeq_3[] = {
526 { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
527 { HCBT_ACTIVATE, hook },
528 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
529 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
530 { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
531 { WM_NCACTIVATE, sent|wparam, 1 },
532 { WM_ACTIVATE, sent|wparam, 1 },
533 { HCBT_SETFOCUS, hook },
534 { WM_KILLFOCUS, sent|parent },
535 { WM_IME_SETCONTEXT, sent|parent|wparam|optional, 0 },
536 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
537 { WM_IME_NOTIFY, sent|defwinproc|optional },
538 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
539 { WM_SETFOCUS, sent|defwinproc },
542 /* CreateWindow (for child window, not initially visible) */
543 static const struct message WmCreateChildSeq[] = {
544 { HCBT_CREATEWND, hook },
545 { WM_NCCREATE, sent },
546 /* child is inserted into parent's child list after WM_NCCREATE returns */
547 { WM_NCCALCSIZE, sent|wparam, 0 },
549 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
552 { WM_PARENTNOTIFY, sent|parent|wparam, WM_CREATE },
555 /* CreateWindow (for maximized child window, not initially visible) */
556 static const struct message WmCreateMaximizedChildSeq[] = {
557 { HCBT_CREATEWND, hook },
558 { WM_NCCREATE, sent },
559 { WM_NCCALCSIZE, sent|wparam, 0 },
561 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
564 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
565 { WM_GETMINMAXINFO, sent },
566 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|0x8000 },
567 { WM_NCCALCSIZE, sent|wparam, 1 },
568 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTMOVE|0x8000 },
569 { WM_SIZE, sent|defwinproc },
570 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
571 { WM_PARENTNOTIFY, sent|parent|wparam, WM_CREATE },
574 /* CreateWindow (for a child window, initially visible) */
575 static const struct message WmCreateVisibleChildSeq[] = {
576 { HCBT_CREATEWND, hook },
577 { WM_NCCREATE, sent },
578 /* child is inserted into parent's child list after WM_NCCREATE returns */
579 { WM_NCCALCSIZE, sent|wparam, 0 },
580 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
582 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
585 { WM_PARENTNOTIFY, sent|parent|wparam, WM_CREATE },
586 { WM_SHOWWINDOW, sent|wparam, 1 },
587 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER },
588 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
589 { WM_ERASEBKGND, sent|parent|optional },
590 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
591 { WM_NCCALCSIZE, sent|wparam|optional, 1 }, /* WinXP */
592 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
595 /* ShowWindow(SW_SHOW) for a not visible child window */
596 static const struct message WmShowChildSeq[] = {
597 { WM_SHOWWINDOW, sent|wparam, 1 },
598 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
599 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
600 { WM_ERASEBKGND, sent|parent|optional },
601 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
604 /* ShowWindow(SW_HIDE) for a visible child window */
605 static const struct message WmHideChildSeq[] = {
606 { WM_SHOWWINDOW, sent|wparam, 0 },
607 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
608 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
609 { WM_ERASEBKGND, sent|parent|optional },
610 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
613 /* SetWindowPos(SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE)
614 * for a not visible child window
616 static const struct message WmShowChildSeq_2[] = {
617 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
618 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
619 { WM_CHILDACTIVATE, sent },
620 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
623 /* SetWindowPos(SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE)
624 * for a not visible child window
626 static const struct message WmShowChildSeq_3[] = {
627 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE },
628 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
629 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
632 /* SetWindowPos(SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE)
633 * for a visible child window with a caption
635 static const struct message WmShowChildSeq_4[] = {
636 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
637 { WM_CHILDACTIVATE, sent },
640 /* ShowWindow(SW_MINIMIZE) for child with invisible parent */
641 static const struct message WmShowChildInvisibleParentSeq_1[] = {
642 { HCBT_MINMAX, hook|lparam, 0, SW_MINIMIZE },
643 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOZORDER|SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOCOPYBITS|0x8000 },
644 { WM_NCCALCSIZE, sent|wparam, 1 },
645 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCOPYBITS|0x8000 },
646 { WM_MOVE, sent|defwinproc },
647 { WM_SIZE, sent|defwinproc },
648 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
649 /* FIXME: Wine creates an icon/title window while Windows doesn't */
650 { WM_PARENTNOTIFY, sent|parent|wparam|optional, WM_CREATE },
651 { WM_GETTEXT, sent|optional },
654 /* repeated ShowWindow(SW_MINIMIZE) for child with invisible parent */
655 static const struct message WmShowChildInvisibleParentSeq_1r[] = {
656 { HCBT_MINMAX, hook|lparam, 0, SW_MINIMIZE },
659 /* ShowWindow(SW_MAXIMIZE) for child with invisible parent */
660 static const struct message WmShowChildInvisibleParentSeq_2[] = {
661 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
662 { WM_GETMINMAXINFO, sent },
663 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|0x8000 },
664 { WM_NCCALCSIZE, sent|wparam, 1 },
665 { WM_CHILDACTIVATE, sent },
666 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTMOVE|0x8000 },
667 { WM_SIZE, sent|defwinproc },
668 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
671 /* repeated ShowWindow(SW_MAXIMIZE) for child with invisible parent */
672 static const struct message WmShowChildInvisibleParentSeq_2r[] = {
673 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
676 /* ShowWindow(SW_SHOWMINIMIZED) for child with invisible parent */
677 static const struct message WmShowChildInvisibleParentSeq_3[] = {
678 { HCBT_MINMAX, hook|lparam, 0, SW_SHOWMINIMIZED },
679 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOCOPYBITS|0x8000 },
680 { WM_NCCALCSIZE, sent|wparam, 1 },
681 { WM_CHILDACTIVATE, sent },
682 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCOPYBITS|0x8000 },
683 { WM_MOVE, sent|defwinproc },
684 { WM_SIZE, sent|defwinproc },
685 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
686 /* FIXME: Wine creates an icon/title window while Windows doesn't */
687 { WM_PARENTNOTIFY, sent|parent|wparam|optional, WM_CREATE },
688 { WM_GETTEXT, sent|optional },
691 /* repeated ShowWindow(SW_SHOWMINIMIZED) for child with invisible parent */
692 static const struct message WmShowChildInvisibleParentSeq_3r[] = {
693 { HCBT_MINMAX, hook|lparam, 0, SW_SHOWMINIMIZED },
696 /* ShowWindow(SW_SHOWMINNOACTIVE) for child with invisible parent */
697 static const struct message WmShowChildInvisibleParentSeq_4[] = {
698 { HCBT_MINMAX, hook|lparam, 0, SW_SHOWMINNOACTIVE },
699 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOCOPYBITS|0x8000 },
700 { WM_NCCALCSIZE, sent|wparam, 1 },
701 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCOPYBITS|0x8000 },
702 { WM_MOVE, sent|defwinproc },
703 { WM_SIZE, sent|defwinproc },
704 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
705 /* FIXME: Wine creates an icon/title window while Windows doesn't */
706 { WM_PARENTNOTIFY, sent|parent|wparam|optional, WM_CREATE },
707 { WM_GETTEXT, sent|optional },
710 /* repeated ShowWindow(SW_SHOWMINNOACTIVE) for child with invisible parent */
711 static const struct message WmShowChildInvisibleParentSeq_4r[] = {
712 { HCBT_MINMAX, hook|lparam, 0, SW_SHOWMINNOACTIVE },
715 /* ShowWindow(SW_SHOW) for child with invisible parent */
716 static const struct message WmShowChildInvisibleParentSeq_5[] = {
717 { WM_SHOWWINDOW, sent|wparam, 1 },
720 /* ShowWindow(SW_HIDE) for child with invisible parent */
721 static const struct message WmHideChildInvisibleParentSeq[] = {
722 { WM_SHOWWINDOW, sent|wparam, 0 },
725 /* SetWindowPos(SWP_SHOWWINDOW) for child with invisible parent */
726 static const struct message WmShowChildInvisibleParentSeq_6[] = {
727 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER },
728 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
729 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
732 /* SetWindowPos(SWP_HIDEWINDOW) for child with invisible parent */
733 static const struct message WmHideChildInvisibleParentSeq_2[] = {
734 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
735 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
736 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
739 /* DestroyWindow for a visible child window */
740 static const struct message WmDestroyChildSeq[] = {
741 { HCBT_DESTROYWND, hook },
742 { WM_PARENTNOTIFY, sent|parent|wparam, WM_DESTROY },
743 { WM_SHOWWINDOW, sent|wparam, 0 },
744 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
745 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
746 { WM_ERASEBKGND, sent|parent|optional },
747 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
748 { HCBT_SETFOCUS, hook }, /* set focus to a parent */
749 { WM_KILLFOCUS, sent },
750 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
751 { WM_IME_SETCONTEXT, sent|wparam|parent|optional, 1 },
752 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
753 { WM_SETFOCUS, sent|parent },
754 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
755 { WM_DESTROY, sent },
756 { WM_DESTROY, sent|optional }, /* some other (IME?) window */
757 { WM_NCDESTROY, sent|optional }, /* some other (IME?) window */
758 { WM_NCDESTROY, sent },
761 /* DestroyWindow for a visible child window with invisible parent */
762 static const struct message WmDestroyInvisibleChildSeq[] = {
763 { HCBT_DESTROYWND, hook },
764 { WM_PARENTNOTIFY, sent|parent|wparam, WM_DESTROY },
765 { WM_SHOWWINDOW, sent|wparam, 0 },
766 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
767 { WM_DESTROY, sent },
768 { WM_NCDESTROY, sent },
771 /* Moving the mouse in nonclient area */
772 static const struct message WmMouseMoveInNonClientAreaSeq[] = { /* FIXME: add */
773 { WM_NCHITTEST, sent },
774 { WM_SETCURSOR, sent },
775 { WM_NCMOUSEMOVE, posted },
778 /* Moving the mouse in client area */
779 static const struct message WmMouseMoveInClientAreaSeq[] = { /* FIXME: add */
780 { WM_NCHITTEST, sent },
781 { WM_SETCURSOR, sent },
782 { WM_MOUSEMOVE, posted },
785 /* Moving by dragging the title bar (after WM_NCHITTEST and WM_SETCURSOR) (outline move) */
786 static const struct message WmDragTitleBarSeq[] = { /* FIXME: add */
787 { WM_NCLBUTTONDOWN, sent|wparam, HTCAPTION },
788 { WM_SYSCOMMAND, sent|defwinproc|wparam, SC_MOVE+2 },
789 { WM_GETMINMAXINFO, sent|defwinproc },
790 { WM_ENTERSIZEMOVE, sent|defwinproc },
791 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, 0 },
792 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, 0 },
793 { WM_MOVE, sent|defwinproc },
794 { WM_EXITSIZEMOVE, sent|defwinproc },
797 /* Sizing by dragging the thick borders (after WM_NCHITTEST and WM_SETCURSOR) (outline move) */
798 static const struct message WmDragThickBordersBarSeq[] = { /* FIXME: add */
799 { WM_NCLBUTTONDOWN, sent|wparam, 0xd },
800 { WM_SYSCOMMAND, sent|defwinproc|wparam, 0xf004 },
801 { WM_GETMINMAXINFO, sent|defwinproc },
802 { WM_ENTERSIZEMOVE, sent|defwinproc },
803 { WM_SIZING, sent|defwinproc|wparam, 4}, /* one for each mouse movement */
804 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, 0 },
805 { WM_GETMINMAXINFO, sent|defwinproc },
806 { WM_NCCALCSIZE, sent|defwinproc|wparam, 1 },
807 { WM_NCPAINT, sent|defwinproc|wparam, 1 },
808 { WM_GETTEXT, sent|defwinproc },
809 { WM_ERASEBKGND, sent|defwinproc },
810 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, 0 },
811 { WM_MOVE, sent|defwinproc },
812 { WM_SIZE, sent|defwinproc },
813 { WM_EXITSIZEMOVE, sent|defwinproc },
816 /* Resizing child window with MoveWindow (32) */
817 static const struct message WmResizingChildWithMoveWindowSeq[] = {
818 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
819 { WM_NCCALCSIZE, sent|wparam, 1 },
820 { WM_ERASEBKGND, sent|optional },
821 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
822 { WM_MOVE, sent|defwinproc },
823 { WM_SIZE, sent|defwinproc },
824 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
827 /* Clicking on inactive button */
828 static const struct message WmClickInactiveButtonSeq[] = { /* FIXME: add */
829 { WM_NCHITTEST, sent },
830 { WM_PARENTNOTIFY, sent|parent|wparam, WM_LBUTTONDOWN },
831 { WM_MOUSEACTIVATE, sent },
832 { WM_MOUSEACTIVATE, sent|parent|defwinproc },
833 { WM_SETCURSOR, sent },
834 { WM_SETCURSOR, sent|parent|defwinproc },
835 { WM_LBUTTONDOWN, posted },
836 { WM_KILLFOCUS, posted|parent },
837 { WM_SETFOCUS, posted },
838 { WM_CTLCOLORBTN, posted|parent },
839 { BM_SETSTATE, posted },
840 { WM_CTLCOLORBTN, posted|parent },
841 { WM_LBUTTONUP, posted },
842 { BM_SETSTATE, posted },
843 { WM_CTLCOLORBTN, posted|parent },
844 { WM_COMMAND, posted|parent },
847 /* Reparenting a button (16/32) */
848 /* The last child (button) reparented gets topmost for its new parent. */
849 static const struct message WmReparentButtonSeq[] = { /* FIXME: add */
850 { WM_SHOWWINDOW, sent|wparam, 0 },
851 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER },
852 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
853 { WM_ERASEBKGND, sent|parent },
854 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER },
855 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOZORDER },
856 { WM_CHILDACTIVATE, sent },
857 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOSIZE|SWP_NOREDRAW|SWP_NOZORDER },
858 { WM_MOVE, sent|defwinproc },
859 { WM_SHOWWINDOW, sent|wparam, 1 },
862 /* Creation of a custom dialog (32) */
863 static const struct message WmCreateCustomDialogSeq[] = {
864 { HCBT_CREATEWND, hook },
865 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
866 { WM_GETMINMAXINFO, sent },
867 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
868 { WM_NCCREATE, sent },
869 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
870 { WM_NCCALCSIZE, sent|wparam, 0 },
871 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
872 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
874 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
875 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
876 { WM_SHOWWINDOW, sent|wparam, 1 },
877 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
878 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
879 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
880 { HCBT_ACTIVATE, hook },
881 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
883 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
885 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
886 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
888 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOMOVE },
890 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
891 { WM_NCACTIVATE, sent|wparam, 1 },
892 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
893 { WM_GETTEXT, sent|optional|defwinproc },
894 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
895 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
896 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
897 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
898 { WM_GETTEXT, sent|optional|defwinproc },
899 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
900 { WM_ACTIVATE, sent|wparam, 1 },
901 { WM_KILLFOCUS, sent|parent },
902 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
903 { WM_IME_SETCONTEXT, sent|parent|wparam|optional, 0 },
904 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
905 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
906 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
907 { WM_IME_NOTIFY, sent|optional|defwinproc },
908 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
909 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
910 { WM_SETFOCUS, sent },
911 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
912 { WM_GETDLGCODE, sent|defwinproc|wparam, 0 },
913 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
914 { WM_NCPAINT, sent|wparam, 1 },
915 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
916 { WM_GETTEXT, sent|optional|defwinproc },
917 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
918 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
919 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
920 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
921 { WM_GETTEXT, sent|optional|defwinproc },
922 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
923 { WM_ERASEBKGND, sent },
924 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
925 { WM_CTLCOLORDLG, sent|defwinproc },
926 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
927 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
928 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
929 { WM_GETTEXT, sent|optional },
930 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
931 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
932 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
933 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
934 { WM_GETTEXT, sent|optional },
935 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
936 { WM_NCCALCSIZE, sent|optional },
937 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
938 { WM_NCPAINT, sent|optional },
939 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
940 { WM_GETTEXT, sent|optional|defwinproc },
941 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
942 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
943 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
944 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
945 { WM_GETTEXT, sent|optional|defwinproc },
946 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
947 { WM_ERASEBKGND, sent|optional },
948 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
949 { WM_CTLCOLORDLG, sent|optional|defwinproc },
950 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
951 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
953 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
957 /* Calling EndDialog for a custom dialog (32) */
958 static const struct message WmEndCustomDialogSeq[] = {
959 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
960 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
961 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
962 { WM_GETTEXT, sent|optional },
963 { HCBT_ACTIVATE, hook },
964 { WM_NCACTIVATE, sent|wparam, 0 },
965 { WM_GETTEXT, sent|optional|defwinproc },
966 { WM_GETTEXT, sent|optional|defwinproc },
967 { WM_ACTIVATE, sent|wparam, 0 },
968 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
969 { WM_WINDOWPOSCHANGING, sent|wparam|optional, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
970 { HCBT_SETFOCUS, hook },
971 { WM_KILLFOCUS, sent },
972 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
973 { WM_IME_SETCONTEXT, sent|parent|wparam|defwinproc|optional, 1 },
974 { WM_IME_NOTIFY, sent|optional },
975 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
976 { WM_SETFOCUS, sent|parent|defwinproc },
979 /* ShowWindow(SW_SHOW) for a custom dialog (initially invisible) */
980 static const struct message WmShowCustomDialogSeq[] = {
981 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
982 { WM_SHOWWINDOW, sent|wparam, 1 },
983 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
984 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
985 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
986 { HCBT_ACTIVATE, hook },
987 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
989 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
990 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
992 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
993 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOMOVE },
994 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
995 { WM_ACTIVATEAPP, sent|wparam|optional, 1 },
996 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
997 { WM_NCACTIVATE, sent|wparam, 1 },
998 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
999 { WM_ACTIVATE, sent|wparam, 1 },
1001 { WM_KILLFOCUS, sent|parent },
1002 { WM_IME_SETCONTEXT, sent|parent|wparam|optional, 0 },
1003 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
1004 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
1005 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
1006 { WM_IME_NOTIFY, sent|optional|defwinproc },
1007 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam|optional, OBJID_CLIENT, 0 },
1008 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1009 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1010 { WM_SETFOCUS, sent },
1011 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1012 { WM_GETDLGCODE, sent|defwinproc|wparam, 0 },
1013 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1014 { WM_NCPAINT, sent|wparam, 1 },
1015 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1016 { WM_ERASEBKGND, sent },
1017 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1018 { WM_CTLCOLORDLG, sent|defwinproc },
1020 { EVENT_OBJECT_DEFACTIONCHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1021 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1024 /* Creation and destruction of a modal dialog (32) */
1025 static const struct message WmModalDialogSeq[] = {
1026 { WM_CANCELMODE, sent|parent },
1027 { HCBT_SETFOCUS, hook },
1028 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1029 { WM_KILLFOCUS, sent|parent },
1030 { WM_IME_SETCONTEXT, sent|parent|wparam|optional, 0 },
1031 { EVENT_OBJECT_STATECHANGE, winevent_hook|wparam|lparam, 0, 0 },
1032 { WM_ENABLE, sent|parent|wparam, 0 },
1033 { HCBT_CREATEWND, hook },
1034 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
1035 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1036 { WM_SETFONT, sent },
1037 { WM_INITDIALOG, sent },
1038 { WM_CHANGEUISTATE, sent|optional },
1039 { WM_SHOWWINDOW, sent },
1040 { HCBT_ACTIVATE, hook },
1041 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
1042 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
1043 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOMOVE },
1044 { WM_NCACTIVATE, sent|wparam, 1 },
1045 { WM_GETTEXT, sent|optional },
1046 { WM_ACTIVATE, sent|wparam, 1 },
1047 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE },
1048 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1049 { WM_NCPAINT, sent },
1050 { WM_GETTEXT, sent|optional },
1051 { WM_ERASEBKGND, sent },
1052 { WM_CTLCOLORDLG, sent },
1053 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1054 { WM_GETTEXT, sent|optional },
1055 { WM_NCCALCSIZE, sent|optional },
1056 { WM_NCPAINT, sent|optional },
1057 { WM_GETTEXT, sent|optional },
1058 { WM_ERASEBKGND, sent|optional },
1059 { WM_CTLCOLORDLG, sent|optional },
1060 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1061 { WM_PAINT, sent|optional },
1062 { WM_CTLCOLORBTN, sent },
1063 { WM_ENTERIDLE, sent|parent|optional },
1064 { WM_ENTERIDLE, sent|parent|optional },
1065 { WM_ENTERIDLE, sent|parent|optional },
1066 { WM_ENTERIDLE, sent|parent|optional },
1067 { WM_ENTERIDLE, sent|parent|optional },
1068 { WM_ENTERIDLE, sent|parent|optional },
1069 { WM_ENTERIDLE, sent|parent|optional },
1070 { WM_ENTERIDLE, sent|parent|optional },
1071 { WM_ENTERIDLE, sent|parent|optional },
1072 { WM_ENTERIDLE, sent|parent|optional },
1073 { WM_ENTERIDLE, sent|parent|optional },
1074 { WM_ENTERIDLE, sent|parent|optional },
1075 { WM_ENTERIDLE, sent|parent|optional },
1076 { WM_ENTERIDLE, sent|parent|optional },
1077 { WM_ENTERIDLE, sent|parent|optional },
1078 { WM_ENTERIDLE, sent|parent|optional },
1079 { WM_ENTERIDLE, sent|parent|optional },
1080 { WM_ENTERIDLE, sent|parent|optional },
1081 { WM_ENTERIDLE, sent|parent|optional },
1082 { WM_ENTERIDLE, sent|parent|optional },
1084 { EVENT_OBJECT_STATECHANGE, winevent_hook|wparam|lparam, 0, 0 },
1085 { WM_ENABLE, sent|parent|wparam, 1 },
1086 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE },
1087 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
1088 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1089 { WM_GETTEXT, sent|optional },
1090 { HCBT_ACTIVATE, hook },
1091 { WM_NCACTIVATE, sent|wparam, 0 },
1092 { WM_GETTEXT, sent|optional },
1093 { WM_ACTIVATE, sent|wparam, 0 },
1094 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
1095 { WM_WINDOWPOSCHANGING, sent|optional },
1096 { HCBT_SETFOCUS, hook },
1097 { WM_IME_SETCONTEXT, sent|parent|wparam|defwinproc|optional, 1 },
1098 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1099 { WM_SETFOCUS, sent|parent|defwinproc },
1100 { EVENT_SYSTEM_DIALOGEND, winevent_hook|wparam|lparam, 0, 0 },
1101 { HCBT_DESTROYWND, hook },
1102 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
1103 { WM_DESTROY, sent },
1104 { WM_NCDESTROY, sent },
1107 /* Creation of a modal dialog that is resized inside WM_INITDIALOG (32) */
1108 static const struct message WmCreateModalDialogResizeSeq[] = { /* FIXME: add */
1109 /* (inside dialog proc, handling WM_INITDIALOG) */
1110 { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
1111 { WM_NCCALCSIZE, sent },
1112 { WM_NCACTIVATE, sent|parent|wparam, 0 },
1113 { WM_GETTEXT, sent|defwinproc },
1114 { WM_ACTIVATE, sent|parent|wparam, 0 },
1115 { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
1116 { WM_WINDOWPOSCHANGING, sent|parent },
1117 { WM_NCACTIVATE, sent|wparam, 1 },
1118 { WM_ACTIVATE, sent|wparam, 1 },
1119 { WM_WINDOWPOSCHANGED, sent|wparam, 0 },
1120 { WM_SIZE, sent|defwinproc },
1121 /* (setting focus) */
1122 { WM_SHOWWINDOW, sent|wparam, 1 },
1123 { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
1124 { WM_NCPAINT, sent },
1125 { WM_GETTEXT, sent|defwinproc },
1126 { WM_ERASEBKGND, sent },
1127 { WM_CTLCOLORDLG, sent|defwinproc },
1128 { WM_WINDOWPOSCHANGED, sent|wparam, 0 },
1130 /* (bunch of WM_CTLCOLOR* for each control) */
1131 { WM_PAINT, sent|parent },
1132 { WM_ENTERIDLE, sent|parent|wparam, 0 },
1133 { WM_SETCURSOR, sent|parent },
1136 /* SetMenu for NonVisible windows with size change*/
1137 static const struct message WmSetMenuNonVisibleSizeChangeSeq[] = {
1138 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1139 { WM_NCCALCSIZE, sent|wparam, 1 },
1140 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
1141 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW },
1142 { WM_MOVE, sent|defwinproc },
1143 { WM_SIZE, sent|defwinproc },
1144 { WM_NCCALCSIZE,sent|wparam|optional, 1 }, /* XP */
1145 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1146 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam|optional, 0, 0 }, /* XP sends a duplicate */
1147 { WM_GETTEXT, sent|optional },
1148 { WM_NCCALCSIZE, sent|wparam|optional, 1 },
1151 /* SetMenu for NonVisible windows with no size change */
1152 static const struct message WmSetMenuNonVisibleNoSizeChangeSeq[] = {
1153 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1154 { WM_NCCALCSIZE, sent|wparam, 1 },
1155 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1156 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1159 /* SetMenu for Visible windows with size change */
1160 static const struct message WmSetMenuVisibleSizeChangeSeq[] = {
1161 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1162 { WM_NCCALCSIZE, sent|wparam, 1 },
1163 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
1164 { WM_NCPAINT, sent }, /* wparam != 1 */
1165 { WM_GETTEXT, sent|defwinproc|optional },
1166 { WM_ERASEBKGND, sent|optional },
1167 { WM_ACTIVATE, sent|optional },
1168 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1169 { WM_MOVE, sent|defwinproc },
1170 { WM_SIZE, sent|defwinproc },
1171 { WM_NCCALCSIZE, sent|wparam|optional, 1 },
1172 { WM_NCPAINT, sent|optional }, /* wparam != 1 */
1173 { WM_ERASEBKGND, sent|optional },
1174 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1175 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam|optional, 0, 0 }, /* XP sends a duplicate */
1178 /* SetMenu for Visible windows with no size change */
1179 static const struct message WmSetMenuVisibleNoSizeChangeSeq[] = {
1180 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1181 { WM_NCCALCSIZE, sent|wparam, 1 },
1182 { WM_NCPAINT, sent }, /* wparam != 1 */
1183 { WM_GETTEXT, sent|defwinproc|optional },
1184 { WM_ERASEBKGND, sent|optional },
1185 { WM_ACTIVATE, sent|optional },
1186 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1187 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1190 /* DrawMenuBar for a visible window */
1191 static const struct message WmDrawMenuBarSeq[] =
1193 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1194 { WM_NCCALCSIZE, sent|wparam, 1 },
1195 { WM_NCPAINT, sent }, /* wparam != 1 */
1196 { WM_GETTEXT, sent|defwinproc|optional },
1197 { WM_ERASEBKGND, sent|optional },
1198 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1199 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1203 static const struct message WmSetRedrawFalseSeq[] =
1205 { WM_SETREDRAW, sent|wparam, 0 },
1209 static const struct message WmSetRedrawTrueSeq[] =
1211 { WM_SETREDRAW, sent|wparam, 1 },
1215 static const struct message WmEnableWindowSeq_1[] =
1217 { WM_CANCELMODE, sent|wparam|lparam, 0, 0 },
1218 { EVENT_OBJECT_STATECHANGE, winevent_hook|wparam|lparam, 0, 0 },
1219 { WM_ENABLE, sent|wparam|lparam, FALSE, 0 },
1223 static const struct message WmEnableWindowSeq_2[] =
1225 { EVENT_OBJECT_STATECHANGE, winevent_hook|wparam|lparam, 0, 0 },
1226 { WM_ENABLE, sent|wparam|lparam, TRUE, 0 },
1230 static const struct message WmGetScrollRangeSeq[] =
1232 { SBM_GETRANGE, sent },
1235 static const struct message WmGetScrollInfoSeq[] =
1237 { SBM_GETSCROLLINFO, sent },
1240 static const struct message WmSetScrollRangeSeq[] =
1242 /* MSDN claims that Windows sends SBM_SETRANGE message, but win2k SP4
1243 sends SBM_SETSCROLLINFO.
1245 { SBM_SETSCROLLINFO, sent },
1248 /* SetScrollRange for a window without a non-client area */
1249 static const struct message WmSetScrollRangeHSeq_empty[] =
1251 { EVENT_OBJECT_VALUECHANGE, winevent_hook|wparam|lparam, OBJID_HSCROLL, 0 },
1254 static const struct message WmSetScrollRangeVSeq_empty[] =
1256 { EVENT_OBJECT_VALUECHANGE, winevent_hook|wparam|lparam, OBJID_VSCROLL, 0 },
1259 static const struct message WmSetScrollRangeHVSeq[] =
1261 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER },
1262 { WM_NCCALCSIZE, sent|wparam, 1 },
1263 { WM_GETTEXT, sent|defwinproc|optional },
1264 { WM_ERASEBKGND, sent|optional },
1265 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1266 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1267 { EVENT_OBJECT_VALUECHANGE, winevent_hook|lparam|optional, 0/*OBJID_HSCROLL or OBJID_VSCROLL*/, 0 },
1270 /* SetScrollRange for a window with a non-client area */
1271 static const struct message WmSetScrollRangeHV_NC_Seq[] =
1273 { WM_WINDOWPOSCHANGING, sent, /*|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER*/ },
1274 { WM_NCCALCSIZE, sent|wparam, 1 },
1275 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
1276 { WM_NCPAINT, sent|optional },
1277 { WM_GETTEXT, sent|defwinproc|optional },
1278 { WM_GETTEXT, sent|defwinproc|optional },
1279 { WM_ERASEBKGND, sent|optional },
1280 { WM_CTLCOLORDLG, sent|defwinproc|optional }, /* sent to a parent of the dialog */
1281 { WM_WINDOWPOSCHANGED, sent, /*|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER|0x1000*/ },
1282 { WM_SIZE, sent|defwinproc },
1283 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1284 { EVENT_OBJECT_VALUECHANGE, winevent_hook|lparam|optional, 0/*OBJID_HSCROLL or OBJID_VSCROLL*/, 0 },
1285 { WM_GETTEXT, sent|optional },
1286 { WM_GETTEXT, sent|optional },
1287 { WM_GETTEXT, sent|optional },
1288 { WM_GETTEXT, sent|optional },
1291 /* test if we receive the right sequence of messages */
1292 /* after calling ShowWindow( SW_SHOWNA) */
1293 static const struct message WmSHOWNAChildInvisParInvis[] = {
1294 { WM_SHOWWINDOW, sent|wparam, 1 },
1297 static const struct message WmSHOWNAChildVisParInvis[] = {
1298 { WM_SHOWWINDOW, sent|wparam, 1 },
1301 static const struct message WmSHOWNAChildVisParVis[] = {
1302 { WM_SHOWWINDOW, sent|wparam, 1 },
1303 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER },
1306 static const struct message WmSHOWNAChildInvisParVis[] = {
1307 { WM_SHOWWINDOW, sent|wparam, 1 },
1308 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER},
1309 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1310 { WM_ERASEBKGND, sent|optional },
1311 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
1314 static const struct message WmSHOWNATopVisible[] = {
1315 { WM_SHOWWINDOW, sent|wparam, 1 },
1316 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE },
1319 static const struct message WmSHOWNATopInvisible[] = {
1320 { WM_SHOWWINDOW, sent|wparam, 1 },
1321 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
1322 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1323 { WM_NCPAINT, sent|wparam, 1 },
1324 { WM_GETTEXT, sent|defwinproc|optional },
1325 { WM_ERASEBKGND, sent|optional },
1326 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1327 { WM_NCCALCSIZE, sent|wparam|optional, 1 },
1328 { WM_NCPAINT, sent|wparam|optional, 1 },
1329 { WM_ERASEBKGND, sent|optional },
1330 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1336 static int after_end_dialog;
1337 static int sequence_cnt, sequence_size;
1338 static struct message* sequence;
1339 static int log_all_parent_messages;
1341 static void add_message(const struct message *msg)
1346 sequence = HeapAlloc( GetProcessHeap(), 0, sequence_size * sizeof (struct message) );
1348 if (sequence_cnt == sequence_size)
1351 sequence = HeapReAlloc( GetProcessHeap(), 0, sequence, sequence_size * sizeof (struct message) );
1355 sequence[sequence_cnt].message = msg->message;
1356 sequence[sequence_cnt].flags = msg->flags;
1357 sequence[sequence_cnt].wParam = msg->wParam;
1358 sequence[sequence_cnt].lParam = msg->lParam;
1363 /* try to make sure pending X events have been processed before continuing */
1364 static void flush_events(void)
1368 DWORD time = GetTickCount() + diff;
1372 MsgWaitForMultipleObjects( 0, NULL, FALSE, diff, QS_ALLINPUT );
1373 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
1374 diff = time - GetTickCount();
1378 static void flush_sequence(void)
1380 HeapFree(GetProcessHeap(), 0, sequence);
1382 sequence_cnt = sequence_size = 0;
1385 #define ok_sequence( exp, contx, todo) \
1386 ok_sequence_( (exp), (contx), (todo), __FILE__, __LINE__)
1389 static void ok_sequence_(const struct message *expected, const char *context, int todo,
1390 const char *file, int line)
1392 static const struct message end_of_sequence = { 0, 0, 0, 0 };
1393 const struct message *actual;
1396 add_message(&end_of_sequence);
1400 while (expected->message && actual->message)
1402 trace_( file, line)("expected %04x - actual %04x\n", expected->message, actual->message);
1404 if (expected->message == actual->message)
1406 if (expected->flags & wparam)
1408 if (expected->wParam != actual->wParam && todo)
1412 ok_( file, line) (FALSE,
1413 "%s: in msg 0x%04x expecting wParam 0x%x got 0x%x\n",
1414 context, expected->message, expected->wParam, actual->wParam);
1418 ok_( file, line) (expected->wParam == actual->wParam,
1419 "%s: in msg 0x%04x expecting wParam 0x%x got 0x%x\n",
1420 context, expected->message, expected->wParam, actual->wParam);
1422 if (expected->flags & lparam)
1424 if (expected->lParam != actual->lParam && todo)
1428 ok_( file, line) (FALSE,
1429 "%s: in msg 0x%04x expecting lParam 0x%lx got 0x%lx\n",
1430 context, expected->message, expected->lParam, actual->lParam);
1434 ok_( file, line) (expected->lParam == actual->lParam,
1435 "%s: in msg 0x%04x expecting lParam 0x%lx got 0x%lx\n",
1436 context, expected->message, expected->lParam, actual->lParam);
1438 if ((expected->flags & defwinproc) != (actual->flags & defwinproc) && todo)
1442 ok_( file, line) (FALSE,
1443 "%s: in msg 0x%04x expecting lParam 0x%lx got 0x%lx\n",
1444 context, expected->message, expected->lParam, actual->lParam);
1448 ok_( file, line) ((expected->flags & defwinproc) == (actual->flags & defwinproc),
1449 "%s: the msg 0x%04x should %shave been sent by DefWindowProc\n",
1450 context, expected->message, (expected->flags & defwinproc) ? "" : "NOT ");
1451 ok_( file, line) ((expected->flags & beginpaint) == (actual->flags & beginpaint),
1452 "%s: the msg 0x%04x should %shave been sent by BeginPaint\n",
1453 context, expected->message, (expected->flags & beginpaint) ? "" : "NOT ");
1454 ok_( file, line) ((expected->flags & (sent|posted)) == (actual->flags & (sent|posted)),
1455 "%s: the msg 0x%04x should have been %s\n",
1456 context, expected->message, (expected->flags & posted) ? "posted" : "sent");
1457 ok_( file, line) ((expected->flags & parent) == (actual->flags & parent),
1458 "%s: the msg 0x%04x was expected in %s\n",
1459 context, expected->message, (expected->flags & parent) ? "parent" : "child");
1460 ok_( file, line) ((expected->flags & hook) == (actual->flags & hook),
1461 "%s: the msg 0x%04x should have been sent by a hook\n",
1462 context, expected->message);
1463 ok_( file, line) ((expected->flags & winevent_hook) == (actual->flags & winevent_hook),
1464 "%s: the msg 0x%04x should have been sent by a winevent hook\n",
1465 context, expected->message);
1469 /* silently drop winevent messages if there is no support for them */
1470 else if ((expected->flags & optional) || ((expected->flags & winevent_hook) && !hEvent_hook))
1476 ok_( file, line) (FALSE, "%s: the msg 0x%04x was expected, but got msg 0x%04x instead\n",
1477 context, expected->message, actual->message);
1484 ok_( file, line) (FALSE, "%s: the msg 0x%04x was expected, but got msg 0x%04x instead\n",
1485 context, expected->message, actual->message);
1491 /* skip all optional trailing messages */
1492 while (expected->message && ((expected->flags & optional) ||
1493 ((expected->flags & winevent_hook) && !hEvent_hook)))
1499 if (expected->message || actual->message) {
1501 ok_( file, line) (FALSE, "%s: the msg sequence is not complete: expected %04x - actual %04x\n",
1502 context, expected->message, actual->message);
1508 if (expected->message || actual->message)
1509 ok_( file, line) (FALSE, "%s: the msg sequence is not complete: expected %04x - actual %04x\n",
1510 context, expected->message, actual->message);
1512 if( todo && !failcount) /* succeeded yet marked todo */
1514 ok_( file, line)( TRUE, "%s: marked \"todo_wine\" but succeeds\n", context);
1520 /******************************** MDI test **********************************/
1522 /* CreateWindow for MDI frame window, initially visible */
1523 static const struct message WmCreateMDIframeSeq[] = {
1524 { HCBT_CREATEWND, hook },
1525 { WM_GETMINMAXINFO, sent },
1526 { WM_NCCREATE, sent },
1527 { WM_NCCALCSIZE, sent|wparam, 0 },
1528 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
1529 { WM_CREATE, sent },
1530 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1531 { WM_SHOWWINDOW, sent|wparam, 1 },
1532 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
1533 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1534 { HCBT_ACTIVATE, hook },
1535 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, 0, 0 },
1536 { WM_QUERYNEWPALETTE, sent|wparam|lparam|optional, 0, 0 },
1537 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOSIZE|SWP_NOMOVE },
1538 { WM_WINDOWPOSCHANGED, sent|wparam|optional, SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE }, /* Win9x */
1539 { WM_ACTIVATEAPP, sent|wparam, 1 },
1540 { WM_NCACTIVATE, sent|wparam, 1 },
1541 { WM_GETTEXT, sent|defwinproc|optional },
1542 { WM_ACTIVATE, sent|wparam, 1 },
1543 { HCBT_SETFOCUS, hook },
1544 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
1545 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1546 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
1547 /* Win9x adds SWP_NOZORDER below */
1548 { WM_WINDOWPOSCHANGED, sent, /*|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE*/ },
1549 { WM_NCCALCSIZE, sent|wparam|optional, 1 }, /* XP */
1550 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1555 /* DestroyWindow for MDI frame window, initially visible */
1556 static const struct message WmDestroyMDIframeSeq[] = {
1557 { HCBT_DESTROYWND, hook },
1558 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1559 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
1560 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1561 { WM_NCACTIVATE, sent|wparam, 0 },
1562 { WM_ACTIVATE, sent|wparam|optional, 0 }, /* Win9x */
1563 { WM_ACTIVATEAPP, sent|wparam|optional, 0 }, /* Win9x */
1564 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
1565 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
1566 { WM_DESTROY, sent },
1567 { WM_NCDESTROY, sent },
1570 /* CreateWindow for MDI client window, initially visible */
1571 static const struct message WmCreateMDIclientSeq[] = {
1572 { HCBT_CREATEWND, hook },
1573 { WM_NCCREATE, sent },
1574 { WM_NCCALCSIZE, sent|wparam, 0 },
1575 { WM_CREATE, sent },
1576 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1579 { WM_PARENTNOTIFY, sent|wparam, WM_CREATE }, /* in MDI frame */
1580 { WM_SHOWWINDOW, sent|wparam, 1 },
1581 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE },
1582 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1583 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1586 /* ShowWindow(SW_SHOW) for MDI client window */
1587 static const struct message WmShowMDIclientSeq[] = {
1588 { WM_SHOWWINDOW, sent|wparam, 1 },
1589 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE },
1590 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1591 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1594 /* ShowWindow(SW_HIDE) for MDI client window */
1595 static const struct message WmHideMDIclientSeq[] = {
1596 { WM_SHOWWINDOW, sent|wparam, 0 },
1597 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE },
1598 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1599 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOZORDER|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1602 /* DestroyWindow for MDI client window, initially visible */
1603 static const struct message WmDestroyMDIclientSeq[] = {
1604 { HCBT_DESTROYWND, hook },
1605 { WM_PARENTNOTIFY, sent|wparam, WM_DESTROY }, /* in MDI frame */
1606 { WM_SHOWWINDOW, sent|wparam, 0 },
1607 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1608 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
1609 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1610 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
1611 { WM_DESTROY, sent },
1612 { WM_NCDESTROY, sent },
1615 /* CreateWindow for MDI child window, initially visible */
1616 static const struct message WmCreateMDIchildVisibleSeq[] = {
1617 { HCBT_CREATEWND, hook },
1618 { WM_NCCREATE, sent },
1619 { WM_NCCALCSIZE, sent|wparam, 0 },
1620 { WM_CREATE, sent },
1621 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1624 /* Win2k sends wparam set to
1625 * MAKEWPARAM(WM_CREATE, MDI_FIRST_CHILD_ID + nTotalCreated),
1626 * while Win9x doesn't bother to set child window id according to
1627 * CLIENTCREATESTRUCT.idFirstChild
1629 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
1630 { WM_SHOWWINDOW, sent|wparam, 1 },
1631 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1632 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1633 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1634 { WM_MDIREFRESHMENU, sent/*|wparam|lparam, 0, 0*/ },
1635 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
1636 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
1637 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
1639 /* Win9x: message sequence terminates here. */
1641 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
1642 { HCBT_SETFOCUS, hook }, /* in MDI client */
1643 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1644 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1645 { WM_SETFOCUS, sent }, /* in MDI client */
1646 { HCBT_SETFOCUS, hook },
1647 { WM_KILLFOCUS, sent }, /* in MDI client */
1648 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1649 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
1650 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1651 { WM_SETFOCUS, sent|defwinproc },
1652 { WM_MDIACTIVATE, sent|defwinproc },
1655 /* CreateWindow for MDI child window with invisible parent */
1656 static const struct message WmCreateMDIchildInvisibleParentSeq[] = {
1657 { HCBT_CREATEWND, hook },
1658 { WM_GETMINMAXINFO, sent },
1659 { WM_NCCREATE, sent },
1660 { WM_NCCALCSIZE, sent|wparam, 0 },
1661 { WM_CREATE, sent },
1662 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1665 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
1666 { WM_SHOWWINDOW, sent|wparam, 1 },
1667 { WM_MDIREFRESHMENU, sent }, /* in MDI client */
1668 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
1669 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
1670 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
1672 /* Win9x: message sequence terminates here. */
1674 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
1675 { HCBT_SETFOCUS, hook }, /* in MDI client */
1676 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1677 { WM_IME_NOTIFY, sent|wparam|optional, 2 }, /* in MDI client */
1678 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1679 { WM_SETFOCUS, sent }, /* in MDI client */
1680 { HCBT_SETFOCUS, hook },
1681 { WM_KILLFOCUS, sent }, /* in MDI client */
1682 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1683 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
1684 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1685 { WM_SETFOCUS, sent|defwinproc },
1686 { WM_MDIACTIVATE, sent|defwinproc },
1689 /* DestroyWindow for MDI child window, initially visible */
1690 static const struct message WmDestroyMDIchildVisibleSeq[] = {
1691 { HCBT_DESTROYWND, hook },
1692 /* Win2k sends wparam set to
1693 * MAKEWPARAM(WM_DESTROY, MDI_FIRST_CHILD_ID + nTotalCreated),
1694 * while Win9x doesn't bother to set child window id according to
1695 * CLIENTCREATESTRUCT.idFirstChild
1697 { WM_PARENTNOTIFY, sent /*|wparam, WM_DESTROY*/ }, /* in MDI client */
1698 { WM_SHOWWINDOW, sent|wparam, 0 },
1699 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1700 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
1701 { WM_ERASEBKGND, sent|parent|optional },
1702 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1704 /* { WM_DESTROY, sent }
1705 * Win9x: message sequence terminates here.
1708 { HCBT_SETFOCUS, hook }, /* set focus to MDI client */
1709 { WM_KILLFOCUS, sent },
1710 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
1711 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1712 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1713 { WM_SETFOCUS, sent }, /* in MDI client */
1715 { HCBT_SETFOCUS, hook }, /* MDI client sets focus back to MDI child */
1716 { WM_KILLFOCUS, sent }, /* in MDI client */
1717 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1718 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
1719 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1720 { WM_SETFOCUS, sent }, /* in MDI client */
1722 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
1724 { HCBT_SETFOCUS, hook }, /* set focus to MDI client */
1725 { WM_KILLFOCUS, sent },
1726 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
1727 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1728 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1729 { WM_SETFOCUS, sent }, /* in MDI client */
1731 { HCBT_SETFOCUS, hook }, /* MDI client sets focus back to MDI child */
1732 { WM_KILLFOCUS, sent }, /* in MDI client */
1733 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1734 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
1735 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1736 { WM_SETFOCUS, sent }, /* in MDI client */
1738 { WM_DESTROY, sent },
1740 { HCBT_SETFOCUS, hook }, /* set focus to MDI client */
1741 { WM_KILLFOCUS, sent },
1742 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
1743 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1744 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1745 { WM_SETFOCUS, sent }, /* in MDI client */
1747 { HCBT_SETFOCUS, hook }, /* MDI client sets focus back to MDI child */
1748 { WM_KILLFOCUS, sent }, /* in MDI client */
1749 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1750 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
1751 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1752 { WM_SETFOCUS, sent }, /* in MDI client */
1754 { WM_NCDESTROY, sent },
1757 /* CreateWindow for MDI child window, initially invisible */
1758 static const struct message WmCreateMDIchildInvisibleSeq[] = {
1759 { HCBT_CREATEWND, hook },
1760 { WM_NCCREATE, sent },
1761 { WM_NCCALCSIZE, sent|wparam, 0 },
1762 { WM_CREATE, sent },
1763 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1766 /* Win2k sends wparam set to
1767 * MAKEWPARAM(WM_CREATE, MDI_FIRST_CHILD_ID + nTotalCreated),
1768 * while Win9x doesn't bother to set child window id according to
1769 * CLIENTCREATESTRUCT.idFirstChild
1771 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
1774 /* DestroyWindow for MDI child window, initially invisible */
1775 static const struct message WmDestroyMDIchildInvisibleSeq[] = {
1776 { HCBT_DESTROYWND, hook },
1777 /* Win2k sends wparam set to
1778 * MAKEWPARAM(WM_DESTROY, MDI_FIRST_CHILD_ID + nTotalCreated),
1779 * while Win9x doesn't bother to set child window id according to
1780 * CLIENTCREATESTRUCT.idFirstChild
1782 { WM_PARENTNOTIFY, sent /*|wparam, WM_DESTROY*/ }, /* in MDI client */
1783 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
1784 { WM_DESTROY, sent },
1785 { WM_NCDESTROY, sent },
1788 /* CreateWindow for the 1st MDI child window, initially visible and maximized */
1789 static const struct message WmCreateMDIchildVisibleMaxSeq1[] = {
1790 { HCBT_CREATEWND, hook },
1791 { WM_NCCREATE, sent },
1792 { WM_NCCALCSIZE, sent|wparam, 0 },
1793 { WM_CREATE, sent },
1794 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1797 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
1798 { WM_GETMINMAXINFO, sent },
1799 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|0x8000 },
1800 { WM_NCCALCSIZE, sent|wparam, 1 },
1801 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
1802 { WM_SIZE, sent|defwinproc },
1804 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1805 { WM_NCCALCSIZE, sent|wparam, 1 },
1806 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1807 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
1808 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
1809 /* Win2k sends wparam set to
1810 * MAKEWPARAM(WM_CREATE, MDI_FIRST_CHILD_ID + nTotalCreated),
1811 * while Win9x doesn't bother to set child window id according to
1812 * CLIENTCREATESTRUCT.idFirstChild
1814 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
1815 { WM_SHOWWINDOW, sent|wparam, 1 },
1816 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1817 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1818 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1819 { WM_MDIREFRESHMENU, sent/*|wparam|lparam, 0, 0*/ },
1820 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
1821 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
1822 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
1824 /* Win9x: message sequence terminates here. */
1826 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
1827 { HCBT_SETFOCUS, hook }, /* in MDI client */
1828 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1829 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1830 { WM_SETFOCUS, sent }, /* in MDI client */
1831 { HCBT_SETFOCUS, hook },
1832 { WM_KILLFOCUS, sent }, /* in MDI client */
1833 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1834 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
1835 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1836 { WM_SETFOCUS, sent|defwinproc },
1837 { WM_MDIACTIVATE, sent|defwinproc },
1839 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1840 { WM_NCCALCSIZE, sent|wparam, 1 },
1841 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1842 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
1845 /* CreateWindow for the 2nd MDI child window, initially visible and maximized */
1846 static const struct message WmCreateMDIchildVisibleMaxSeq2[] = {
1847 /* restore the 1st MDI child */
1848 { WM_SETREDRAW, sent|wparam, 0 },
1849 { HCBT_MINMAX, hook|lparam, 0, SW_SHOWNORMAL },
1850 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|0x8000 },
1851 { WM_NCCALCSIZE, sent|wparam, 1 },
1852 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
1853 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
1854 { WM_SIZE, sent|defwinproc },
1856 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1857 { WM_NCCALCSIZE, sent|wparam, 1 },
1858 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1859 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
1860 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
1861 { WM_SETREDRAW, sent|wparam, 1 }, /* in the 1st MDI child */
1862 /* create the 2nd MDI child */
1863 { HCBT_CREATEWND, hook },
1864 { WM_NCCREATE, sent },
1865 { WM_NCCALCSIZE, sent|wparam, 0 },
1866 { WM_CREATE, sent },
1867 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1870 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
1871 { WM_GETMINMAXINFO, sent },
1872 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|0x8000 },
1873 { WM_NCCALCSIZE, sent|wparam, 1 },
1874 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1875 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
1876 { WM_SIZE, sent|defwinproc },
1878 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1879 { WM_NCCALCSIZE, sent|wparam, 1 },
1880 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1881 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
1882 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
1883 /* Win2k sends wparam set to
1884 * MAKEWPARAM(WM_CREATE, MDI_FIRST_CHILD_ID + nTotalCreated),
1885 * while Win9x doesn't bother to set child window id according to
1886 * CLIENTCREATESTRUCT.idFirstChild
1888 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
1889 { WM_SHOWWINDOW, sent|wparam, 1 },
1890 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1891 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1892 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1893 { WM_MDIREFRESHMENU, sent/*|wparam|lparam, 0, 0*/ },
1894 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
1895 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
1897 { WM_NCACTIVATE, sent|wparam|defwinproc, 0 }, /* in the 1st MDI child */
1898 { WM_MDIACTIVATE, sent|defwinproc }, /* in the 1st MDI child */
1900 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
1902 /* Win9x: message sequence terminates here. */
1904 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
1905 { HCBT_SETFOCUS, hook },
1906 { WM_KILLFOCUS, sent|defwinproc }, /* in the 1st MDI child */
1907 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 0 }, /* in the 1st MDI child */
1908 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1909 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1910 { WM_SETFOCUS, sent }, /* in MDI client */
1911 { HCBT_SETFOCUS, hook },
1912 { WM_KILLFOCUS, sent }, /* in MDI client */
1913 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1914 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
1915 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1916 { WM_SETFOCUS, sent|defwinproc },
1918 { WM_MDIACTIVATE, sent|defwinproc },
1920 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1921 { WM_NCCALCSIZE, sent|wparam, 1 },
1922 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1923 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
1926 /* WM_MDICREATE MDI child window, initially visible and maximized */
1927 static const struct message WmCreateMDIchildVisibleMaxSeq3[] = {
1928 { WM_MDICREATE, sent },
1929 { HCBT_CREATEWND, hook },
1930 { WM_NCCREATE, sent },
1931 { WM_NCCALCSIZE, sent|wparam, 0 },
1932 { WM_CREATE, sent },
1933 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
1936 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
1937 { WM_GETMINMAXINFO, sent },
1938 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|0x8000 },
1939 { WM_NCCALCSIZE, sent|wparam, 1 },
1940 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
1941 { WM_SIZE, sent|defwinproc },
1944 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1945 { WM_NCCALCSIZE, sent|wparam, 1 },
1946 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1947 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
1948 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
1950 /* Win2k sends wparam set to
1951 * MAKEWPARAM(WM_CREATE, MDI_FIRST_CHILD_ID + nTotalCreated),
1952 * while Win9x doesn't bother to set child window id according to
1953 * CLIENTCREATESTRUCT.idFirstChild
1955 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
1956 { WM_SHOWWINDOW, sent|wparam, 1 },
1957 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1959 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
1961 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1962 { WM_MDIREFRESHMENU, sent/*|wparam|lparam, 0, 0*/ },
1963 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE },
1965 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
1966 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
1968 /* Win9x: message sequence terminates here. */
1970 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
1971 { WM_SETFOCUS, sent|optional }, /* in MDI client */
1972 { HCBT_SETFOCUS, hook }, /* in MDI client */
1973 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
1974 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1975 { WM_SETFOCUS, sent|optional }, /* in MDI client */
1976 { HCBT_SETFOCUS, hook|optional },
1977 { WM_KILLFOCUS, sent }, /* in MDI client */
1978 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
1979 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
1980 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
1981 { WM_SETFOCUS, sent|defwinproc },
1983 { WM_MDIACTIVATE, sent|defwinproc },
1986 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1987 { WM_NCCALCSIZE, sent|wparam, 1 },
1988 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
1989 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 },
1992 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1993 { WM_NCCALCSIZE, sent|wparam, 1 },
1994 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
1995 { WM_MOVE, sent|defwinproc },
1996 { WM_SIZE, sent|defwinproc },
1999 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2000 { WM_NCCALCSIZE, sent|wparam, 1 },
2001 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
2005 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2006 { WM_NCCALCSIZE, sent|wparam, 1 },
2007 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
2008 { WM_SIZE, sent|defwinproc },
2010 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2011 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI client */
2012 { WM_NCCALCSIZE, sent|wparam|optional, 1 }, /* XP sends it to MDI frame */
2013 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2014 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* XP sends a duplicate */
2018 /* CreateWindow for the 1st MDI child window, initially invisible and maximized */
2019 static const struct message WmCreateMDIchildInvisibleMaxSeq4[] = {
2020 { HCBT_CREATEWND, hook },
2021 { WM_GETMINMAXINFO, sent },
2022 { WM_NCCREATE, sent },
2023 { WM_NCCALCSIZE, sent|wparam, 0 },
2024 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, 0, 0 },
2025 { WM_CREATE, sent },
2026 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, 0, 0 },
2029 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
2030 { WM_GETMINMAXINFO, sent },
2031 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|0x8000 },
2032 { WM_GETMINMAXINFO, sent|defwinproc },
2033 { WM_NCCALCSIZE, sent|wparam, 1 },
2034 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|0x8000 },
2035 { WM_MOVE, sent|defwinproc },
2036 { WM_SIZE, sent|defwinproc },
2038 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2039 { WM_NCCALCSIZE, sent|wparam, 1 },
2040 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2041 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2042 { WM_NCCALCSIZE, sent|wparam, 1 }, /* MDI child */
2043 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2044 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2045 /* Win2k sends wparam set to
2046 * MAKEWPARAM(WM_CREATE, MDI_FIRST_CHILD_ID + nTotalCreated),
2047 * while Win9x doesn't bother to set child window id according to
2048 * CLIENTCREATESTRUCT.idFirstChild
2050 { WM_PARENTNOTIFY, sent /*|wparam, WM_CREATE*/ }, /* in MDI client */
2053 /* WM_SYSCOMMAND/SC_CLOSE for the 2nd MDI child window, initially visible and maximized */
2054 static const struct message WmDestroyMDIchildVisibleMaxSeq2[] = {
2055 { WM_SYSCOMMAND, sent|wparam, SC_CLOSE },
2056 { HCBT_SYSCOMMAND, hook },
2057 { WM_CLOSE, sent|defwinproc },
2058 { WM_MDIDESTROY, sent }, /* in MDI client */
2060 /* bring the 1st MDI child to top */
2061 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOSIZE|SWP_NOMOVE }, /* in the 1st MDI child */
2062 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE }, /* in the 2nd MDI child */
2064 { EVENT_OBJECT_REORDER, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2066 { WM_CHILDACTIVATE, sent|defwinproc|wparam|lparam, 0, 0 }, /* in the 1st MDI child */
2067 { WM_NCACTIVATE, sent|wparam|defwinproc, 0 }, /* in the 1st MDI child */
2068 { WM_MDIACTIVATE, sent|defwinproc }, /* in the 1st MDI child */
2070 /* maximize the 1st MDI child */
2071 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
2072 { WM_GETMINMAXINFO, sent|defwinproc },
2073 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_FRAMECHANGED|0x8000 },
2074 { WM_NCCALCSIZE, sent|defwinproc|wparam, 1 },
2075 { WM_CHILDACTIVATE, sent|defwinproc|wparam|lparam, 0, 0 },
2076 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
2077 { WM_SIZE, sent|defwinproc },
2079 /* restore the 2nd MDI child */
2080 { WM_SETREDRAW, sent|defwinproc|wparam, 0 },
2081 { HCBT_MINMAX, hook|lparam, 0, SW_NORMALNA },
2082 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_FRAMECHANGED|SWP_SHOWWINDOW|SWP_NOZORDER|0x8000 },
2083 { WM_NCCALCSIZE, sent|defwinproc|wparam, 1 },
2085 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2087 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_FRAMECHANGED|SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
2088 { WM_SIZE, sent|defwinproc },
2090 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2092 { WM_SETREDRAW, sent|defwinproc|wparam, 1 },
2094 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2095 { WM_NCCALCSIZE, sent|wparam, 1 },
2096 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2097 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2098 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2100 /* bring the 1st MDI child to top */
2101 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
2102 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
2103 { HCBT_SETFOCUS, hook },
2104 { WM_KILLFOCUS, sent|defwinproc },
2105 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 0 },
2106 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
2107 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2108 { WM_SETFOCUS, sent }, /* in MDI client */
2109 { HCBT_SETFOCUS, hook },
2110 { WM_KILLFOCUS, sent }, /* in MDI client */
2111 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
2112 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
2113 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2114 { WM_SETFOCUS, sent|defwinproc },
2115 { WM_MDIACTIVATE, sent|defwinproc },
2116 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_NOSIZE|SWP_NOMOVE|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2118 /* apparently ShowWindow(SW_SHOW) on an MDI client */
2119 { WM_SHOWWINDOW, sent|wparam, 1 },
2120 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2121 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2122 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2123 { WM_MDIREFRESHMENU, sent },
2125 { HCBT_DESTROYWND, hook },
2126 /* Win2k sends wparam set to
2127 * MAKEWPARAM(WM_DESTROY, MDI_FIRST_CHILD_ID + nTotalCreated),
2128 * while Win9x doesn't bother to set child window id according to
2129 * CLIENTCREATESTRUCT.idFirstChild
2131 { WM_PARENTNOTIFY, sent /*|wparam, WM_DESTROY*/ }, /* in MDI client */
2132 { WM_SHOWWINDOW, sent|defwinproc|wparam, 0 },
2133 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2134 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
2135 { WM_ERASEBKGND, sent|parent|optional },
2136 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2138 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
2139 { WM_DESTROY, sent|defwinproc },
2140 { WM_NCDESTROY, sent|defwinproc },
2143 /* WM_MDIDESTROY for the single MDI child window, initially visible and maximized */
2144 static const struct message WmDestroyMDIchildVisibleMaxSeq1[] = {
2145 { WM_MDIDESTROY, sent }, /* in MDI client */
2146 { WM_SHOWWINDOW, sent|wparam, 0 },
2147 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2148 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
2149 { WM_ERASEBKGND, sent|parent|optional },
2150 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2152 { HCBT_SETFOCUS, hook },
2153 { WM_KILLFOCUS, sent },
2154 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
2155 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
2156 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2157 { WM_SETFOCUS, sent }, /* in MDI client */
2158 { HCBT_SETFOCUS, hook },
2159 { WM_KILLFOCUS, sent }, /* in MDI client */
2160 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
2161 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
2162 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2163 { WM_SETFOCUS, sent },
2166 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2167 { WM_NCCALCSIZE, sent|wparam, 1 },
2168 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2169 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2172 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2173 { WM_NCCALCSIZE, sent|wparam, 1 },
2174 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2175 { WM_MOVE, sent|defwinproc },
2176 { WM_SIZE, sent|defwinproc },
2179 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2180 { WM_NCCALCSIZE, sent|wparam, 1 },
2181 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
2185 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2186 { WM_NCCALCSIZE, sent|wparam, 1 },
2187 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTMOVE },
2188 { WM_SIZE, sent|defwinproc },
2191 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2192 { WM_NCCALCSIZE, sent|wparam|defwinproc, 1 },
2193 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2194 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2197 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2198 { WM_NCCALCSIZE, sent|wparam|defwinproc, 1 },
2199 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2200 { WM_MOVE, sent|defwinproc },
2201 { WM_SIZE, sent|defwinproc },
2204 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2205 { WM_NCCALCSIZE, sent|wparam, 1 },
2206 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
2210 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOZORDER },
2211 { WM_NCCALCSIZE, sent|wparam|defwinproc, 1 },
2212 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTMOVE },
2213 { WM_SIZE, sent|defwinproc },
2214 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2215 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI client */
2217 { WM_NCCALCSIZE, sent|wparam|defwinproc|optional, 1 }, /* XP sends it to MDI frame */
2219 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2220 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2221 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI client */
2222 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2223 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* XP sends a duplicate */
2226 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2227 { WM_NCCALCSIZE, sent|wparam, 1 },
2228 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2229 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2231 { WM_NCACTIVATE, sent|wparam, 0 },
2232 { WM_MDIACTIVATE, sent },
2234 { HCBT_MINMAX, hook|lparam, 0, SW_SHOWNORMAL },
2235 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_SHOWWINDOW|0x8000 },
2236 { WM_NCCALCSIZE, sent|wparam, 1 },
2238 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2240 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2241 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_SHOWWINDOW|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE|0x8000 },
2242 { WM_SIZE, sent|defwinproc },
2245 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2246 { WM_NCCALCSIZE, sent|wparam|defwinproc, 1 },
2247 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2248 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2251 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2252 { WM_NCCALCSIZE, sent|wparam, 1 },
2253 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2254 { WM_MOVE, sent|defwinproc },
2255 { WM_SIZE, sent|defwinproc },
2258 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2259 { WM_NCCALCSIZE, sent|wparam, 1 },
2260 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTMOVE },
2262 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2263 { WM_NCCALCSIZE, sent|wparam|optional, 1 }, /* XP */
2264 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI client */
2265 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2266 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* XP sends a duplicate */
2268 { HCBT_SETFOCUS, hook },
2269 { WM_KILLFOCUS, sent },
2270 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
2271 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
2272 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2273 { WM_SETFOCUS, sent }, /* in MDI client */
2275 { WM_MDIREFRESHMENU, sent }, /* in MDI client */
2277 { HCBT_DESTROYWND, hook },
2278 /* Win2k sends wparam set to
2279 * MAKEWPARAM(WM_DESTROY, MDI_FIRST_CHILD_ID + nTotalCreated),
2280 * while Win9x doesn't bother to set child window id according to
2281 * CLIENTCREATESTRUCT.idFirstChild
2283 { WM_PARENTNOTIFY, sent /*|wparam, WM_DESTROY*/ }, /* in MDI client */
2285 { WM_SHOWWINDOW, sent|wparam, 0 },
2286 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2287 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, 0, 0 },
2288 { WM_ERASEBKGND, sent|parent|optional },
2289 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_HIDEWINDOW|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2291 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 },
2292 { WM_DESTROY, sent },
2293 { WM_NCDESTROY, sent },
2296 /* ShowWindow(SW_MAXIMIZE) for a not visible MDI child window */
2297 static const struct message WmMaximizeMDIchildInvisibleSeq[] = {
2298 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
2299 { WM_GETMINMAXINFO, sent },
2300 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|0x8000 },
2301 { WM_NCCALCSIZE, sent|wparam, 1 },
2302 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2303 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2305 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
2306 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
2307 { HCBT_SETFOCUS, hook },
2308 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
2309 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2310 { WM_SETFOCUS, sent }, /* in MDI client */
2311 { HCBT_SETFOCUS, hook },
2312 { WM_KILLFOCUS, sent }, /* in MDI client */
2313 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
2314 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
2315 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2316 { WM_SETFOCUS, sent|defwinproc },
2317 { WM_MDIACTIVATE, sent|defwinproc },
2318 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
2319 { WM_SIZE, sent|defwinproc },
2321 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2322 { WM_NCCALCSIZE, sent|wparam, 1 },
2323 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2324 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2325 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2328 /* ShowWindow(SW_MAXIMIZE) for a not visible maximized MDI child window */
2329 static const struct message WmMaximizeMDIchildInvisibleSeq2[] = {
2330 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
2331 { WM_GETMINMAXINFO, sent },
2332 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED },
2333 { WM_GETMINMAXINFO, sent|defwinproc },
2334 { WM_NCCALCSIZE, sent|wparam, 1 },
2335 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2336 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2338 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE },
2339 { WM_NCACTIVATE, sent|wparam|defwinproc, 1 },
2340 { HCBT_SETFOCUS, hook },
2341 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 }, /* in MDI client */
2342 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2343 { WM_SETFOCUS, sent }, /* in MDI client */
2344 { HCBT_SETFOCUS, hook },
2345 { WM_KILLFOCUS, sent }, /* in MDI client */
2346 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 }, /* in MDI client */
2347 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
2348 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
2349 { WM_SETFOCUS, sent|defwinproc },
2350 { WM_MDIACTIVATE, sent|defwinproc },
2351 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2352 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2355 /* WM_MDIMAXIMIZE for an MDI child window with invisible parent */
2356 static const struct message WmMaximizeMDIchildInvisibleParentSeq[] = {
2357 { WM_MDIMAXIMIZE, sent }, /* in MDI client */
2358 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
2359 { WM_GETMINMAXINFO, sent },
2360 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|0x8000 },
2361 { WM_GETMINMAXINFO, sent|defwinproc },
2362 { WM_NCCALCSIZE, sent|wparam, 1 },
2363 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2364 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2365 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOREDRAW|SWP_NOZORDER|0x8000 },
2366 { WM_MOVE, sent|defwinproc },
2367 { WM_SIZE, sent|defwinproc|wparam, SIZE_MAXIMIZED },
2369 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2370 { WM_NCCALCSIZE, sent|wparam|defwinproc, 1 },
2371 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2372 { WM_NCCALCSIZE, sent|wparam|defwinproc|optional, 1 },
2374 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2375 { WM_NCCALCSIZE, sent|wparam, 1 },
2376 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2377 { WM_MOVE, sent|defwinproc },
2378 { WM_SIZE, sent|defwinproc },
2379 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2381 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_NOACTIVATE|SWP_NOZORDER },
2382 { WM_NCCALCSIZE, sent|wparam, 1 },
2383 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTMOVE },
2386 { WM_WINDOWPOSCHANGING, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOZORDER },
2387 { WM_GETMINMAXINFO, sent|defwinproc },
2388 { WM_NCCALCSIZE, sent|wparam|defwinproc, 1 },
2389 { WM_WINDOWPOSCHANGED, sent|wparam|defwinproc, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOREDRAW|SWP_NOCLIENTMOVE },
2390 { WM_SIZE, sent|defwinproc|wparam, SIZE_MAXIMIZED },
2391 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2392 { WM_NCCALCSIZE, sent|wparam|defwinproc|optional, 1 },
2394 { WM_NCCALCSIZE, sent|wparam|optional, 1 },
2397 /* ShowWindow(SW_MAXIMIZE) for a visible MDI child window */
2398 static const struct message WmMaximizeMDIchildVisibleSeq[] = {
2399 { HCBT_MINMAX, hook|lparam, 0, SW_MAXIMIZE },
2400 { WM_GETMINMAXINFO, sent },
2401 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|0x8000 },
2402 { WM_NCCALCSIZE, sent|wparam, 1 },
2403 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2404 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
2405 { WM_SIZE, sent|defwinproc },
2407 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2408 { WM_NCCALCSIZE, sent|wparam, 1 },
2409 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2410 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2411 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2414 /* ShowWindow(SW_RESTORE) for a visible MDI child window */
2415 static const struct message WmRestoreMDIchildVisibleSeq[] = {
2416 { HCBT_MINMAX, hook|lparam, 0, SW_RESTORE },
2417 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|0x8000 },
2418 { WM_NCCALCSIZE, sent|wparam, 1 },
2419 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2420 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
2421 { WM_SIZE, sent|defwinproc },
2423 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2424 { WM_NCCALCSIZE, sent|wparam, 1 },
2425 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2426 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2427 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2430 /* ShowWindow(SW_RESTORE) for a not visible MDI child window */
2431 static const struct message WmRestoreMDIchildInisibleSeq[] = {
2432 { HCBT_MINMAX, hook|lparam, 0, SW_RESTORE },
2433 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|0x8000 },
2434 { WM_NCCALCSIZE, sent|wparam, 1 },
2435 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, 0, 0 },
2436 { WM_CHILDACTIVATE, sent|wparam|lparam, 0, 0 },
2437 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_SHOWWINDOW|SWP_FRAMECHANGED|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE|0x8000 },
2438 { WM_SIZE, sent|defwinproc },
2440 { WM_WINDOWPOSCHANGING, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER },
2441 { WM_NCCALCSIZE, sent|wparam, 1 },
2442 { WM_WINDOWPOSCHANGED, sent|wparam, SWP_FRAMECHANGED|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_NOZORDER|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE },
2443 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI frame */
2444 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, 0, 0 }, /* MDI child */
2448 static HWND mdi_client;
2449 static WNDPROC old_mdi_client_proc;
2451 static LRESULT WINAPI mdi_client_hook_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
2455 /* do not log painting messages */
2456 if (message != WM_PAINT &&
2457 message != WM_ERASEBKGND &&
2458 message != WM_NCPAINT &&
2459 message != WM_NCHITTEST &&
2460 message != WM_GETTEXT &&
2461 message != WM_MDIGETACTIVE &&
2462 message != WM_GETICON &&
2463 message != WM_DEVICECHANGE)
2465 trace("mdi client: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
2469 case WM_WINDOWPOSCHANGING:
2470 case WM_WINDOWPOSCHANGED:
2472 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
2474 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
2475 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
2476 winpos->hwnd, winpos->hwndInsertAfter,
2477 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
2479 /* Log only documented flags, win2k uses 0x1000 and 0x2000
2480 * in the high word for internal purposes
2482 wParam = winpos->flags & 0xffff;
2487 msg.message = message;
2488 msg.flags = sent|wparam|lparam;
2489 msg.wParam = wParam;
2490 msg.lParam = lParam;
2494 return CallWindowProcA(old_mdi_client_proc, hwnd, message, wParam, lParam);
2497 static LRESULT WINAPI mdi_child_wnd_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
2499 static long defwndproc_counter = 0;
2503 /* do not log painting messages */
2504 if (message != WM_PAINT &&
2505 message != WM_ERASEBKGND &&
2506 message != WM_NCPAINT &&
2507 message != WM_NCHITTEST &&
2508 message != WM_GETTEXT &&
2509 message != WM_GETICON &&
2510 message != WM_DEVICECHANGE)
2512 trace("mdi child: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
2516 case WM_WINDOWPOSCHANGING:
2517 case WM_WINDOWPOSCHANGED:
2519 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
2521 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
2522 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
2523 winpos->hwnd, winpos->hwndInsertAfter,
2524 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
2526 /* Log only documented flags, win2k uses 0x1000 and 0x2000
2527 * in the high word for internal purposes
2529 wParam = winpos->flags & 0xffff;
2533 case WM_MDIACTIVATE:
2535 HWND active, client = GetParent(hwnd);
2537 active = (HWND)SendMessageA(client, WM_MDIGETACTIVE, 0, 0);
2539 if (hwnd == (HWND)lParam) /* if we are being activated */
2540 ok (active == (HWND)lParam, "new active %p != active %p\n", (HWND)lParam, active);
2542 ok (active == (HWND)wParam, "old active %p != active %p\n", (HWND)wParam, active);
2547 msg.message = message;
2548 msg.flags = sent|wparam|lparam;
2549 if (defwndproc_counter) msg.flags |= defwinproc;
2550 msg.wParam = wParam;
2551 msg.lParam = lParam;
2555 defwndproc_counter++;
2556 ret = DefMDIChildProcA(hwnd, message, wParam, lParam);
2557 defwndproc_counter--;
2562 static LRESULT WINAPI mdi_frame_wnd_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
2564 static long defwndproc_counter = 0;
2568 /* do not log painting messages */
2569 if (message != WM_PAINT &&
2570 message != WM_ERASEBKGND &&
2571 message != WM_NCPAINT &&
2572 message != WM_NCHITTEST &&
2573 message != WM_GETTEXT &&
2574 message != WM_GETICON &&
2575 message != WM_DEVICECHANGE)
2577 trace("mdi frame: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
2581 case WM_WINDOWPOSCHANGING:
2582 case WM_WINDOWPOSCHANGED:
2584 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
2586 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
2587 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
2588 winpos->hwnd, winpos->hwndInsertAfter,
2589 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
2591 /* Log only documented flags, win2k uses 0x1000 and 0x2000
2592 * in the high word for internal purposes
2594 wParam = winpos->flags & 0xffff;
2599 msg.message = message;
2600 msg.flags = sent|wparam|lparam;
2601 if (defwndproc_counter) msg.flags |= defwinproc;
2602 msg.wParam = wParam;
2603 msg.lParam = lParam;
2607 defwndproc_counter++;
2608 ret = DefFrameProcA(hwnd, mdi_client, message, wParam, lParam);
2609 defwndproc_counter--;
2614 static BOOL mdi_RegisterWindowClasses(void)
2619 cls.lpfnWndProc = mdi_frame_wnd_proc;
2622 cls.hInstance = GetModuleHandleA(0);
2624 cls.hCursor = LoadCursorA(0, (LPSTR)IDC_ARROW);
2625 cls.hbrBackground = GetStockObject(WHITE_BRUSH);
2626 cls.lpszMenuName = NULL;
2627 cls.lpszClassName = "MDI_frame_class";
2628 if (!RegisterClassA(&cls)) return FALSE;
2630 cls.lpfnWndProc = mdi_child_wnd_proc;
2631 cls.lpszClassName = "MDI_child_class";
2632 if (!RegisterClassA(&cls)) return FALSE;
2634 if (!GetClassInfoA(0, "MDIClient", &cls)) assert(0);
2635 old_mdi_client_proc = cls.lpfnWndProc;
2636 cls.hInstance = GetModuleHandleA(0);
2637 cls.lpfnWndProc = mdi_client_hook_proc;
2638 cls.lpszClassName = "MDI_client_class";
2639 if (!RegisterClassA(&cls)) assert(0);
2644 static void test_mdi_messages(void)
2646 MDICREATESTRUCTA mdi_cs;
2647 CLIENTCREATESTRUCT client_cs;
2648 HWND mdi_frame, mdi_child, mdi_child2, active_child;
2650 HMENU hMenu = CreateMenu();
2652 assert(mdi_RegisterWindowClasses());
2656 trace("creating MDI frame window\n");
2657 mdi_frame = CreateWindowExA(0, "MDI_frame_class", "MDI frame window",
2658 WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX |
2659 WS_MAXIMIZEBOX | WS_VISIBLE,
2660 100, 100, CW_USEDEFAULT, CW_USEDEFAULT,
2661 GetDesktopWindow(), hMenu,
2662 GetModuleHandleA(0), NULL);
2664 ok_sequence(WmCreateMDIframeSeq, "Create MDI frame window", TRUE);
2666 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2667 ok(GetFocus() == mdi_frame, "wrong focus window %p\n", GetFocus());
2669 trace("creating MDI client window\n");
2670 client_cs.hWindowMenu = 0;
2671 client_cs.idFirstChild = MDI_FIRST_CHILD_ID;
2672 mdi_client = CreateWindowExA(0, "MDI_client_class",
2674 WS_CHILD | WS_VISIBLE | MDIS_ALLCHILDSTYLES,
2676 mdi_frame, 0, GetModuleHandleA(0), &client_cs);
2678 ok_sequence(WmCreateMDIclientSeq, "Create visible MDI client window", FALSE);
2680 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2681 ok(GetFocus() == mdi_frame, "input focus should be on MDI frame not on %p\n", GetFocus());
2683 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2684 ok(!active_child, "wrong active MDI child %p\n", active_child);
2685 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2690 trace("creating invisible MDI child window\n");
2691 mdi_child = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2693 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2694 mdi_client, 0, GetModuleHandleA(0), NULL);
2698 ShowWindow(mdi_child, SW_SHOWNORMAL);
2699 ok_sequence(WmShowChildSeq, "ShowWindow(SW_SHOWNORMAL) MDI child window", FALSE);
2701 ok(GetWindowLongA(mdi_child, GWL_STYLE) & WS_VISIBLE, "MDI child should be visible\n");
2702 ok(IsWindowVisible(mdi_child), "MDI child should be visible\n");
2704 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2705 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2707 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2708 ok(!active_child, "wrong active MDI child %p\n", active_child);
2709 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2711 ShowWindow(mdi_child, SW_HIDE);
2712 ok_sequence(WmHideChildSeq, "ShowWindow(SW_HIDE) MDI child window", FALSE);
2715 ShowWindow(mdi_child, SW_SHOW);
2716 ok_sequence(WmShowChildSeq, "ShowWindow(SW_SHOW) MDI child window", FALSE);
2718 ok(GetWindowLongA(mdi_child, GWL_STYLE) & WS_VISIBLE, "MDI child should be visible\n");
2719 ok(IsWindowVisible(mdi_child), "MDI child should be visible\n");
2721 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2722 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2724 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2725 ok(!active_child, "wrong active MDI child %p\n", active_child);
2726 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2728 DestroyWindow(mdi_child);
2731 trace("creating visible MDI child window\n");
2732 mdi_child = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2733 WS_CHILD | WS_VISIBLE,
2734 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2735 mdi_client, 0, GetModuleHandleA(0), NULL);
2737 ok_sequence(WmCreateMDIchildVisibleSeq, "Create visible MDI child window", FALSE);
2739 ok(GetWindowLongA(mdi_child, GWL_STYLE) & WS_VISIBLE, "MDI child should be visible\n");
2740 ok(IsWindowVisible(mdi_child), "MDI child should be visible\n");
2742 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2743 ok(GetFocus() == mdi_child, "wrong focus window %p\n", GetFocus());
2745 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2746 ok(active_child == mdi_child, "wrong active MDI child %p\n", active_child);
2747 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2750 DestroyWindow(mdi_child);
2751 ok_sequence(WmDestroyMDIchildVisibleSeq, "Destroy visible MDI child window", TRUE);
2753 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2754 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2756 /* Win2k: MDI client still returns a just destroyed child as active
2757 * Win9x: MDI client returns 0
2759 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2760 ok(active_child == mdi_child || /* win2k */
2761 !active_child, /* win9x */
2762 "wrong active MDI child %p\n", active_child);
2763 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2767 trace("creating invisible MDI child window\n");
2768 mdi_child2 = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2770 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2771 mdi_client, 0, GetModuleHandleA(0), NULL);
2773 ok_sequence(WmCreateMDIchildInvisibleSeq, "Create invisible MDI child window", FALSE);
2775 ok(!(GetWindowLongA(mdi_child2, GWL_STYLE) & WS_VISIBLE), "MDI child should not be visible\n");
2776 ok(!IsWindowVisible(mdi_child2), "MDI child should not be visible\n");
2778 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2779 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2781 /* Win2k: MDI client still returns a just destroyed child as active
2782 * Win9x: MDI client returns mdi_child2
2784 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2785 ok(active_child == mdi_child || /* win2k */
2786 active_child == mdi_child2, /* win9x */
2787 "wrong active MDI child %p\n", active_child);
2788 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2791 ShowWindow(mdi_child2, SW_MAXIMIZE);
2792 ok_sequence(WmMaximizeMDIchildInvisibleSeq, "ShowWindow(SW_MAXIMIZE):invisible MDI child", TRUE);
2794 ok(GetWindowLongA(mdi_child2, GWL_STYLE) & WS_VISIBLE, "MDI child should be visible\n");
2795 ok(IsWindowVisible(mdi_child2), "MDI child should be visible\n");
2797 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2798 ok(active_child == mdi_child2, "wrong active MDI child %p\n", active_child);
2799 ok(zoomed, "wrong zoomed state %d\n", zoomed);
2802 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2803 ok(GetFocus() == mdi_child2 || /* win2k */
2804 GetFocus() == 0, /* win9x */
2805 "wrong focus window %p\n", GetFocus());
2810 ShowWindow(mdi_child2, SW_HIDE);
2811 ok_sequence(WmHideChildSeq, "ShowWindow(SW_HIDE):MDI child", FALSE);
2813 ShowWindow(mdi_child2, SW_RESTORE);
2814 ok_sequence(WmRestoreMDIchildInisibleSeq, "ShowWindow(SW_RESTORE):invisible MDI child", TRUE);
2817 ok(GetWindowLongA(mdi_child2, GWL_STYLE) & WS_VISIBLE, "MDI child should be visible\n");
2818 ok(IsWindowVisible(mdi_child2), "MDI child should be visible\n");
2820 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2821 ok(active_child == mdi_child2, "wrong active MDI child %p\n", active_child);
2822 ok(!zoomed, "wrong zoomed state %d\n", zoomed);
2828 ShowWindow(mdi_child2, SW_HIDE);
2829 ok_sequence(WmHideChildSeq, "ShowWindow(SW_HIDE):MDI child", FALSE);
2831 ShowWindow(mdi_child2, SW_SHOW);
2832 ok_sequence(WmShowChildSeq, "ShowWindow(SW_SHOW):MDI child", FALSE);
2834 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2835 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2837 ShowWindow(mdi_child2, SW_MAXIMIZE);
2838 ok_sequence(WmMaximizeMDIchildVisibleSeq, "ShowWindow(SW_MAXIMIZE):MDI child", TRUE);
2840 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2841 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2843 ShowWindow(mdi_child2, SW_RESTORE);
2844 ok_sequence(WmRestoreMDIchildVisibleSeq, "ShowWindow(SW_RESTORE):MDI child", TRUE);
2846 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2847 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2852 ShowWindow(mdi_child2, SW_HIDE);
2853 ok_sequence(WmHideChildSeq, "ShowWindow(SW_HIDE):MDI child", FALSE);
2855 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2856 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2858 DestroyWindow(mdi_child2);
2859 ok_sequence(WmDestroyMDIchildInvisibleSeq, "Destroy invisible MDI child window", FALSE);
2861 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2862 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2864 /* test for maximized MDI children */
2865 trace("creating maximized visible MDI child window 1\n");
2866 mdi_child = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2867 WS_CHILD | WS_VISIBLE | WS_MAXIMIZEBOX | WS_MAXIMIZE,
2868 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2869 mdi_client, 0, GetModuleHandleA(0), NULL);
2871 ok_sequence(WmCreateMDIchildVisibleMaxSeq1, "Create maximized visible 1st MDI child window", TRUE);
2872 ok(IsZoomed(mdi_child), "1st MDI child should be maximized\n");
2874 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2875 ok(GetFocus() == mdi_child || /* win2k */
2876 GetFocus() == 0, /* win9x */
2877 "wrong focus window %p\n", GetFocus());
2879 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2880 ok(active_child == mdi_child, "wrong active MDI child %p\n", active_child);
2881 ok(zoomed, "wrong zoomed state %d\n", zoomed);
2884 trace("creating maximized visible MDI child window 2\n");
2885 mdi_child2 = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2886 WS_CHILD | WS_VISIBLE | WS_MAXIMIZEBOX | WS_MAXIMIZE,
2887 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2888 mdi_client, 0, GetModuleHandleA(0), NULL);
2890 ok_sequence(WmCreateMDIchildVisibleMaxSeq2, "Create maximized visible 2nd MDI child 2 window", TRUE);
2891 ok(IsZoomed(mdi_child2), "2nd MDI child should be maximized\n");
2892 ok(!IsZoomed(mdi_child), "1st MDI child should NOT be maximized\n");
2894 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2895 ok(GetFocus() == mdi_child2, "wrong focus window %p\n", GetFocus());
2897 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2898 ok(active_child == mdi_child2, "wrong active MDI child %p\n", active_child);
2899 ok(zoomed, "wrong zoomed state %d\n", zoomed);
2902 trace("destroying maximized visible MDI child window 2\n");
2903 DestroyWindow(mdi_child2);
2904 ok_sequence(WmDestroyMDIchildVisibleSeq, "Destroy visible MDI child window", TRUE);
2906 ok(!IsZoomed(mdi_child), "1st MDI child should NOT be maximized\n");
2908 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2909 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2911 /* Win2k: MDI client still returns a just destroyed child as active
2912 * Win9x: MDI client returns 0
2914 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2915 ok(active_child == mdi_child2 || /* win2k */
2916 !active_child, /* win9x */
2917 "wrong active MDI child %p\n", active_child);
2920 ShowWindow(mdi_child, SW_MAXIMIZE);
2921 ok(IsZoomed(mdi_child), "1st MDI child should be maximized\n");
2924 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2925 ok(GetFocus() == mdi_child, "wrong focus window %p\n", GetFocus());
2927 trace("re-creating maximized visible MDI child window 2\n");
2928 mdi_child2 = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2929 WS_CHILD | WS_VISIBLE | WS_MAXIMIZEBOX | WS_MAXIMIZE,
2930 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2931 mdi_client, 0, GetModuleHandleA(0), NULL);
2933 ok_sequence(WmCreateMDIchildVisibleMaxSeq2, "Create maximized visible 2nd MDI child 2 window", TRUE);
2934 ok(IsZoomed(mdi_child2), "2nd MDI child should be maximized\n");
2935 ok(!IsZoomed(mdi_child), "1st MDI child should NOT be maximized\n");
2937 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2938 ok(GetFocus() == mdi_child2, "wrong focus window %p\n", GetFocus());
2940 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2941 ok(active_child == mdi_child2, "wrong active MDI child %p\n", active_child);
2942 ok(zoomed, "wrong zoomed state %d\n", zoomed);
2945 SendMessageA(mdi_child2, WM_SYSCOMMAND, SC_CLOSE, 0);
2946 ok_sequence(WmDestroyMDIchildVisibleMaxSeq2, "WM_SYSCOMMAND/SC_CLOSE on a visible maximized MDI child window", TRUE);
2947 ok(!IsWindow(mdi_child2), "MDI child 2 should be destroyed\n");
2949 ok(IsZoomed(mdi_child), "1st MDI child should be maximized\n");
2950 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2951 ok(GetFocus() == mdi_child, "wrong focus window %p\n", GetFocus());
2953 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2954 ok(active_child == mdi_child, "wrong active MDI child %p\n", active_child);
2955 ok(zoomed, "wrong zoomed state %d\n", zoomed);
2958 DestroyWindow(mdi_child);
2959 ok_sequence(WmDestroyMDIchildVisibleSeq, "Destroy visible MDI child window", TRUE);
2961 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
2962 ok(GetFocus() == 0, "wrong focus window %p\n", GetFocus());
2964 /* Win2k: MDI client still returns a just destroyed child as active
2965 * Win9x: MDI client returns 0
2967 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2968 ok(active_child == mdi_child || /* win2k */
2969 !active_child, /* win9x */
2970 "wrong active MDI child %p\n", active_child);
2973 trace("creating maximized invisible MDI child window\n");
2974 mdi_child2 = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
2975 WS_CHILD | WS_MAXIMIZE | WS_CAPTION | WS_THICKFRAME,
2976 0, 0, CW_USEDEFAULT, CW_USEDEFAULT,
2977 mdi_client, 0, GetModuleHandleA(0), NULL);
2979 ok_sequence(WmCreateMDIchildInvisibleMaxSeq4, "Create maximized invisible MDI child window", TRUE);
2980 ok(IsZoomed(mdi_child2), "MDI child should be maximized\n");
2981 ok(!(GetWindowLongA(mdi_child2, GWL_STYLE) & WS_VISIBLE), "MDI child should be not visible\n");
2982 ok(!IsWindowVisible(mdi_child2), "MDI child should be not visible\n");
2984 /* Win2k: MDI client still returns a just destroyed child as active
2985 * Win9x: MDI client returns 0
2987 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
2988 ok(active_child == mdi_child || /* win2k */
2989 !active_child, /* win9x */
2990 "wrong active MDI child %p\n", active_child);
2993 trace("call ShowWindow(mdi_child, SW_MAXIMIZE)\n");
2994 ShowWindow(mdi_child2, SW_MAXIMIZE);
2995 ok_sequence(WmMaximizeMDIchildInvisibleSeq2, "ShowWindow(SW_MAXIMIZE):invisible maximized MDI child", TRUE);
2996 ok(IsZoomed(mdi_child2), "MDI child should be maximized\n");
2997 ok(GetWindowLongA(mdi_child2, GWL_STYLE) & WS_VISIBLE, "MDI child should be visible\n");
2998 ok(IsWindowVisible(mdi_child2), "MDI child should be visible\n");
3000 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
3001 ok(active_child == mdi_child2, "wrong active MDI child %p\n", active_child);
3002 ok(zoomed, "wrong zoomed state %d\n", zoomed);
3005 SendMessageA(mdi_client, WM_MDIDESTROY, (WPARAM)mdi_child2, 0);
3008 /* end of test for maximized MDI children */
3010 mdi_cs.szClass = "MDI_child_Class";
3011 mdi_cs.szTitle = "MDI child";
3012 mdi_cs.hOwner = GetModuleHandleA(0);
3015 mdi_cs.cx = CW_USEDEFAULT;
3016 mdi_cs.cy = CW_USEDEFAULT;
3017 mdi_cs.style = WS_CHILD | WS_SYSMENU | WS_VISIBLE | WS_MAXIMIZEBOX | WS_MAXIMIZE;
3019 mdi_child = (HWND)SendMessageA(mdi_client, WM_MDICREATE, 0, (LPARAM)&mdi_cs);
3020 ok(mdi_child != 0, "MDI child creation failed\n");
3021 ok_sequence(WmCreateMDIchildVisibleMaxSeq3, "WM_MDICREATE for maximized visible MDI child window", TRUE);
3023 ok(GetMenuItemID(hMenu, GetMenuItemCount(hMenu) - 1) == SC_CLOSE, "SC_CLOSE menu item not found\n");
3025 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
3026 ok(active_child == mdi_child, "wrong active MDI child %p\n", active_child);
3028 ok(IsZoomed(mdi_child), "MDI child should be maximized\n");
3029 ok(GetActiveWindow() == mdi_frame, "wrong active window %p\n", GetActiveWindow());
3030 ok(GetFocus() == mdi_child, "wrong focus window %p\n", GetFocus());
3032 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
3033 ok(active_child == mdi_child, "wrong active MDI child %p\n", active_child);
3034 ok(zoomed, "wrong zoomed state %d\n", zoomed);
3037 SendMessageA(mdi_client, WM_MDIDESTROY, (WPARAM)mdi_child, 0);
3038 ok_sequence(WmDestroyMDIchildVisibleMaxSeq1, "Destroy visible maximized MDI child window", TRUE);
3040 ok(!IsWindow(mdi_child), "MDI child should be destroyed\n");
3041 active_child = (HWND)SendMessageA(mdi_client, WM_MDIGETACTIVE, 0, (LPARAM)&zoomed);
3042 ok(!active_child, "wrong active MDI child %p\n", active_child);
3047 DestroyWindow(mdi_client);
3048 ok_sequence(WmDestroyMDIclientSeq, "Destroy MDI client window", FALSE);
3050 /* test maximization of MDI child with invisible parent */
3051 client_cs.hWindowMenu = 0;
3052 mdi_client = CreateWindow("MDI_client_class",
3054 WS_CHILD | WS_CLIPCHILDREN | WS_VSCROLL | WS_HSCROLL | WS_VISIBLE,
3056 mdi_frame, 0, GetModuleHandleA(0), &client_cs);
3057 ok_sequence(WmCreateMDIclientSeq, "Create MDI client window", FALSE);
3059 ShowWindow(mdi_client, SW_HIDE);
3060 ok_sequence(WmHideMDIclientSeq, "Hide MDI client window", FALSE);
3062 mdi_child = CreateWindowExA(WS_EX_MDICHILD, "MDI_child_class", "MDI child",
3063 WS_CHILD | WS_CLIPCHILDREN | WS_VSCROLL | WS_HSCROLL,
3065 mdi_client, 0, GetModuleHandleA(0), NULL);
3066 ok_sequence(WmCreateMDIchildInvisibleParentSeq, "Create MDI child window with invisible parent", FALSE);
3068 SendMessage(mdi_client, WM_MDIMAXIMIZE, (WPARAM) mdi_child, 0);
3069 ok_sequence(WmMaximizeMDIchildInvisibleParentSeq, "Maximize MDI child window with invisible parent", TRUE);
3070 zoomed = IsZoomed(mdi_child);
3071 ok(zoomed, "wrong zoomed state %d\n", zoomed);
3073 ShowWindow(mdi_client, SW_SHOW);
3074 ok_sequence(WmShowMDIclientSeq, "Show MDI client window", FALSE);
3076 DestroyWindow(mdi_child);
3077 ok_sequence(WmDestroyMDIchildVisibleSeq, "Destroy visible maximized MDI child window", TRUE);
3079 DestroyWindow(mdi_client);
3080 ok_sequence(WmDestroyMDIclientSeq, "Destroy MDI client window", FALSE);
3081 /* end of test for maximization of MDI child with invisible parent */
3083 DestroyWindow(mdi_frame);
3084 ok_sequence(WmDestroyMDIframeSeq, "Destroy MDI frame window", FALSE);
3086 /************************* End of MDI test **********************************/
3088 static void test_WM_SETREDRAW(HWND hwnd)
3090 DWORD style = GetWindowLongA(hwnd, GWL_STYLE);
3094 SendMessageA(hwnd, WM_SETREDRAW, FALSE, 0);
3095 ok_sequence(WmSetRedrawFalseSeq, "SetRedraw:FALSE", FALSE);
3097 ok(!(GetWindowLongA(hwnd, GWL_STYLE) & WS_VISIBLE), "WS_VISIBLE should NOT be set\n");
3098 ok(!IsWindowVisible(hwnd), "IsWindowVisible() should return FALSE\n");
3101 SendMessageA(hwnd, WM_SETREDRAW, TRUE, 0);
3102 ok_sequence(WmSetRedrawTrueSeq, "SetRedraw:TRUE", FALSE);
3104 ok(GetWindowLongA(hwnd, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3105 ok(IsWindowVisible(hwnd), "IsWindowVisible() should return TRUE\n");
3107 /* restore original WS_VISIBLE state */
3108 SetWindowLongA(hwnd, GWL_STYLE, style);
3113 static INT_PTR CALLBACK TestModalDlgProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
3117 trace("dialog: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
3119 /* explicitly ignore WM_GETICON message */
3120 if (message == WM_GETICON) return 0;
3124 case WM_WINDOWPOSCHANGING:
3125 case WM_WINDOWPOSCHANGED:
3127 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
3129 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
3130 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
3131 winpos->hwnd, winpos->hwndInsertAfter,
3132 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
3134 /* Log only documented flags, win2k uses 0x1000 and 0x2000
3135 * in the high word for internal purposes
3137 wParam = winpos->flags & 0xffff;
3142 msg.message = message;
3143 msg.flags = sent|wparam|lparam;
3144 msg.wParam = wParam;
3145 msg.lParam = lParam;
3148 if (message == WM_INITDIALOG) SetTimer( hwnd, 1, 100, NULL );
3149 if (message == WM_TIMER) EndDialog( hwnd, 0 );
3153 static void test_hv_scroll_1(HWND hwnd, INT ctl, DWORD clear, DWORD set, INT min, INT max)
3155 DWORD style, exstyle;
3159 exstyle = GetWindowLongA(hwnd, GWL_EXSTYLE);
3160 style = GetWindowLongA(hwnd, GWL_STYLE);
3161 /* do not be confused by WS_DLGFRAME set */
3162 if ((style & WS_CAPTION) == WS_CAPTION) style &= ~WS_CAPTION;
3164 if (clear) ok(style & clear, "style %08x should be set\n", clear);
3165 if (set) ok(!(style & set), "style %08x should not be set\n", set);
3167 ret = SetScrollRange(hwnd, ctl, min, max, FALSE);
3168 ok( ret, "SetScrollRange(%d) error %d\n", ctl, GetLastError());
3169 if ((style & (WS_DLGFRAME | WS_BORDER | WS_THICKFRAME)) || (exstyle & WS_EX_DLGMODALFRAME))
3170 ok_sequence(WmSetScrollRangeHV_NC_Seq, "SetScrollRange(SB_HORZ/SB_VERT) NC", FALSE);
3172 ok_sequence(WmSetScrollRangeHVSeq, "SetScrollRange(SB_HORZ/SB_VERT)", FALSE);
3174 style = GetWindowLongA(hwnd, GWL_STYLE);
3175 if (set) ok(style & set, "style %08x should be set\n", set);
3176 if (clear) ok(!(style & clear), "style %08x should not be set\n", clear);
3178 /* a subsequent call should do nothing */
3179 ret = SetScrollRange(hwnd, ctl, min, max, FALSE);
3180 ok( ret, "SetScrollRange(%d) error %d\n", ctl, GetLastError());
3181 ok_sequence(WmEmptySeq, "SetScrollRange(SB_HORZ/SB_VERT) empty sequence", FALSE);
3185 trace("Ignore GetScrollRange error below if you are on Win9x\n");
3186 ret = GetScrollRange(hwnd, ctl, &xmin, &xmax);
3187 ok( ret, "GetScrollRange(%d) error %d\n", ctl, GetLastError());
3188 ok_sequence(WmEmptySeq, "GetScrollRange(SB_HORZ/SB_VERT) empty sequence", FALSE);
3189 ok(xmin == min, "unexpected min scroll value %d\n", xmin);
3190 ok(xmax == max, "unexpected max scroll value %d\n", xmax);
3193 static void test_hv_scroll_2(HWND hwnd, INT ctl, DWORD clear, DWORD set, INT min, INT max)
3195 DWORD style, exstyle;
3199 exstyle = GetWindowLongA(hwnd, GWL_EXSTYLE);
3200 style = GetWindowLongA(hwnd, GWL_STYLE);
3201 /* do not be confused by WS_DLGFRAME set */
3202 if ((style & WS_CAPTION) == WS_CAPTION) style &= ~WS_CAPTION;
3204 if (clear) ok(style & clear, "style %08x should be set\n", clear);
3205 if (set) ok(!(style & set), "style %08x should not be set\n", set);
3207 si.cbSize = sizeof(si);
3208 si.fMask = SIF_RANGE;
3211 SetScrollInfo(hwnd, ctl, &si, TRUE);
3212 if ((style & (WS_DLGFRAME | WS_BORDER | WS_THICKFRAME)) || (exstyle & WS_EX_DLGMODALFRAME))
3213 ok_sequence(WmSetScrollRangeHV_NC_Seq, "SetScrollInfo(SB_HORZ/SB_VERT) NC", FALSE);
3215 ok_sequence(WmSetScrollRangeHVSeq, "SetScrollInfo(SB_HORZ/SB_VERT)", FALSE);
3217 style = GetWindowLongA(hwnd, GWL_STYLE);
3218 if (set) ok(style & set, "style %08x should be set\n", set);
3219 if (clear) ok(!(style & clear), "style %08x should not be set\n", clear);
3221 /* a subsequent call should do nothing */
3222 SetScrollInfo(hwnd, ctl, &si, TRUE);
3223 if (style & WS_HSCROLL)
3224 ok_sequence(WmSetScrollRangeHSeq_empty, "SetScrollInfo(SB_HORZ/SB_VERT) empty sequence", FALSE);
3225 else if (style & WS_VSCROLL)
3226 ok_sequence(WmSetScrollRangeVSeq_empty, "SetScrollInfo(SB_HORZ/SB_VERT) empty sequence", FALSE);
3228 ok_sequence(WmEmptySeq, "SetScrollInfo(SB_HORZ/SB_VERT) empty sequence", FALSE);
3230 si.fMask = SIF_PAGE;
3232 SetScrollInfo(hwnd, ctl, &si, FALSE);
3233 ok_sequence(WmEmptySeq, "SetScrollInfo(SB_HORZ/SB_VERT) empty sequence", FALSE);
3237 SetScrollInfo(hwnd, ctl, &si, FALSE);
3238 ok_sequence(WmEmptySeq, "SetScrollInfo(SB_HORZ/SB_VERT) empty sequence", FALSE);
3240 si.fMask = SIF_RANGE;
3241 si.nMin = 0xdeadbeef;
3242 si.nMax = 0xdeadbeef;
3243 ret = GetScrollInfo(hwnd, ctl, &si);
3244 ok( ret, "GetScrollInfo error %d\n", GetLastError());
3245 ok_sequence(WmEmptySeq, "GetScrollRange(SB_HORZ/SB_VERT) empty sequence", FALSE);
3246 ok(si.nMin == min, "unexpected min scroll value %d\n", si.nMin);
3247 ok(si.nMax == max, "unexpected max scroll value %d\n", si.nMax);
3250 /* Win9x sends WM_USER+xxx while and NT versions send SBM_xxx messages */
3251 static void test_scroll_messages(HWND hwnd)
3259 ret = GetScrollRange(hwnd, SB_CTL, &min, &max);
3260 ok( ret, "GetScrollRange error %d\n", GetLastError());
3261 if (sequence->message != WmGetScrollRangeSeq[0].message)
3262 trace("GetScrollRange(SB_CTL) generated unknown message %04x\n", sequence->message);
3263 /* values of min and max are undefined */
3266 ret = SetScrollRange(hwnd, SB_CTL, 10, 150, FALSE);
3267 ok( ret, "SetScrollRange error %d\n", GetLastError());
3268 if (sequence->message != WmSetScrollRangeSeq[0].message)
3269 trace("SetScrollRange(SB_CTL) generated unknown message %04x\n", sequence->message);
3274 ret = GetScrollRange(hwnd, SB_CTL, &min, &max);
3275 ok( ret, "GetScrollRange error %d\n", GetLastError());
3276 if (sequence->message != WmGetScrollRangeSeq[0].message)
3277 trace("GetScrollRange(SB_CTL) generated unknown message %04x\n", sequence->message);
3278 /* values of min and max are undefined */
3281 si.cbSize = sizeof(si);
3282 si.fMask = SIF_RANGE;
3285 SetScrollInfo(hwnd, SB_CTL, &si, FALSE);
3286 if (sequence->message != WmSetScrollRangeSeq[0].message)
3287 trace("SetScrollInfo(SB_CTL) generated unknown message %04x\n", sequence->message);
3290 si.fMask = SIF_PAGE;
3292 SetScrollInfo(hwnd, SB_CTL, &si, FALSE);
3293 if (sequence->message != WmSetScrollRangeSeq[0].message)
3294 trace("SetScrollInfo(SB_CTL) generated unknown message %04x\n", sequence->message);
3299 SetScrollInfo(hwnd, SB_CTL, &si, FALSE);
3300 if (sequence->message != WmSetScrollRangeSeq[0].message)
3301 trace("SetScrollInfo(SB_CTL) generated unknown message %04x\n", sequence->message);
3304 si.fMask = SIF_RANGE;
3305 si.nMin = 0xdeadbeef;
3306 si.nMax = 0xdeadbeef;
3307 ret = GetScrollInfo(hwnd, SB_CTL, &si);
3308 ok( ret, "GetScrollInfo error %d\n", GetLastError());
3309 if (sequence->message != WmGetScrollInfoSeq[0].message)
3310 trace("GetScrollInfo(SB_CTL) generated unknown message %04x\n", sequence->message);
3311 /* values of min and max are undefined */
3314 /* set WS_HSCROLL */
3315 test_hv_scroll_1(hwnd, SB_HORZ, 0, WS_HSCROLL, 10, 150);
3316 /* clear WS_HSCROLL */
3317 test_hv_scroll_1(hwnd, SB_HORZ, WS_HSCROLL, 0, 0, 0);
3319 /* set WS_HSCROLL */
3320 test_hv_scroll_2(hwnd, SB_HORZ, 0, WS_HSCROLL, 10, 150);
3321 /* clear WS_HSCROLL */
3322 test_hv_scroll_2(hwnd, SB_HORZ, WS_HSCROLL, 0, 0, 0);
3324 /* set WS_VSCROLL */
3325 test_hv_scroll_1(hwnd, SB_VERT, 0, WS_VSCROLL, 10, 150);
3326 /* clear WS_VSCROLL */
3327 test_hv_scroll_1(hwnd, SB_VERT, WS_VSCROLL, 0, 0, 0);
3329 /* set WS_VSCROLL */
3330 test_hv_scroll_2(hwnd, SB_VERT, 0, WS_VSCROLL, 10, 150);
3331 /* clear WS_VSCROLL */
3332 test_hv_scroll_2(hwnd, SB_VERT, WS_VSCROLL, 0, 0, 0);
3335 static void test_showwindow(void)
3340 hwnd = CreateWindowExA(0, "TestWindowClass", "Test overlapped", WS_OVERLAPPEDWINDOW,
3341 100, 100, 200, 200, 0, 0, 0, NULL);
3342 ok (hwnd != 0, "Failed to create overlapped window\n");
3343 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3344 0, 0, 10, 10, hwnd, 0, 0, NULL);
3345 ok (hchild != 0, "Failed to create child\n");
3348 /* ShowWindow( SW_SHOWNA) for invisible top level window */
3349 trace("calling ShowWindow( SW_SHOWNA) for invisible top level window\n");
3350 ok( ShowWindow(hwnd, SW_SHOWNA) == FALSE, "ShowWindow: window was visible\n" );
3351 ok_sequence(WmSHOWNATopInvisible, "ShowWindow(SW_SHOWNA) on invisible top level window", TRUE);
3354 /* ShowWindow( SW_SHOWNA) for now visible top level window */
3355 trace("calling ShowWindow( SW_SHOWNA) for now visible top level window\n");
3356 ok( ShowWindow(hwnd, SW_SHOWNA) != FALSE, "ShowWindow: window was invisible\n" );
3357 ok_sequence(WmSHOWNATopVisible, "ShowWindow(SW_SHOWNA) on visible top level window", FALSE);
3359 /* back to invisible */
3360 ShowWindow(hchild, SW_HIDE);
3361 ShowWindow(hwnd, SW_HIDE);
3363 /* ShowWindow(SW_SHOWNA) with child and parent invisible */
3364 trace("calling ShowWindow( SW_SHOWNA) for invisible child with invisible parent\n");
3365 ok( ShowWindow(hchild, SW_SHOWNA) == FALSE, "ShowWindow: window was visible\n" );
3366 ok_sequence(WmSHOWNAChildInvisParInvis, "ShowWindow(SW_SHOWNA) invisible child and parent", FALSE);
3368 /* ShowWindow(SW_SHOWNA) with child visible and parent invisible */
3369 ok( ShowWindow(hchild, SW_SHOW) != FALSE, "ShowWindow: window was invisible\n" );
3371 trace("calling ShowWindow( SW_SHOWNA) for the visible child and invisible parent\n");
3372 ok( ShowWindow(hchild, SW_SHOWNA) != FALSE, "ShowWindow: window was invisible\n" );
3373 ok_sequence(WmSHOWNAChildVisParInvis, "ShowWindow(SW_SHOWNA) visible child and invisible parent", FALSE);
3375 /* ShowWindow(SW_SHOWNA) with child visible and parent visible */
3376 ShowWindow( hwnd, SW_SHOW);
3378 trace("calling ShowWindow( SW_SHOWNA) for the visible child and parent\n");
3379 ok( ShowWindow(hchild, SW_SHOWNA) != FALSE, "ShowWindow: window was invisible\n" );
3380 ok_sequence(WmSHOWNAChildVisParVis, "ShowWindow(SW_SHOWNA) for the visible child and parent", FALSE);
3383 /* ShowWindow(SW_SHOWNA) with child invisible and parent visible */
3384 ShowWindow( hchild, SW_HIDE);
3386 trace("calling ShowWindow( SW_SHOWNA) for the invisible child and visible parent\n");
3387 ok( ShowWindow(hchild, SW_SHOWNA) == FALSE, "ShowWindow: window was visible\n" );
3388 ok_sequence(WmSHOWNAChildInvisParVis, "ShowWindow(SW_SHOWNA) for the invisible child and visible parent", FALSE);
3392 ok(GetCapture() == hchild, "wrong capture window %p\n", GetCapture());
3393 DestroyWindow(hchild);
3394 ok(!GetCapture(), "wrong capture window %p\n", GetCapture());
3396 DestroyWindow(hwnd);
3401 * 1. Create invisible maximized popup window.
3402 * 2. Move and resize it.
3403 * 3. Show it maximized.
3405 trace("calling CreateWindowExA( WS_MAXIMIZE ) for invisible maximized popup window\n");
3406 hwnd = CreateWindowExA(0, "TestWindowClass", "Test popup", WS_POPUP | WS_MAXIMIZE,
3407 100, 100, 200, 200, 0, 0, 0, NULL);
3408 ok (hwnd != 0, "Failed to create popup window\n");
3409 ok(IsZoomed(hwnd), "window should be maximized\n");
3410 ok_sequence(WmCreateInvisibleMaxPopupSeq, "CreateWindow(WS_MAXIMIZED):popup", FALSE);
3413 GetWindowRect(hwnd, &rc);
3414 ok( rc.right-rc.left == GetSystemMetrics(SM_CXSCREEN) &&
3415 rc.bottom-rc.top == GetSystemMetrics(SM_CYSCREEN),
3416 "Invalid maximized size before ShowWindow (%d,%d)-(%d,%d)\n",
3417 rc.left, rc.top, rc.right, rc.bottom);
3418 /* Reset window's size & position */
3419 SetWindowPos(hwnd, 0, 10, 10, 200, 200, SWP_NOZORDER | SWP_NOACTIVATE);
3420 ok(IsZoomed(hwnd), "window should be maximized\n");
3423 trace("calling ShowWindow( SW_SHOWMAXIMIZE ) for invisible maximized popup window\n");
3424 ShowWindow(hwnd, SW_SHOWMAXIMIZED);
3425 ok(IsZoomed(hwnd), "window should be maximized\n");
3426 ok_sequence(WmShowMaxPopupResizedSeq, "ShowWindow(SW_SHOWMAXIMIZED):popup", FALSE);
3429 GetWindowRect(hwnd, &rc);
3430 ok( rc.right-rc.left == GetSystemMetrics(SM_CXSCREEN) &&
3431 rc.bottom-rc.top == GetSystemMetrics(SM_CYSCREEN),
3432 "Invalid maximized size after ShowWindow (%d,%d)-(%d,%d)\n",
3433 rc.left, rc.top, rc.right, rc.bottom);
3434 DestroyWindow(hwnd);
3438 * 1. Create invisible maximized popup window.
3439 * 2. Show it maximized.
3441 trace("calling CreateWindowExA( WS_MAXIMIZE ) for invisible maximized popup window\n");
3442 hwnd = CreateWindowExA(0, "TestWindowClass", "Test popup", WS_POPUP | WS_MAXIMIZE,
3443 100, 100, 200, 200, 0, 0, 0, NULL);
3444 ok (hwnd != 0, "Failed to create popup window\n");
3445 ok(IsZoomed(hwnd), "window should be maximized\n");
3446 ok_sequence(WmCreateInvisibleMaxPopupSeq, "CreateWindow(WS_MAXIMIZED):popup", FALSE);
3449 trace("calling ShowWindow( SW_SHOWMAXIMIZE ) for invisible maximized popup window\n");
3450 ShowWindow(hwnd, SW_SHOWMAXIMIZED);
3451 ok(IsZoomed(hwnd), "window should be maximized\n");
3452 ok_sequence(WmShowMaxPopupSeq, "ShowWindow(SW_SHOWMAXIMIZED):popup", FALSE);
3454 DestroyWindow(hwnd);
3458 * 1. Create visible maximized popup window.
3460 trace("calling CreateWindowExA( WS_MAXIMIZE ) for maximized popup window\n");
3461 hwnd = CreateWindowExA(0, "TestWindowClass", "Test popup", WS_POPUP | WS_MAXIMIZE | WS_VISIBLE,
3462 100, 100, 200, 200, 0, 0, 0, NULL);
3463 ok (hwnd != 0, "Failed to create popup window\n");
3464 ok(IsZoomed(hwnd), "window should be maximized\n");
3465 ok_sequence(WmCreateMaxPopupSeq, "CreateWindow(WS_MAXIMIZED):popup", FALSE);
3467 DestroyWindow(hwnd);
3471 * 1. Create visible popup window.
3474 trace("calling CreateWindowExA( WS_VISIBLE ) for popup window\n");
3475 hwnd = CreateWindowExA(0, "TestWindowClass", "Test popup", WS_POPUP | WS_VISIBLE,
3476 100, 100, 200, 200, 0, 0, 0, NULL);
3477 ok (hwnd != 0, "Failed to create popup window\n");
3478 ok(!IsZoomed(hwnd), "window should NOT be maximized\n");
3479 ok_sequence(WmCreatePopupSeq, "CreateWindow(WS_VISIBLE):popup", TRUE);
3482 trace("calling ShowWindow( SW_SHOWMAXIMIZE ) for visible popup window\n");
3483 ShowWindow(hwnd, SW_SHOWMAXIMIZED);
3484 ok(IsZoomed(hwnd), "window should be maximized\n");
3485 ok_sequence(WmShowVisMaxPopupSeq, "ShowWindow(SW_SHOWMAXIMIZED):popup", TRUE);
3487 DestroyWindow(hwnd);
3491 static void test_sys_menu(HWND hwnd)
3496 /* test existing window without CS_NOCLOSE style */
3497 hmenu = GetSystemMenu(hwnd, FALSE);
3498 ok(hmenu != 0, "GetSystemMenu error %d\n", GetLastError());
3500 state = GetMenuState(hmenu, SC_CLOSE, MF_BYCOMMAND);
3501 ok(state != 0xffffffff, "wrong SC_CLOSE state %x\n", state);
3502 ok(!(state & (MF_DISABLED | MF_GRAYED)), "wrong SC_CLOSE state %x\n", state);
3504 EnableMenuItem(hmenu, SC_CLOSE, MF_BYCOMMAND | MF_GRAYED);
3505 ok_sequence(WmEmptySeq, "WmEnableMenuItem", FALSE);
3507 state = GetMenuState(hmenu, SC_CLOSE, MF_BYCOMMAND);
3508 ok(state != 0xffffffff, "wrong SC_CLOSE state %x\n", state);
3509 ok((state & (MF_DISABLED | MF_GRAYED)) == MF_GRAYED, "wrong SC_CLOSE state %x\n", state);
3511 EnableMenuItem(hmenu, SC_CLOSE, 0);
3512 ok_sequence(WmEmptySeq, "WmEnableMenuItem", FALSE);
3514 state = GetMenuState(hmenu, SC_CLOSE, MF_BYCOMMAND);
3515 ok(state != 0xffffffff, "wrong SC_CLOSE state %x\n", state);
3516 ok(!(state & (MF_DISABLED | MF_GRAYED)), "wrong SC_CLOSE state %x\n", state);
3518 /* test new window with CS_NOCLOSE style */
3519 hwnd = CreateWindowExA(0, "NoCloseWindowClass", NULL, WS_OVERLAPPEDWINDOW,
3520 100, 100, 200, 200, 0, 0, 0, NULL);
3521 ok (hwnd != 0, "Failed to create overlapped window\n");
3523 hmenu = GetSystemMenu(hwnd, FALSE);
3524 ok(hmenu != 0, "GetSystemMenu error %d\n", GetLastError());
3526 state = GetMenuState(hmenu, SC_CLOSE, MF_BYCOMMAND);
3527 ok(state == 0xffffffff, "wrong SC_CLOSE state %x\n", state);
3529 DestroyWindow(hwnd);
3532 /* test if we receive the right sequence of messages */
3533 static void test_messages(void)
3535 HWND hwnd, hparent, hchild;
3536 HWND hchild2, hbutton;
3543 hwnd = CreateWindowExA(0, "TestWindowClass", "Test overlapped", WS_OVERLAPPEDWINDOW,
3544 100, 100, 200, 200, 0, 0, 0, NULL);
3545 ok (hwnd != 0, "Failed to create overlapped window\n");
3546 ok_sequence(WmCreateOverlappedSeq, "CreateWindow:overlapped", FALSE);
3548 /* test ShowWindow(SW_HIDE) on a newly created invisible window */
3549 ok( ShowWindow(hwnd, SW_HIDE) == FALSE, "ShowWindow: window was visible\n" );
3550 ok_sequence(WmEmptySeq, "ShowWindow(SW_HIDE):overlapped, invisible", FALSE);
3552 /* test WM_SETREDRAW on a not visible top level window */
3553 test_WM_SETREDRAW(hwnd);
3555 SetWindowPos(hwnd, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE);
3556 ok_sequence(WmSWP_ShowOverlappedSeq, "SetWindowPos:SWP_SHOWWINDOW:overlapped", FALSE);
3557 ok(IsWindowVisible(hwnd), "window should be visible at this point\n");
3559 ok(GetActiveWindow() == hwnd, "window should be active\n");
3560 ok(GetFocus() == hwnd, "window should have input focus\n");
3561 ShowWindow(hwnd, SW_HIDE);
3562 ok_sequence(WmHideOverlappedSeq, "ShowWindow(SW_HIDE):overlapped", TRUE);
3564 ShowWindow(hwnd, SW_SHOW);
3565 ok_sequence(WmShowOverlappedSeq, "ShowWindow(SW_SHOW):overlapped", TRUE);
3567 ShowWindow(hwnd, SW_HIDE);
3568 ok_sequence(WmHideOverlappedSeq, "ShowWindow(SW_HIDE):overlapped", FALSE);
3570 ShowWindow(hwnd, SW_SHOWMAXIMIZED);
3571 ok_sequence(WmShowMaxOverlappedSeq, "ShowWindow(SW_SHOWMAXIMIZED):overlapped", TRUE);
3573 ShowWindow(hwnd, SW_RESTORE);
3574 /* FIXME: add ok_sequence() here */
3577 ShowWindow(hwnd, SW_SHOW);
3578 ok_sequence(WmEmptySeq, "ShowWindow(SW_SHOW):overlapped already visible", FALSE);
3580 ok(GetActiveWindow() == hwnd, "window should be active\n");
3581 ok(GetFocus() == hwnd, "window should have input focus\n");
3582 SetWindowPos(hwnd, 0,0,0,0,0, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE);
3583 ok_sequence(WmSWP_HideOverlappedSeq, "SetWindowPos:SWP_HIDEWINDOW:overlapped", FALSE);
3584 ok(!IsWindowVisible(hwnd), "window should not be visible at this point\n");
3585 ok(GetActiveWindow() == hwnd, "window should still be active\n");
3587 /* test WM_SETREDRAW on a visible top level window */
3588 ShowWindow(hwnd, SW_SHOW);
3589 test_WM_SETREDRAW(hwnd);
3591 trace("testing scroll APIs on a visible top level window %p\n", hwnd);
3592 test_scroll_messages(hwnd);
3594 /* test resizing and moving */
3595 SetWindowPos( hwnd, 0, 0, 0, 300, 300, SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE );
3596 ok_sequence(WmSWP_ResizeSeq, "SetWindowPos:Resize", FALSE );
3597 SetWindowPos( hwnd, 0, 200, 200, 0, 0, SWP_NOSIZE|SWP_NOZORDER|SWP_NOACTIVATE );
3598 ok_sequence(WmSWP_MoveSeq, "SetWindowPos:Move", FALSE );
3600 /* popups don't get WM_GETMINMAXINFO */
3601 SetWindowLongW( hwnd, GWL_STYLE, WS_VISIBLE|WS_POPUP );
3602 SetWindowPos( hwnd, 0, 0, 0, 0, 0, SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOMOVE|SWP_FRAMECHANGED);
3604 SetWindowPos( hwnd, 0, 0, 0, 200, 200, SWP_NOMOVE|SWP_NOZORDER|SWP_NOACTIVATE );
3605 ok_sequence(WmSWP_ResizePopupSeq, "SetWindowPos:ResizePopup", FALSE );
3607 test_sys_menu(hwnd);
3610 DestroyWindow(hwnd);
3611 ok_sequence(WmDestroyOverlappedSeq, "DestroyWindow:overlapped", FALSE);
3613 hparent = CreateWindowExA(0, "TestParentClass", "Test parent", WS_OVERLAPPEDWINDOW | WS_VISIBLE,
3614 100, 100, 200, 200, 0, 0, 0, NULL);
3615 ok (hparent != 0, "Failed to create parent window\n");
3618 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD | WS_MAXIMIZE,
3619 0, 0, 10, 10, hparent, 0, 0, NULL);
3620 ok (hchild != 0, "Failed to create child window\n");
3621 ok_sequence(WmCreateMaximizedChildSeq, "CreateWindow:maximized child", TRUE);
3622 DestroyWindow(hchild);
3625 /* visible child window with a caption */
3626 hchild = CreateWindowExA(0, "TestWindowClass", "Test child",
3627 WS_CHILD | WS_VISIBLE | WS_CAPTION,
3628 0, 0, 10, 10, hparent, 0, 0, NULL);
3629 ok (hchild != 0, "Failed to create child window\n");
3630 ok_sequence(WmCreateVisibleChildSeq, "CreateWindow:visible child", FALSE);
3632 trace("testing scroll APIs on a visible child window %p\n", hchild);
3633 test_scroll_messages(hchild);
3635 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE);
3636 ok_sequence(WmShowChildSeq_4, "SetWindowPos(SWP_SHOWWINDOW):child with a caption", FALSE);
3638 DestroyWindow(hchild);
3641 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3642 0, 0, 10, 10, hparent, 0, 0, NULL);
3643 ok (hchild != 0, "Failed to create child window\n");
3644 ok_sequence(WmCreateChildSeq, "CreateWindow:child", FALSE);
3646 hchild2 = CreateWindowExA(0, "SimpleWindowClass", "Test child2", WS_CHILD,
3647 100, 100, 50, 50, hparent, 0, 0, NULL);
3648 ok (hchild2 != 0, "Failed to create child2 window\n");
3651 hbutton = CreateWindowExA(0, "TestWindowClass", "Test button", WS_CHILD,
3652 0, 100, 50, 50, hchild, 0, 0, NULL);
3653 ok (hbutton != 0, "Failed to create button window\n");
3655 /* test WM_SETREDRAW on a not visible child window */
3656 test_WM_SETREDRAW(hchild);
3658 ShowWindow(hchild, SW_SHOW);
3659 ok_sequence(WmShowChildSeq, "ShowWindow(SW_SHOW):child", FALSE);
3661 ShowWindow(hchild, SW_HIDE);
3662 ok_sequence(WmHideChildSeq, "ShowWindow(SW_HIDE):child", FALSE);
3664 ShowWindow(hchild, SW_SHOW);
3665 ok_sequence(WmShowChildSeq, "ShowWindow(SW_SHOW):child", FALSE);
3667 /* test WM_SETREDRAW on a visible child window */
3668 test_WM_SETREDRAW(hchild);
3670 MoveWindow(hchild, 10, 10, 20, 20, TRUE);
3671 ok_sequence(WmResizingChildWithMoveWindowSeq, "MoveWindow:child", FALSE);
3673 ShowWindow(hchild, SW_HIDE);
3675 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE);
3676 ok_sequence(WmShowChildSeq_2, "SetWindowPos:show_child_2", FALSE);
3678 ShowWindow(hchild, SW_HIDE);
3680 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE);
3681 ok_sequence(WmShowChildSeq_3, "SetWindowPos:show_child_3", FALSE);
3683 /* DestroyWindow sequence below expects that a child has focus */
3687 DestroyWindow(hchild);
3688 ok_sequence(WmDestroyChildSeq, "DestroyWindow:child", FALSE);
3689 DestroyWindow(hchild2);
3690 DestroyWindow(hbutton);
3693 hchild = CreateWindowExA(0, "TestWindowClass", "Test Child Popup", WS_CHILD | WS_POPUP,
3694 0, 0, 100, 100, hparent, 0, 0, NULL);
3695 ok (hchild != 0, "Failed to create child popup window\n");
3696 ok_sequence(WmCreateChildPopupSeq, "CreateWindow:child_popup", FALSE);
3697 DestroyWindow(hchild);
3699 /* test what happens to a window which sets WS_VISIBLE in WM_CREATE */
3701 hchild = CreateWindowExA(0, "TestPopupClass", "Test Popup", WS_POPUP,
3702 0, 0, 100, 100, hparent, 0, 0, NULL);
3703 ok (hchild != 0, "Failed to create popup window\n");
3704 ok_sequence(WmCreateInvisiblePopupSeq, "CreateWindow:invisible_popup", FALSE);
3705 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3706 ok(IsWindowVisible(hchild), "IsWindowVisible() should return TRUE\n");
3708 ShowWindow(hchild, SW_SHOW);
3709 ok_sequence(WmEmptySeq, "ShowWindow:show_visible_popup", FALSE);
3711 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER);
3712 ok_sequence(WmShowVisiblePopupSeq_2, "SetWindowPos:show_visible_popup_2", FALSE);
3714 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE);
3715 ok_sequence(WmShowVisiblePopupSeq_3, "SetWindowPos:show_visible_popup_3", FALSE);
3716 DestroyWindow(hchild);
3718 /* this time add WS_VISIBLE for CreateWindowEx, but this fact actually
3719 * changes nothing in message sequences.
3722 hchild = CreateWindowExA(0, "TestPopupClass", "Test Popup", WS_POPUP | WS_VISIBLE,
3723 0, 0, 100, 100, hparent, 0, 0, NULL);
3724 ok (hchild != 0, "Failed to create popup window\n");
3725 ok_sequence(WmCreateInvisiblePopupSeq, "CreateWindow:invisible_popup", FALSE);
3726 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3727 ok(IsWindowVisible(hchild), "IsWindowVisible() should return TRUE\n");
3729 ShowWindow(hchild, SW_SHOW);
3730 ok_sequence(WmEmptySeq, "ShowWindow:show_visible_popup", FALSE);
3732 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER);
3733 ok_sequence(WmShowVisiblePopupSeq_2, "SetWindowPos:show_visible_popup_2", FALSE);
3734 DestroyWindow(hchild);
3737 hwnd = CreateWindowExA(WS_EX_DLGMODALFRAME, "TestDialogClass", NULL, WS_VISIBLE|WS_CAPTION|WS_SYSMENU|WS_DLGFRAME,
3738 0, 0, 100, 100, hparent, 0, 0, NULL);
3739 ok(hwnd != 0, "Failed to create custom dialog window\n");
3740 ok_sequence(WmCreateCustomDialogSeq, "CreateCustomDialog", TRUE);
3743 trace("testing scroll APIs on a visible dialog %p\n", hwnd);
3744 test_scroll_messages(hwnd);
3748 after_end_dialog = 1;
3749 EndDialog( hwnd, 0 );
3750 ok_sequence(WmEndCustomDialogSeq, "EndCustomDialog", FALSE);
3752 DestroyWindow(hwnd);
3753 after_end_dialog = 0;
3755 hwnd = CreateWindowExA(0, "TestDialogClass", NULL, WS_POPUP,
3756 0, 0, 100, 100, 0, 0, GetModuleHandleA(0), NULL);
3757 ok(hwnd != 0, "Failed to create custom dialog window\n");
3759 trace("call ShowWindow(%p, SW_SHOW)\n", hwnd);
3760 ShowWindow(hwnd, SW_SHOW);
3761 ok_sequence(WmShowCustomDialogSeq, "ShowCustomDialog", TRUE);
3762 DestroyWindow(hwnd);
3765 DialogBoxA( 0, "TEST_DIALOG", hparent, TestModalDlgProcA );
3766 ok_sequence(WmModalDialogSeq, "ModalDialog", TRUE);
3768 DestroyWindow(hparent);
3771 /* Message sequence for SetMenu */
3772 ok(!DrawMenuBar(hwnd), "DrawMenuBar should return FALSE for a window without a menu\n");
3773 ok_sequence(WmEmptySeq, "DrawMenuBar for a window without a menu", FALSE);
3775 hmenu = CreateMenu();
3776 ok (hmenu != 0, "Failed to create menu\n");
3777 ok (InsertMenuA(hmenu, -1, MF_BYPOSITION, 0x1000, "foo"), "InsertMenu failed\n");
3778 hwnd = CreateWindowExA(0, "TestWindowClass", "Test overlapped", WS_OVERLAPPEDWINDOW,
3779 100, 100, 200, 200, 0, hmenu, 0, NULL);
3780 ok_sequence(WmCreateOverlappedSeq, "CreateWindow:overlapped", FALSE);
3781 ok (SetMenu(hwnd, 0), "SetMenu\n");
3782 ok_sequence(WmSetMenuNonVisibleSizeChangeSeq, "SetMenu:NonVisibleSizeChange", FALSE);
3783 ok (SetMenu(hwnd, 0), "SetMenu\n");
3784 ok_sequence(WmSetMenuNonVisibleNoSizeChangeSeq, "SetMenu:NonVisibleNoSizeChange", FALSE);
3785 ShowWindow(hwnd, SW_SHOW);
3786 UpdateWindow( hwnd );
3787 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
3789 ok (SetMenu(hwnd, 0), "SetMenu\n");
3790 ok_sequence(WmSetMenuVisibleNoSizeChangeSeq, "SetMenu:VisibleNoSizeChange", FALSE);
3791 ok (SetMenu(hwnd, hmenu), "SetMenu\n");
3792 ok_sequence(WmSetMenuVisibleSizeChangeSeq, "SetMenu:VisibleSizeChange", FALSE);
3794 UpdateWindow( hwnd );
3795 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
3797 ok(DrawMenuBar(hwnd), "DrawMenuBar\n");
3798 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
3799 ok_sequence(WmDrawMenuBarSeq, "DrawMenuBar", FALSE);
3801 DestroyWindow(hwnd);
3804 /* Message sequence for EnableWindow */
3805 hparent = CreateWindowExA(0, "TestWindowClass", "Test parent", WS_OVERLAPPEDWINDOW | WS_VISIBLE,
3806 100, 100, 200, 200, 0, 0, 0, NULL);
3807 ok (hparent != 0, "Failed to create parent window\n");
3808 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD | WS_VISIBLE,
3809 0, 0, 10, 10, hparent, 0, 0, NULL);
3810 ok (hchild != 0, "Failed to create child window\n");
3815 EnableWindow(hparent, FALSE);
3816 ok_sequence(WmEnableWindowSeq_1, "EnableWindow(FALSE)", FALSE);
3818 EnableWindow(hparent, TRUE);
3819 ok_sequence(WmEnableWindowSeq_2, "EnableWindow(TRUE)", FALSE);
3821 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
3824 /* MsgWaitForMultipleObjects test */
3825 ret = MsgWaitForMultipleObjects(0, NULL, FALSE, 0, QS_POSTMESSAGE);
3826 ok(ret == WAIT_TIMEOUT, "MsgWaitForMultipleObjects returned %x\n", ret);
3828 PostMessageA(hparent, WM_USER, 0, 0);
3830 ret = MsgWaitForMultipleObjects(0, NULL, FALSE, 0, QS_POSTMESSAGE);
3831 ok(ret == WAIT_OBJECT_0, "MsgWaitForMultipleObjects returned %x\n", ret);
3833 ok(PeekMessageW( &msg, 0, 0, 0, PM_REMOVE ), "PeekMessage should succeed\n");
3834 ok(msg.message == WM_USER, "got %04x instead of WM_USER\n", msg.message);
3836 ret = MsgWaitForMultipleObjects(0, NULL, FALSE, 0, QS_POSTMESSAGE);
3837 ok(ret == WAIT_TIMEOUT, "MsgWaitForMultipleObjects returned %x\n", ret);
3838 /* end of MsgWaitForMultipleObjects test */
3840 /* the following test causes an exception in user.exe under win9x */
3841 if (!PostMessageW( hparent, WM_USER, 0, 0 )) return;
3842 PostMessageW( hparent, WM_USER+1, 0, 0 );
3843 /* PeekMessage(NULL) fails, but still removes the message */
3844 SetLastError(0xdeadbeef);
3845 ok( !PeekMessageW( NULL, 0, 0, 0, PM_REMOVE ), "PeekMessage(NULL) should fail\n" );
3846 ok( GetLastError() == ERROR_NOACCESS || /* Win2k */
3847 GetLastError() == 0xdeadbeef, /* NT4 */
3848 "last error is %d\n", GetLastError() );
3849 ok( PeekMessageW( &msg, 0, 0, 0, PM_REMOVE ), "PeekMessage should succeed\n" );
3850 ok( msg.message == WM_USER+1, "got %x instead of WM_USER+1\n", msg.message );
3852 DestroyWindow(hchild);
3853 DestroyWindow(hparent);
3859 static void invisible_parent_tests(void)
3861 HWND hparent, hchild;
3863 hparent = CreateWindowExA(0, "TestParentClass", "Test parent", WS_OVERLAPPEDWINDOW,
3864 100, 100, 200, 200, 0, 0, 0, NULL);
3865 ok (hparent != 0, "Failed to create parent window\n");
3868 /* test showing child with hidden parent */
3870 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3871 0, 0, 10, 10, hparent, 0, 0, NULL);
3872 ok (hchild != 0, "Failed to create child window\n");
3873 ok_sequence(WmCreateChildSeq, "CreateWindow:child", FALSE);
3875 ShowWindow( hchild, SW_MINIMIZE );
3876 ok_sequence(WmShowChildInvisibleParentSeq_1, "ShowWindow(SW_MINIMIZE) child with invisible parent", FALSE);
3877 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3878 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3882 ShowWindow( hchild, SW_MINIMIZE );
3883 ok_sequence(WmShowChildInvisibleParentSeq_1r, "ShowWindow(SW_MINIMIZE) child with invisible parent", FALSE);
3885 DestroyWindow(hchild);
3886 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3887 0, 0, 10, 10, hparent, 0, 0, NULL);
3890 ShowWindow( hchild, SW_MAXIMIZE );
3891 ok_sequence(WmShowChildInvisibleParentSeq_2, "ShowWindow(SW_MAXIMIZE) child with invisible parent", FALSE);
3892 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3893 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3897 ShowWindow( hchild, SW_MAXIMIZE );
3898 ok_sequence(WmShowChildInvisibleParentSeq_2r, "ShowWindow(SW_MAXIMIZE) child with invisible parent", FALSE);
3900 DestroyWindow(hchild);
3901 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3902 0, 0, 10, 10, hparent, 0, 0, NULL);
3905 ShowWindow( hchild, SW_SHOWMINIMIZED );
3906 ok_sequence(WmShowChildInvisibleParentSeq_3, "ShowWindow(SW_SHOWMINIMIZED) child with invisible parent", FALSE);
3907 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3908 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3912 ShowWindow( hchild, SW_SHOWMINIMIZED );
3913 ok_sequence(WmShowChildInvisibleParentSeq_3r, "ShowWindow(SW_SHOWMINIMIZED) child with invisible parent", FALSE);
3915 DestroyWindow(hchild);
3916 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3917 0, 0, 10, 10, hparent, 0, 0, NULL);
3920 /* same as ShowWindow( hchild, SW_MAXIMIZE ); */
3921 ShowWindow( hchild, SW_SHOWMAXIMIZED );
3922 ok_sequence(WmShowChildInvisibleParentSeq_2, "ShowWindow(SW_SHOWMAXIMIZED) child with invisible parent", FALSE);
3923 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3924 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3926 DestroyWindow(hchild);
3927 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3928 0, 0, 10, 10, hparent, 0, 0, NULL);
3931 ShowWindow( hchild, SW_SHOWMINNOACTIVE );
3932 ok_sequence(WmShowChildInvisibleParentSeq_4, "ShowWindow(SW_SHOWMINNOACTIVE) child with invisible parent", FALSE);
3933 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3934 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3938 ShowWindow( hchild, SW_SHOWMINNOACTIVE );
3939 ok_sequence(WmShowChildInvisibleParentSeq_4r, "ShowWindow(SW_SHOWMINNOACTIVE) child with invisible parent", FALSE);
3941 DestroyWindow(hchild);
3942 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3943 0, 0, 10, 10, hparent, 0, 0, NULL);
3946 /* FIXME: looks like XP SP2 doesn't know about SW_FORCEMINIMIZE at all */
3947 ShowWindow( hchild, SW_FORCEMINIMIZE );
3948 ok_sequence(WmEmptySeq, "ShowWindow(SW_FORCEMINIMIZE) child with invisible parent", TRUE);
3950 ok(!(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE), "WS_VISIBLE should be not set\n");
3952 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3954 DestroyWindow(hchild);
3955 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3956 0, 0, 10, 10, hparent, 0, 0, NULL);
3959 ShowWindow( hchild, SW_SHOWNA );
3960 ok_sequence(WmShowChildInvisibleParentSeq_5, "ShowWindow(SW_SHOWNA) child with invisible parent", FALSE);
3961 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3962 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3966 ShowWindow( hchild, SW_SHOWNA );
3967 ok_sequence(WmShowChildInvisibleParentSeq_5, "ShowWindow(SW_SHOWNA) child with invisible parent", FALSE);
3969 DestroyWindow(hchild);
3970 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD,
3971 0, 0, 10, 10, hparent, 0, 0, NULL);
3974 ShowWindow( hchild, SW_SHOW );
3975 ok_sequence(WmShowChildInvisibleParentSeq_5, "ShowWindow(SW_SHOW) child with invisible parent", FALSE);
3976 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3977 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3981 ShowWindow( hchild, SW_SHOW );
3982 ok_sequence(WmEmptySeq, "ShowWindow(SW_SHOW) child with invisible parent", FALSE);
3984 ShowWindow( hchild, SW_HIDE );
3985 ok_sequence(WmHideChildInvisibleParentSeq, "ShowWindow:hide child with invisible parent", FALSE);
3986 ok(!(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE), "WS_VISIBLE should be not set\n");
3987 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3989 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER);
3990 ok_sequence(WmShowChildInvisibleParentSeq_6, "SetWindowPos:show child with invisible parent", FALSE);
3991 ok(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE, "WS_VISIBLE should be set\n");
3992 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3994 SetWindowPos(hchild, 0,0,0,0,0, SWP_HIDEWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER);
3995 ok_sequence(WmHideChildInvisibleParentSeq_2, "SetWindowPos:hide child with invisible parent", FALSE);
3996 ok(!(GetWindowLongA(hchild, GWL_STYLE) & WS_VISIBLE), "WS_VISIBLE should not be set\n");
3997 ok(!IsWindowVisible(hchild), "IsWindowVisible() should return FALSE\n");
3999 SetWindowPos(hchild, 0,0,0,0,0, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOACTIVATE|SWP_NOZORDER);
4001 DestroyWindow(hchild);
4002 ok_sequence(WmDestroyInvisibleChildSeq, "DestroyInvisibleChildSeq", FALSE);
4004 DestroyWindow(hparent);
4008 /****************** button message test *************************/
4009 static const struct message WmSetFocusButtonSeq[] =
4011 { HCBT_SETFOCUS, hook },
4012 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
4013 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4014 { WM_SETFOCUS, sent|wparam, 0 },
4015 { WM_CTLCOLORBTN, sent|defwinproc },
4018 static const struct message WmKillFocusButtonSeq[] =
4020 { HCBT_SETFOCUS, hook },
4021 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4022 { WM_KILLFOCUS, sent|wparam, 0 },
4023 { WM_CTLCOLORBTN, sent|defwinproc },
4024 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
4027 static const struct message WmSetFocusStaticSeq[] =
4029 { HCBT_SETFOCUS, hook },
4030 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
4031 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4032 { WM_SETFOCUS, sent|wparam, 0 },
4033 { WM_CTLCOLORSTATIC, sent|defwinproc },
4036 static const struct message WmKillFocusStaticSeq[] =
4038 { HCBT_SETFOCUS, hook },
4039 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4040 { WM_KILLFOCUS, sent|wparam, 0 },
4041 { WM_CTLCOLORSTATIC, sent|defwinproc },
4042 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
4045 static const struct message WmLButtonDownSeq[] =
4047 { WM_LBUTTONDOWN, sent|wparam|lparam, 0, 0 },
4048 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 },
4049 { HCBT_SETFOCUS, hook },
4050 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
4051 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4052 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
4053 { WM_CTLCOLORBTN, sent|defwinproc },
4054 { BM_SETSTATE, sent|wparam|defwinproc, TRUE },
4055 { WM_CTLCOLORBTN, sent|defwinproc },
4056 { EVENT_OBJECT_STATECHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4059 static const struct message WmLButtonUpSeq[] =
4061 { WM_LBUTTONUP, sent|wparam|lparam, 0, 0 },
4062 { BM_SETSTATE, sent|wparam|defwinproc, FALSE },
4063 { WM_CTLCOLORBTN, sent|defwinproc },
4064 { EVENT_OBJECT_STATECHANGE, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
4065 { EVENT_SYSTEM_CAPTUREEND, winevent_hook|wparam|lparam, 0, 0 },
4066 { WM_CAPTURECHANGED, sent|wparam|defwinproc, 0 },
4070 static WNDPROC old_button_proc;
4072 static LRESULT CALLBACK button_hook_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
4074 static long defwndproc_counter = 0;
4078 trace("button: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
4080 /* explicitly ignore WM_GETICON message */
4081 if (message == WM_GETICON) return 0;
4083 msg.message = message;
4084 msg.flags = sent|wparam|lparam;
4085 if (defwndproc_counter) msg.flags |= defwinproc;
4086 msg.wParam = wParam;
4087 msg.lParam = lParam;
4090 if (message == BM_SETSTATE)
4091 ok(GetCapture() == hwnd, "GetCapture() = %p\n", GetCapture());
4093 defwndproc_counter++;
4094 ret = CallWindowProcA(old_button_proc, hwnd, message, wParam, lParam);
4095 defwndproc_counter--;
4100 static void subclass_button(void)
4104 if (!GetClassInfoA(0, "button", &cls)) assert(0);
4106 old_button_proc = cls.lpfnWndProc;
4108 cls.hInstance = GetModuleHandle(0);
4109 cls.lpfnWndProc = button_hook_proc;
4110 cls.lpszClassName = "my_button_class";
4111 if (!RegisterClassA(&cls)) assert(0);
4114 static void test_button_messages(void)
4120 const struct message *setfocus;
4121 const struct message *killfocus;
4123 { BS_PUSHBUTTON, DLGC_BUTTON | DLGC_UNDEFPUSHBUTTON,
4124 WmSetFocusButtonSeq, WmKillFocusButtonSeq },
4125 { BS_DEFPUSHBUTTON, DLGC_BUTTON | DLGC_DEFPUSHBUTTON,
4126 WmSetFocusButtonSeq, WmKillFocusButtonSeq },
4127 { BS_CHECKBOX, DLGC_BUTTON,
4128 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4129 { BS_AUTOCHECKBOX, DLGC_BUTTON,
4130 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4131 { BS_RADIOBUTTON, DLGC_BUTTON | DLGC_RADIOBUTTON,
4132 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4133 { BS_3STATE, DLGC_BUTTON,
4134 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4135 { BS_AUTO3STATE, DLGC_BUTTON,
4136 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4137 { BS_GROUPBOX, DLGC_STATIC,
4138 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4139 { BS_USERBUTTON, DLGC_BUTTON | DLGC_UNDEFPUSHBUTTON,
4140 WmSetFocusButtonSeq, WmKillFocusButtonSeq },
4141 { BS_AUTORADIOBUTTON, DLGC_BUTTON | DLGC_RADIOBUTTON,
4142 WmSetFocusStaticSeq, WmKillFocusStaticSeq },
4143 { BS_OWNERDRAW, DLGC_BUTTON,
4144 WmSetFocusButtonSeq, WmKillFocusButtonSeq }
4152 for (i = 0; i < sizeof(button)/sizeof(button[0]); i++)
4154 hwnd = CreateWindowExA(0, "my_button_class", "test", button[i].style | WS_POPUP,
4155 0, 0, 50, 14, 0, 0, 0, NULL);
4156 ok(hwnd != 0, "Failed to create button window\n");
4158 dlg_code = SendMessageA(hwnd, WM_GETDLGCODE, 0, 0);
4159 ok(dlg_code == button[i].dlg_code, "%d: wrong dlg_code %08x\n", i, dlg_code);
4161 ShowWindow(hwnd, SW_SHOW);
4166 trace("button style %08x\n", button[i].style);
4168 ok_sequence(button[i].setfocus, "SetFocus(hwnd) on a button", FALSE);
4171 ok_sequence(button[i].killfocus, "SetFocus(0) on a button", FALSE);
4173 DestroyWindow(hwnd);
4176 hwnd = CreateWindowExA(0, "my_button_class", "test", BS_PUSHBUTTON | WS_POPUP | WS_VISIBLE,
4177 0, 0, 50, 14, 0, 0, 0, NULL);
4178 ok(hwnd != 0, "Failed to create button window\n");
4183 SendMessageA(hwnd, WM_LBUTTONDOWN, 0, 0);
4184 ok_sequence(WmLButtonDownSeq, "WM_LBUTTONDOWN on a button", FALSE);
4186 SendMessageA(hwnd, WM_LBUTTONUP, 0, 0);
4187 ok_sequence(WmLButtonUpSeq, "WM_LBUTTONUP on a button", FALSE);
4188 DestroyWindow(hwnd);
4191 /************* painting message test ********************/
4193 void dump_region(HRGN hrgn)
4196 RGNDATA *data = NULL;
4201 printf( "null region\n" );
4204 if (!(size = GetRegionData( hrgn, 0, NULL ))) return;
4205 if (!(data = HeapAlloc( GetProcessHeap(), 0, size ))) return;
4206 GetRegionData( hrgn, size, data );
4207 printf("%d rects:", data->rdh.nCount );
4208 for (i = 0, rect = (RECT *)data->Buffer; i < data->rdh.nCount; i++, rect++)
4209 printf( " (%d,%d)-(%d,%d)", rect->left, rect->top, rect->right, rect->bottom );
4211 HeapFree( GetProcessHeap(), 0, data );
4214 static void check_update_rgn( HWND hwnd, HRGN hrgn )
4218 HRGN tmp = CreateRectRgn( 0, 0, 0, 0 );
4219 HRGN update = CreateRectRgn( 0, 0, 0, 0 );
4221 ret = GetUpdateRgn( hwnd, update, FALSE );
4222 ok( ret != ERROR, "GetUpdateRgn failed\n" );
4223 if (ret == NULLREGION)
4225 ok( !hrgn, "Update region shouldn't be empty\n" );
4229 if (CombineRgn( tmp, hrgn, update, RGN_XOR ) != NULLREGION)
4231 ok( 0, "Regions are different\n" );
4232 if (winetest_debug > 0)
4234 printf( "Update region: " );
4235 dump_region( update );
4236 printf( "Wanted region: " );
4237 dump_region( hrgn );
4241 GetRgnBox( update, &r1 );
4242 GetUpdateRect( hwnd, &r2, FALSE );
4243 ok( r1.left == r2.left && r1.top == r2.top && r1.right == r2.right && r1.bottom == r2.bottom,
4244 "Rectangles are different: %d,%d-%d,%d / %d,%d-%d,%d\n",
4245 r1.left, r1.top, r1.right, r1.bottom, r2.left, r2.top, r2.right, r2.bottom );
4247 DeleteObject( tmp );
4248 DeleteObject( update );
4251 static const struct message WmInvalidateRgn[] = {
4252 { WM_NCPAINT, sent },
4253 { WM_GETTEXT, sent|defwinproc|optional },
4257 static const struct message WmGetUpdateRect[] = {
4258 { WM_NCPAINT, sent },
4259 { WM_GETTEXT, sent|defwinproc|optional },
4264 static const struct message WmInvalidateFull[] = {
4265 { WM_NCPAINT, sent|wparam, 1 },
4266 { WM_GETTEXT, sent|defwinproc|optional },
4270 static const struct message WmInvalidateErase[] = {
4271 { WM_NCPAINT, sent|wparam, 1 },
4272 { WM_GETTEXT, sent|defwinproc|optional },
4273 { WM_ERASEBKGND, sent },
4277 static const struct message WmInvalidatePaint[] = {
4279 { WM_NCPAINT, sent|wparam|beginpaint, 1 },
4280 { WM_GETTEXT, sent|beginpaint|defwinproc|optional },
4284 static const struct message WmInvalidateErasePaint[] = {
4286 { WM_NCPAINT, sent|wparam|beginpaint, 1 },
4287 { WM_GETTEXT, sent|beginpaint|defwinproc|optional },
4288 { WM_ERASEBKGND, sent|beginpaint },
4292 static const struct message WmInvalidateErasePaint2[] = {
4294 { WM_NCPAINT, sent|beginpaint },
4295 { WM_GETTEXT, sent|beginpaint|defwinproc|optional },
4296 { WM_ERASEBKGND, sent|beginpaint },
4300 static const struct message WmErase[] = {
4301 { WM_ERASEBKGND, sent },
4305 static const struct message WmPaint[] = {
4310 static const struct message WmParentOnlyPaint[] = {
4311 { WM_PAINT, sent|parent },
4315 static const struct message WmInvalidateParent[] = {
4316 { WM_NCPAINT, sent|parent },
4317 { WM_GETTEXT, sent|defwinproc|parent|optional },
4318 { WM_ERASEBKGND, sent|parent },
4322 static const struct message WmInvalidateParentChild[] = {
4323 { WM_NCPAINT, sent|parent },
4324 { WM_GETTEXT, sent|defwinproc|parent|optional },
4325 { WM_ERASEBKGND, sent|parent },
4326 { WM_NCPAINT, sent },
4327 { WM_GETTEXT, sent|defwinproc|optional },
4328 { WM_ERASEBKGND, sent },
4332 static const struct message WmInvalidateParentChild2[] = {
4333 { WM_ERASEBKGND, sent|parent },
4334 { WM_NCPAINT, sent },
4335 { WM_GETTEXT, sent|defwinproc|optional },
4336 { WM_ERASEBKGND, sent },
4340 static const struct message WmParentPaint[] = {
4341 { WM_PAINT, sent|parent },
4346 static const struct message WmParentPaintNc[] = {
4347 { WM_PAINT, sent|parent },
4349 { WM_NCPAINT, sent|beginpaint },
4350 { WM_GETTEXT, sent|beginpaint|defwinproc|optional },
4351 { WM_ERASEBKGND, sent|beginpaint },
4355 static const struct message WmChildPaintNc[] = {
4357 { WM_NCPAINT, sent|beginpaint },
4358 { WM_GETTEXT, sent|beginpaint|defwinproc|optional },
4359 { WM_ERASEBKGND, sent|beginpaint },
4363 static const struct message WmParentErasePaint[] = {
4364 { WM_PAINT, sent|parent },
4365 { WM_NCPAINT, sent|parent|beginpaint },
4366 { WM_GETTEXT, sent|parent|beginpaint|defwinproc|optional },
4367 { WM_ERASEBKGND, sent|parent|beginpaint },
4369 { WM_NCPAINT, sent|beginpaint },
4370 { WM_GETTEXT, sent|beginpaint|defwinproc|optional },
4371 { WM_ERASEBKGND, sent|beginpaint },
4375 static const struct message WmParentOnlyNcPaint[] = {
4376 { WM_PAINT, sent|parent },
4377 { WM_NCPAINT, sent|parent|beginpaint },
4378 { WM_GETTEXT, sent|parent|beginpaint|defwinproc|optional },
4382 static const struct message WmSetParentStyle[] = {
4383 { WM_STYLECHANGING, sent|parent },
4384 { WM_STYLECHANGED, sent|parent },
4388 static void test_paint_messages(void)
4394 HWND hparent, hchild;
4395 HRGN hrgn = CreateRectRgn( 0, 0, 0, 0 );
4396 HRGN hrgn2 = CreateRectRgn( 0, 0, 0, 0 );
4397 HWND hwnd = CreateWindowExA(0, "TestWindowClass", "Test overlapped", WS_OVERLAPPEDWINDOW,
4398 100, 100, 200, 200, 0, 0, 0, NULL);
4399 ok (hwnd != 0, "Failed to create overlapped window\n");
4401 ShowWindow( hwnd, SW_SHOW );
4402 UpdateWindow( hwnd );
4405 check_update_rgn( hwnd, 0 );
4406 SetRectRgn( hrgn, 10, 10, 20, 20 );
4407 ret = RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE );
4408 ok(ret, "RedrawWindow returned %d instead of TRUE\n", ret);
4409 check_update_rgn( hwnd, hrgn );
4410 SetRectRgn( hrgn2, 20, 20, 30, 30 );
4411 ret = RedrawWindow( hwnd, NULL, hrgn2, RDW_INVALIDATE );
4412 ok(ret, "RedrawWindow returned %d instead of TRUE\n", ret);
4413 CombineRgn( hrgn, hrgn, hrgn2, RGN_OR );
4414 check_update_rgn( hwnd, hrgn );
4415 /* validate everything */
4416 ret = RedrawWindow( hwnd, NULL, NULL, RDW_VALIDATE );
4417 ok(ret, "RedrawWindow returned %d instead of TRUE\n", ret);
4418 check_update_rgn( hwnd, 0 );
4420 /* test empty region */
4421 SetRectRgn( hrgn, 10, 10, 10, 15 );
4422 ret = RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE );
4423 ok(ret, "RedrawWindow returned %d instead of TRUE\n", ret);
4424 check_update_rgn( hwnd, 0 );
4425 /* test empty rect */
4426 SetRect( &rect, 10, 10, 10, 15 );
4427 ret = RedrawWindow( hwnd, &rect, NULL, RDW_INVALIDATE );
4428 ok(ret, "RedrawWindow returned %d instead of TRUE\n", ret);
4429 check_update_rgn( hwnd, 0 );
4431 /* flush pending messages */
4432 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4435 GetClientRect( hwnd, &rect );
4436 SetRectRgn( hrgn, 0, 0, rect.right - rect.left, rect.bottom - rect.top );
4437 /* MSDN: if hwnd parameter is NULL, InvalidateRect invalidates and redraws
4438 * all windows and sends WM_ERASEBKGND and WM_NCPAINT.
4440 trace("testing InvalidateRect(0, NULL, FALSE)\n");
4441 SetRectEmpty( &rect );
4442 ok(InvalidateRect(0, &rect, FALSE), "InvalidateRect(0, &rc, FALSE) should fail\n");
4443 check_update_rgn( hwnd, hrgn );
4444 ok_sequence( WmInvalidateErase, "InvalidateErase", FALSE );
4445 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4446 ok_sequence( WmPaint, "Paint", FALSE );
4447 RedrawWindow( hwnd, NULL, NULL, RDW_VALIDATE );
4448 check_update_rgn( hwnd, 0 );
4450 /* MSDN: if hwnd parameter is NULL, ValidateRect invalidates and redraws
4451 * all windows and sends WM_ERASEBKGND and WM_NCPAINT.
4453 trace("testing ValidateRect(0, NULL)\n");
4454 SetRectEmpty( &rect );
4455 ok(ValidateRect(0, &rect), "ValidateRect(0, &rc) should not fail\n");
4456 check_update_rgn( hwnd, hrgn );
4457 ok_sequence( WmInvalidateErase, "InvalidateErase", FALSE );
4458 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4459 ok_sequence( WmPaint, "Paint", FALSE );
4460 RedrawWindow( hwnd, NULL, NULL, RDW_VALIDATE );
4461 check_update_rgn( hwnd, 0 );
4463 trace("testing InvalidateRgn(0, NULL, FALSE)\n");
4464 SetLastError(0xdeadbeef);
4465 ok(!InvalidateRgn(0, NULL, FALSE), "InvalidateRgn(0, NULL, FALSE) should fail\n");
4466 ok(GetLastError() == ERROR_INVALID_WINDOW_HANDLE, "wrong error code %d\n", GetLastError());
4467 check_update_rgn( hwnd, 0 );
4468 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4469 ok_sequence( WmEmptySeq, "WmEmptySeq", FALSE );
4471 trace("testing ValidateRgn(0, NULL)\n");
4472 SetLastError(0xdeadbeef);
4473 ok(!ValidateRgn(0, NULL), "ValidateRgn(0, NULL) should fail\n");
4474 ok(GetLastError() == ERROR_INVALID_WINDOW_HANDLE, "wrong error code %d\n", GetLastError());
4475 check_update_rgn( hwnd, 0 );
4476 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4477 ok_sequence( WmEmptySeq, "WmEmptySeq", FALSE );
4479 /* now with frame */
4480 SetRectRgn( hrgn, -5, -5, 20, 20 );
4482 /* flush pending messages */
4483 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4486 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_FRAME );
4487 ok_sequence( WmEmptySeq, "EmptySeq", FALSE );
4489 SetRectRgn( hrgn, 0, 0, 20, 20 ); /* GetUpdateRgn clips to client area */
4490 check_update_rgn( hwnd, hrgn );
4493 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_FRAME | RDW_ERASENOW );
4494 ok_sequence( WmInvalidateRgn, "InvalidateRgn", FALSE );
4497 RedrawWindow( hwnd, NULL, NULL, RDW_INVALIDATE | RDW_FRAME | RDW_ERASENOW );
4498 ok_sequence( WmInvalidateFull, "InvalidateFull", FALSE );
4500 GetClientRect( hwnd, &rect );
4501 SetRectRgn( hrgn, rect.left, rect.top, rect.right, rect.bottom );
4502 check_update_rgn( hwnd, hrgn );
4505 RedrawWindow( hwnd, NULL, NULL, RDW_INVALIDATE | RDW_FRAME | RDW_ERASE | RDW_ERASENOW );
4506 ok_sequence( WmInvalidateErase, "InvalidateErase", FALSE );
4509 RedrawWindow( hwnd, NULL, NULL, RDW_INVALIDATE | RDW_FRAME | RDW_ERASENOW | RDW_UPDATENOW );
4510 ok_sequence( WmInvalidatePaint, "InvalidatePaint", FALSE );
4511 check_update_rgn( hwnd, 0 );
4514 RedrawWindow( hwnd, NULL, NULL, RDW_INVALIDATE | RDW_FRAME | RDW_ERASE | RDW_UPDATENOW );
4515 ok_sequence( WmInvalidateErasePaint, "InvalidateErasePaint", FALSE );
4516 check_update_rgn( hwnd, 0 );
4519 SetRectRgn( hrgn, 0, 0, 100, 100 );
4520 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE );
4521 SetRectRgn( hrgn, 0, 0, 50, 100 );
4522 RedrawWindow( hwnd, NULL, hrgn, RDW_VALIDATE );
4523 SetRectRgn( hrgn, 50, 0, 100, 100 );
4524 check_update_rgn( hwnd, hrgn );
4525 RedrawWindow( hwnd, NULL, hrgn, RDW_VALIDATE | RDW_ERASENOW );
4526 ok_sequence( WmEmptySeq, "EmptySeq", FALSE ); /* must not generate messages, everything is valid */
4527 check_update_rgn( hwnd, 0 );
4530 SetRectRgn( hrgn, 0, 0, 100, 100 );
4531 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_ERASE );
4532 SetRectRgn( hrgn, 0, 0, 100, 50 );
4533 RedrawWindow( hwnd, NULL, hrgn, RDW_VALIDATE | RDW_ERASENOW );
4534 ok_sequence( WmErase, "Erase", FALSE );
4535 SetRectRgn( hrgn, 0, 50, 100, 100 );
4536 check_update_rgn( hwnd, hrgn );
4539 SetRectRgn( hrgn, 0, 0, 100, 100 );
4540 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_ERASE );
4541 SetRectRgn( hrgn, 0, 0, 50, 50 );
4542 RedrawWindow( hwnd, NULL, hrgn, RDW_VALIDATE | RDW_NOERASE | RDW_UPDATENOW );
4543 ok_sequence( WmPaint, "Paint", FALSE );
4546 SetRectRgn( hrgn, -4, -4, -2, -2 );
4547 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_FRAME );
4548 SetRectRgn( hrgn, -200, -200, -198, -198 );
4549 RedrawWindow( hwnd, NULL, hrgn, RDW_VALIDATE | RDW_NOFRAME | RDW_ERASENOW );
4550 ok_sequence( WmEmptySeq, "EmptySeq", FALSE );
4553 SetRectRgn( hrgn, -4, -4, -2, -2 );
4554 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_FRAME );
4555 SetRectRgn( hrgn, -4, -4, -3, -3 );
4556 RedrawWindow( hwnd, NULL, hrgn, RDW_VALIDATE | RDW_NOFRAME );
4557 SetRectRgn( hrgn, 0, 0, 1, 1 );
4558 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_UPDATENOW );
4559 ok_sequence( WmPaint, "Paint", FALSE );
4562 SetRectRgn( hrgn, -4, -4, -1, -1 );
4563 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_FRAME );
4564 RedrawWindow( hwnd, NULL, 0, RDW_ERASENOW );
4565 /* make sure no WM_PAINT was generated */
4566 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4567 ok_sequence( WmInvalidateRgn, "InvalidateRgn", FALSE );
4570 SetRectRgn( hrgn, -4, -4, -1, -1 );
4571 RedrawWindow( hwnd, NULL, hrgn, RDW_INVALIDATE | RDW_FRAME );
4572 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE ))
4574 if (msg.hwnd == hwnd && msg.message == WM_PAINT)
4576 /* GetUpdateRgn must return empty region since only nonclient area is invalidated */
4577 INT ret = GetUpdateRgn( hwnd, hrgn, FALSE );
4578 ok( ret == NULLREGION, "Invalid GetUpdateRgn result %d\n", ret );
4579 ret = GetUpdateRect( hwnd, &rect, FALSE );
4580 ok( ret, "Invalid GetUpdateRect result %d\n", ret );
4581 /* this will send WM_NCPAINT and validate the non client area */
4582 ret = GetUpdateRect( hwnd, &rect, TRUE );
4583 ok( !ret, "Invalid GetUpdateRect result %d\n", ret );
4585 DispatchMessage( &msg );
4587 ok_sequence( WmGetUpdateRect, "GetUpdateRect", FALSE );
4589 DestroyWindow( hwnd );
4591 /* now test with a child window */
4593 hparent = CreateWindowExA(0, "TestParentClass", "Test parent", WS_OVERLAPPEDWINDOW,
4594 100, 100, 200, 200, 0, 0, 0, NULL);
4595 ok (hparent != 0, "Failed to create parent window\n");
4597 hchild = CreateWindowExA(0, "TestWindowClass", "Test child", WS_CHILD | WS_VISIBLE | WS_BORDER,
4598 10, 10, 100, 100, hparent, 0, 0, NULL);
4599 ok (hchild != 0, "Failed to create child window\n");
4601 ShowWindow( hparent, SW_SHOW );
4602 UpdateWindow( hparent );
4603 UpdateWindow( hchild );
4606 log_all_parent_messages++;
4608 SetRect( &rect, 0, 0, 50, 50 );
4609 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4610 RedrawWindow( hparent, NULL, 0, RDW_ERASENOW | RDW_ALLCHILDREN );
4611 ok_sequence( WmInvalidateParentChild, "InvalidateParentChild", FALSE );
4613 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4615 MapWindowPoints( hchild, hparent, &pt, 1 );
4616 SetRectRgn( hrgn, 0, 0, 50 - pt.x, 50 - pt.y );
4617 check_update_rgn( hchild, hrgn );
4618 SetRectRgn( hrgn, 0, 0, 50, 50 );
4619 check_update_rgn( hparent, hrgn );
4620 RedrawWindow( hparent, NULL, 0, RDW_ERASENOW );
4621 ok_sequence( WmInvalidateParent, "InvalidateParent", FALSE );
4622 RedrawWindow( hchild, NULL, 0, RDW_ERASENOW );
4623 ok_sequence( WmEmptySeq, "EraseNow child", FALSE );
4625 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4626 ok_sequence( WmParentPaintNc, "WmParentPaintNc", FALSE );
4628 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME | RDW_ALLCHILDREN );
4629 RedrawWindow( hparent, NULL, 0, RDW_ERASENOW );
4630 ok_sequence( WmInvalidateParent, "InvalidateParent2", FALSE );
4631 RedrawWindow( hchild, NULL, 0, RDW_ERASENOW );
4632 ok_sequence( WmEmptySeq, "EraseNow child", FALSE );
4634 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE );
4635 RedrawWindow( hparent, NULL, 0, RDW_ERASENOW | RDW_ALLCHILDREN );
4636 ok_sequence( WmInvalidateParentChild2, "InvalidateParentChild2", FALSE );
4638 SetWindowLong( hparent, GWL_STYLE, GetWindowLong(hparent,GWL_STYLE) | WS_CLIPCHILDREN );
4640 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME | RDW_ALLCHILDREN );
4641 RedrawWindow( hparent, NULL, 0, RDW_ERASENOW );
4642 ok_sequence( WmInvalidateParentChild, "InvalidateParentChild3", FALSE );
4644 /* flush all paint messages */
4645 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4648 /* RDW_UPDATENOW on child with WS_CLIPCHILDREN doesn't change corresponding parent area */
4649 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME | RDW_ALLCHILDREN );
4650 SetRectRgn( hrgn, 0, 0, 50, 50 );
4651 check_update_rgn( hparent, hrgn );
4652 RedrawWindow( hchild, NULL, 0, RDW_UPDATENOW );
4653 ok_sequence( WmInvalidateErasePaint2, "WmInvalidateErasePaint2", FALSE );
4654 SetRectRgn( hrgn, 0, 0, 50, 50 );
4655 check_update_rgn( hparent, hrgn );
4657 /* flush all paint messages */
4658 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4659 SetWindowLong( hparent, GWL_STYLE, GetWindowLong(hparent,GWL_STYLE) & ~WS_CLIPCHILDREN );
4662 /* RDW_UPDATENOW on child without WS_CLIPCHILDREN will validate corresponding parent area */
4663 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4664 SetRectRgn( hrgn, 0, 0, 50, 50 );
4665 check_update_rgn( hparent, hrgn );
4666 RedrawWindow( hchild, NULL, 0, RDW_UPDATENOW );
4667 ok_sequence( WmInvalidateErasePaint2, "WmInvalidateErasePaint2", FALSE );
4668 SetRectRgn( hrgn2, 10, 10, 50, 50 );
4669 CombineRgn( hrgn, hrgn, hrgn2, RGN_DIFF );
4670 check_update_rgn( hparent, hrgn );
4671 /* flush all paint messages */
4672 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4675 /* same as above but parent gets completely validated */
4676 SetRect( &rect, 20, 20, 30, 30 );
4677 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4678 SetRectRgn( hrgn, 20, 20, 30, 30 );
4679 check_update_rgn( hparent, hrgn );
4680 RedrawWindow( hchild, NULL, 0, RDW_UPDATENOW );
4681 ok_sequence( WmInvalidateErasePaint2, "WmInvalidateErasePaint2", FALSE );
4682 check_update_rgn( hparent, 0 ); /* no update region */
4683 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4684 ok_sequence( WmEmptySeq, "WmEmpty", FALSE ); /* and no paint messages */
4686 /* make sure RDW_VALIDATE on child doesn't have the same effect */
4688 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4689 SetRectRgn( hrgn, 20, 20, 30, 30 );
4690 check_update_rgn( hparent, hrgn );
4691 RedrawWindow( hchild, NULL, 0, RDW_VALIDATE | RDW_NOERASE );
4692 SetRectRgn( hrgn, 20, 20, 30, 30 );
4693 check_update_rgn( hparent, hrgn );
4695 /* same as above but normal WM_PAINT doesn't validate parent */
4697 SetRect( &rect, 20, 20, 30, 30 );
4698 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4699 SetRectRgn( hrgn, 20, 20, 30, 30 );
4700 check_update_rgn( hparent, hrgn );
4701 /* no WM_PAINT in child while parent still pending */
4702 while (PeekMessage( &msg, hchild, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4703 ok_sequence( WmEmptySeq, "No WM_PAINT", FALSE );
4704 while (PeekMessage( &msg, hparent, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4705 ok_sequence( WmParentErasePaint, "WmParentErasePaint", FALSE );
4708 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4709 /* no WM_PAINT in child while parent still pending */
4710 while (PeekMessage( &msg, hchild, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4711 ok_sequence( WmEmptySeq, "No WM_PAINT", FALSE );
4712 RedrawWindow( hparent, &rect, 0, RDW_VALIDATE | RDW_NOERASE | RDW_NOCHILDREN );
4713 /* now that parent is valid child should get WM_PAINT */
4714 while (PeekMessage( &msg, hchild, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4715 ok_sequence( WmInvalidateErasePaint2, "WmInvalidateErasePaint2", FALSE );
4716 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4717 ok_sequence( WmEmptySeq, "No other message", FALSE );
4719 /* same thing with WS_CLIPCHILDREN in parent */
4721 SetWindowLong( hparent, GWL_STYLE, GetWindowLong(hparent,GWL_STYLE) | WS_CLIPCHILDREN );
4722 ok_sequence( WmSetParentStyle, "WmSetParentStyle", FALSE );
4723 /* changing style invalidates non client area, but we need to invalidate something else to see it */
4724 RedrawWindow( hparent, &rect, 0, RDW_UPDATENOW );
4725 ok_sequence( WmEmptySeq, "No message", FALSE );
4726 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_UPDATENOW );
4727 ok_sequence( WmParentOnlyNcPaint, "WmParentOnlyNcPaint", FALSE );
4730 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ERASE | RDW_ALLCHILDREN );
4731 SetRectRgn( hrgn, 20, 20, 30, 30 );
4732 check_update_rgn( hparent, hrgn );
4733 /* no WM_PAINT in child while parent still pending */
4734 while (PeekMessage( &msg, hchild, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4735 ok_sequence( WmEmptySeq, "No WM_PAINT", FALSE );
4736 /* WM_PAINT in parent first */
4737 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4738 ok_sequence( WmParentPaintNc, "WmParentPaintNc2", FALSE );
4740 /* no RDW_ERASE in parent still causes RDW_ERASE and RDW_FRAME in child */
4742 SetRect( &rect, 0, 0, 30, 30 );
4743 RedrawWindow( hparent, &rect, 0, RDW_INVALIDATE | RDW_ALLCHILDREN );
4744 SetRectRgn( hrgn, 0, 0, 30, 30 );
4745 check_update_rgn( hparent, hrgn );
4746 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4747 ok_sequence( WmParentPaintNc, "WmParentPaintNc3", FALSE );
4749 /* validate doesn't cause RDW_NOERASE or RDW_NOFRAME in child */
4751 SetRect( &rect, -10, 0, 30, 30 );
4752 RedrawWindow( hchild, &rect, 0, RDW_INVALIDATE | RDW_FRAME | RDW_ERASE );
4753 SetRect( &rect, 0, 0, 20, 20 );
4754 RedrawWindow( hparent, &rect, 0, RDW_VALIDATE | RDW_ALLCHILDREN );
4755 RedrawWindow( hparent, NULL, 0, RDW_UPDATENOW );
4756 ok_sequence( WmChildPaintNc, "WmChildPaintNc", FALSE );
4758 /* validate doesn't cause RDW_NOERASE or RDW_NOFRAME in child */
4760 SetRect( &rect, -10, 0, 30, 30 );
4761 RedrawWindow( hchild, &rect, 0, RDW_INVALIDATE | RDW_FRAME | RDW_ERASE );
4762 SetRect( &rect, 0, 0, 100, 100 );
4763 RedrawWindow( hparent, &rect, 0, RDW_VALIDATE | RDW_ALLCHILDREN );
4764 RedrawWindow( hparent, NULL, 0, RDW_UPDATENOW );
4765 ok_sequence( WmEmptySeq, "WmChildPaintNc2", FALSE );
4766 RedrawWindow( hparent, NULL, 0, RDW_ERASENOW );
4767 ok_sequence( WmEmptySeq, "WmChildPaintNc3", FALSE );
4769 /* test RDW_INTERNALPAINT behavior */
4772 RedrawWindow( hparent, NULL, 0, RDW_INTERNALPAINT | RDW_NOCHILDREN );
4773 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4774 ok_sequence( WmParentOnlyPaint, "WmParentOnlyPaint", FALSE );
4776 RedrawWindow( hparent, NULL, 0, RDW_INTERNALPAINT | RDW_ALLCHILDREN );
4777 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4778 ok_sequence( WmParentPaint, "WmParentPaint", FALSE );
4780 RedrawWindow( hparent, NULL, 0, RDW_INTERNALPAINT );
4781 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4782 ok_sequence( WmParentOnlyPaint, "WmParentOnlyPaint", FALSE );
4784 assert( GetWindowLong(hparent, GWL_STYLE) & WS_CLIPCHILDREN );
4785 UpdateWindow( hparent );
4786 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4788 trace("testing SWP_FRAMECHANGED on parent with WS_CLIPCHILDREN\n");
4789 RedrawWindow( hchild, NULL, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4790 SetWindowPos( hparent, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE |
4791 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED );
4792 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4793 ok_sequence(WmSWP_FrameChanged_clip, "SetWindowPos:FrameChanged_clip", FALSE );
4795 UpdateWindow( hparent );
4796 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4798 trace("testing SWP_FRAMECHANGED|SWP_DEFERERASE on parent with WS_CLIPCHILDREN\n");
4799 RedrawWindow( hchild, NULL, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4800 SetWindowPos( hparent, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE | SWP_DEFERERASE |
4801 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED );
4802 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4803 ok_sequence(WmSWP_FrameChangedDeferErase, "SetWindowPos:FrameChangedDeferErase", FALSE );
4805 SetWindowLong( hparent, GWL_STYLE, GetWindowLong(hparent,GWL_STYLE) & ~WS_CLIPCHILDREN );
4806 ok_sequence( WmSetParentStyle, "WmSetParentStyle", FALSE );
4807 RedrawWindow( hparent, NULL, 0, RDW_INTERNALPAINT );
4808 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4809 ok_sequence( WmParentPaint, "WmParentPaint", FALSE );
4811 assert( !(GetWindowLong(hparent, GWL_STYLE) & WS_CLIPCHILDREN) );
4812 UpdateWindow( hparent );
4813 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4815 trace("testing SWP_FRAMECHANGED on parent without WS_CLIPCHILDREN\n");
4816 RedrawWindow( hchild, NULL, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4817 SetWindowPos( hparent, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE |
4818 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED );
4819 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4820 ok_sequence(WmSWP_FrameChanged_noclip, "SetWindowPos:FrameChanged_noclip", FALSE );
4822 UpdateWindow( hparent );
4823 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4825 trace("testing SWP_FRAMECHANGED|SWP_DEFERERASE on parent without WS_CLIPCHILDREN\n");
4826 RedrawWindow( hchild, NULL, 0, RDW_INVALIDATE | RDW_ERASE | RDW_FRAME );
4827 SetWindowPos( hparent, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE | SWP_DEFERERASE |
4828 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED );
4829 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
4830 ok_sequence(WmSWP_FrameChangedDeferErase, "SetWindowPos:FrameChangedDeferErase", FALSE );
4832 log_all_parent_messages--;
4833 DestroyWindow( hparent );
4834 ok(!IsWindow(hchild), "child must be destroyed with its parent\n");
4836 DeleteObject( hrgn );
4837 DeleteObject( hrgn2 );
4846 static DWORD WINAPI thread_proc(void *param)
4849 struct wnd_event *wnd_event = (struct wnd_event *)param;
4851 wnd_event->hwnd = CreateWindowExA(0, "TestWindowClass", "window caption text", WS_OVERLAPPEDWINDOW,
4852 100, 100, 200, 200, 0, 0, 0, NULL);
4853 ok(wnd_event->hwnd != 0, "Failed to create overlapped window\n");
4855 SetEvent(wnd_event->event);
4857 while (GetMessage(&msg, 0, 0, 0))
4859 TranslateMessage(&msg);
4860 DispatchMessage(&msg);
4863 ok(IsWindow(wnd_event->hwnd), "window should still exist\n");
4868 static void test_interthread_messages(void)
4875 int len, expected_len;
4876 struct wnd_event wnd_event;
4879 wnd_event.event = CreateEventW(NULL, 0, 0, NULL);
4880 if (!wnd_event.event)
4882 trace("skipping interthread message test under win9x\n");
4886 hThread = CreateThread(NULL, 0, thread_proc, &wnd_event, 0, &tid);
4887 ok(hThread != NULL, "CreateThread failed, error %d\n", GetLastError());
4889 ok(WaitForSingleObject(wnd_event.event, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
4891 CloseHandle(wnd_event.event);
4893 SetLastError(0xdeadbeef);
4894 ok(!DestroyWindow(wnd_event.hwnd), "DestroyWindow succeded\n");
4895 ok(GetLastError() == ERROR_ACCESS_DENIED, "wrong error code %d\n", GetLastError());
4897 proc = (WNDPROC)GetWindowLongPtrA(wnd_event.hwnd, GWLP_WNDPROC);
4898 ok(proc != NULL, "GetWindowLongPtrA(GWLP_WNDPROC) error %d\n", GetLastError());
4900 expected_len = lstrlenA("window caption text");
4901 memset(buf, 0, sizeof(buf));
4902 SetLastError(0xdeadbeef);
4903 len = CallWindowProcA(proc, wnd_event.hwnd, WM_GETTEXT, sizeof(buf), (LPARAM)buf);
4904 ok(len == expected_len, "CallWindowProcA(WM_GETTEXT) error %d, len %d, expected len %d\n", GetLastError(), len, expected_len);
4905 ok(!lstrcmpA(buf, "window caption text"), "window text mismatch\n");
4907 msg.hwnd = wnd_event.hwnd;
4908 msg.message = WM_GETTEXT;
4909 msg.wParam = sizeof(buf);
4910 msg.lParam = (LPARAM)buf;
4911 memset(buf, 0, sizeof(buf));
4912 SetLastError(0xdeadbeef);
4913 len = DispatchMessageA(&msg);
4914 ok(!len && GetLastError() == ERROR_MESSAGE_SYNC_ONLY,
4915 "DispatchMessageA(WM_GETTEXT) succeded on another thread window: ret %d, error %d\n", len, GetLastError());
4917 /* the following test causes an exception in user.exe under win9x */
4918 msg.hwnd = wnd_event.hwnd;
4919 msg.message = WM_TIMER;
4921 msg.lParam = GetWindowLongPtrA(wnd_event.hwnd, GWLP_WNDPROC);
4922 SetLastError(0xdeadbeef);
4923 len = DispatchMessageA(&msg);
4924 ok(!len && GetLastError() == 0xdeadbeef,
4925 "DispatchMessageA(WM_TIMER) failed on another thread window: ret %d, error %d\n", len, GetLastError());
4927 ret = PostMessageA(wnd_event.hwnd, WM_QUIT, 0, 0);
4928 ok( ret, "PostMessageA(WM_QUIT) error %d\n", GetLastError());
4930 ok(WaitForSingleObject(hThread, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
4931 CloseHandle(hThread);
4933 ok(!IsWindow(wnd_event.hwnd), "window should be destroyed on thread exit\n");
4937 static const struct message WmVkN[] = {
4938 { WM_KEYDOWN, wparam|lparam, 'N', 1 },
4939 { WM_KEYDOWN, sent|wparam|lparam, 'N', 1 },
4940 { WM_CHAR, wparam|lparam, 'n', 1 },
4941 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1002,1), 0 },
4942 { WM_KEYUP, wparam|lparam, 'N', 0xc0000001 },
4943 { WM_KEYUP, sent|wparam|lparam, 'N', 0xc0000001 },
4946 static const struct message WmShiftVkN[] = {
4947 { WM_KEYDOWN, wparam|lparam, VK_SHIFT, 1 },
4948 { WM_KEYDOWN, sent|wparam|lparam, VK_SHIFT, 1 },
4949 { WM_KEYDOWN, wparam|lparam, 'N', 1 },
4950 { WM_KEYDOWN, sent|wparam|lparam, 'N', 1 },
4951 { WM_CHAR, wparam|lparam, 'N', 1 },
4952 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1001,1), 0 },
4953 { WM_KEYUP, wparam|lparam, 'N', 0xc0000001 },
4954 { WM_KEYUP, sent|wparam|lparam, 'N', 0xc0000001 },
4955 { WM_KEYUP, wparam|lparam, VK_SHIFT, 0xc0000001 },
4956 { WM_KEYUP, sent|wparam|lparam, VK_SHIFT, 0xc0000001 },
4959 static const struct message WmCtrlVkN[] = {
4960 { WM_KEYDOWN, wparam|lparam, VK_CONTROL, 1 },
4961 { WM_KEYDOWN, sent|wparam|lparam, VK_CONTROL, 1 },
4962 { WM_KEYDOWN, wparam|lparam, 'N', 1 },
4963 { WM_KEYDOWN, sent|wparam|lparam, 'N', 1 },
4964 { WM_CHAR, wparam|lparam, 0x000e, 1 },
4965 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1000,1), 0 },
4966 { WM_KEYUP, wparam|lparam, 'N', 0xc0000001 },
4967 { WM_KEYUP, sent|wparam|lparam, 'N', 0xc0000001 },
4968 { WM_KEYUP, wparam|lparam, VK_CONTROL, 0xc0000001 },
4969 { WM_KEYUP, sent|wparam|lparam, VK_CONTROL, 0xc0000001 },
4972 static const struct message WmCtrlVkN_2[] = {
4973 { WM_KEYDOWN, wparam|lparam, VK_CONTROL, 1 },
4974 { WM_KEYDOWN, sent|wparam|lparam, VK_CONTROL, 1 },
4975 { WM_KEYDOWN, wparam|lparam, 'N', 1 },
4976 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1000,1), 0 },
4977 { WM_KEYUP, wparam|lparam, 'N', 0xc0000001 },
4978 { WM_KEYUP, sent|wparam|lparam, 'N', 0xc0000001 },
4979 { WM_KEYUP, wparam|lparam, VK_CONTROL, 0xc0000001 },
4980 { WM_KEYUP, sent|wparam|lparam, VK_CONTROL, 0xc0000001 },
4983 static const struct message WmAltVkN[] = {
4984 { WM_SYSKEYDOWN, wparam|lparam, VK_MENU, 0x20000001 },
4985 { WM_SYSKEYDOWN, sent|wparam|lparam, VK_MENU, 0x20000001 },
4986 { WM_SYSKEYDOWN, wparam|lparam, 'N', 0x20000001 },
4987 { WM_SYSKEYDOWN, sent|wparam|lparam, 'N', 0x20000001 },
4988 { WM_SYSCHAR, wparam|lparam, 'n', 0x20000001 },
4989 { WM_SYSCHAR, sent|wparam|lparam, 'n', 0x20000001 },
4990 { WM_SYSCOMMAND, sent|defwinproc|wparam|lparam, SC_KEYMENU, 'n' },
4991 { HCBT_SYSCOMMAND, hook },
4992 { WM_ENTERMENULOOP, sent|defwinproc|wparam|lparam, 0, 0 },
4993 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 },
4994 { 0x00AE, sent|defwinproc|optional }, /* XP */
4995 { WM_GETTEXT, sent|defwinproc|optional }, /* XP */
4996 { WM_INITMENU, sent|defwinproc },
4997 { EVENT_SYSTEM_MENUSTART, winevent_hook|wparam|lparam, OBJID_SYSMENU, 0 },
4998 { WM_MENUCHAR, sent|defwinproc|wparam, MAKEWPARAM('n',MF_SYSMENU) },
4999 { EVENT_SYSTEM_CAPTUREEND, winevent_hook|wparam|lparam, 0, 0 },
5000 { WM_CAPTURECHANGED, sent|defwinproc },
5001 { WM_MENUSELECT, sent|defwinproc|wparam, MAKEWPARAM(0,0xffff) },
5002 { EVENT_SYSTEM_MENUEND, winevent_hook|wparam|lparam, OBJID_SYSMENU, 0 },
5003 { WM_EXITMENULOOP, sent|defwinproc },
5004 { WM_MENUSELECT, sent|defwinproc|wparam|optional, MAKEWPARAM(0,0xffff) }, /* Win95 bug */
5005 { WM_EXITMENULOOP, sent|defwinproc|optional }, /* Win95 bug */
5006 { WM_SYSKEYUP, wparam|lparam, 'N', 0xe0000001 },
5007 { WM_SYSKEYUP, sent|wparam|lparam, 'N', 0xe0000001 },
5008 { WM_KEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5009 { WM_KEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5012 static const struct message WmAltVkN_2[] = {
5013 { WM_SYSKEYDOWN, wparam|lparam, VK_MENU, 0x20000001 },
5014 { WM_SYSKEYDOWN, sent|wparam|lparam, VK_MENU, 0x20000001 },
5015 { WM_SYSKEYDOWN, wparam|lparam, 'N', 0x20000001 },
5016 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1003,1), 0 },
5017 { WM_SYSKEYUP, wparam|lparam, 'N', 0xe0000001 },
5018 { WM_SYSKEYUP, sent|wparam|lparam, 'N', 0xe0000001 },
5019 { WM_KEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5020 { WM_KEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5023 static const struct message WmCtrlAltVkN[] = {
5024 { WM_KEYDOWN, wparam|lparam, VK_CONTROL, 1 },
5025 { WM_KEYDOWN, sent|wparam|lparam, VK_CONTROL, 1 },
5026 { WM_KEYDOWN, wparam|lparam, VK_MENU, 0x20000001 },
5027 { WM_KEYDOWN, sent|wparam|lparam, VK_MENU, 0x20000001 },
5028 { WM_KEYDOWN, wparam|lparam, 'N', 0x20000001 },
5029 { WM_KEYDOWN, sent|wparam|lparam, 'N', 0x20000001 },
5030 { WM_KEYUP, wparam|lparam, 'N', 0xe0000001 },
5031 { WM_KEYUP, sent|wparam|lparam, 'N', 0xe0000001 },
5032 { WM_KEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5033 { WM_KEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5034 { WM_KEYUP, wparam|lparam, VK_CONTROL, 0xc0000001 },
5035 { WM_KEYUP, sent|wparam|lparam, VK_CONTROL, 0xc0000001 },
5038 static const struct message WmCtrlShiftVkN[] = {
5039 { WM_KEYDOWN, wparam|lparam, VK_CONTROL, 1 },
5040 { WM_KEYDOWN, sent|wparam|lparam, VK_CONTROL, 1 },
5041 { WM_KEYDOWN, wparam|lparam, VK_SHIFT, 1 },
5042 { WM_KEYDOWN, sent|wparam|lparam, VK_SHIFT, 1 },
5043 { WM_KEYDOWN, wparam|lparam, 'N', 1 },
5044 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1004,1), 0 },
5045 { WM_KEYUP, wparam|lparam, 'N', 0xc0000001 },
5046 { WM_KEYUP, sent|wparam|lparam, 'N', 0xc0000001 },
5047 { WM_KEYUP, wparam|lparam, VK_SHIFT, 0xc0000001 },
5048 { WM_KEYUP, sent|wparam|lparam, VK_SHIFT, 0xc0000001 },
5049 { WM_KEYUP, wparam|lparam, VK_CONTROL, 0xc0000001 },
5050 { WM_KEYUP, sent|wparam|lparam, VK_CONTROL, 0xc0000001 },
5053 static const struct message WmCtrlAltShiftVkN[] = {
5054 { WM_KEYDOWN, wparam|lparam, VK_CONTROL, 1 },
5055 { WM_KEYDOWN, sent|wparam|lparam, VK_CONTROL, 1 },
5056 { WM_KEYDOWN, wparam|lparam, VK_MENU, 0x20000001 },
5057 { WM_KEYDOWN, sent|wparam|lparam, VK_MENU, 0x20000001 },
5058 { WM_KEYDOWN, wparam|lparam, VK_SHIFT, 0x20000001 },
5059 { WM_KEYDOWN, sent|wparam|lparam, VK_SHIFT, 0x20000001 },
5060 { WM_KEYDOWN, wparam|lparam, 'N', 0x20000001 },
5061 { WM_COMMAND, sent|wparam|lparam, MAKEWPARAM(1005,1), 0 },
5062 { WM_KEYUP, wparam|lparam, 'N', 0xe0000001 },
5063 { WM_KEYUP, sent|wparam|lparam, 'N', 0xe0000001 },
5064 { WM_KEYUP, wparam|lparam, VK_SHIFT, 0xe0000001 },
5065 { WM_KEYUP, sent|wparam|lparam, VK_SHIFT, 0xe0000001 },
5066 { WM_KEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5067 { WM_KEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5068 { WM_KEYUP, wparam|lparam, VK_CONTROL, 0xc0000001 },
5069 { WM_KEYUP, sent|wparam|lparam, VK_CONTROL, 0xc0000001 },
5072 static const struct message WmAltPressRelease[] = {
5073 { WM_SYSKEYDOWN, wparam|lparam, VK_MENU, 0x20000001 },
5074 { WM_SYSKEYDOWN, sent|wparam|lparam, VK_MENU, 0x20000001 },
5075 { WM_SYSKEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5076 { WM_SYSKEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5077 { WM_SYSCOMMAND, sent|defwinproc|wparam|lparam, SC_KEYMENU, 0 },
5078 { HCBT_SYSCOMMAND, hook },
5079 { WM_ENTERMENULOOP, sent|defwinproc|wparam|lparam, 0, 0 },
5080 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 },
5081 { WM_INITMENU, sent|defwinproc },
5082 { EVENT_SYSTEM_MENUSTART, winevent_hook|wparam|lparam, OBJID_SYSMENU, 0 },
5083 { WM_MENUSELECT, sent|defwinproc|wparam, MAKEWPARAM(0,MF_SYSMENU|MF_POPUP|MF_HILITE) },
5084 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_SYSMENU, 1 },
5086 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_SYSMENU, 0 },
5087 { EVENT_SYSTEM_CAPTUREEND, winevent_hook|wparam|lparam, 0, 0, },
5088 { WM_CAPTURECHANGED, sent|defwinproc },
5089 { WM_MENUSELECT, sent|defwinproc|wparam|optional, MAKEWPARAM(0,0xffff) },
5090 { EVENT_SYSTEM_MENUEND, winevent_hook|wparam|lparam, OBJID_SYSMENU, 0 },
5091 { WM_EXITMENULOOP, sent|defwinproc },
5092 { WM_SYSKEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5093 { WM_SYSKEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5096 static const struct message WmAltMouseButton[] = {
5097 { WM_SYSKEYDOWN, wparam|lparam, VK_MENU, 0x20000001 },
5098 { WM_SYSKEYDOWN, sent|wparam|lparam, VK_MENU, 0x20000001 },
5099 { WM_MOUSEMOVE, wparam|optional, 0, 0 },
5100 { WM_MOUSEMOVE, sent|wparam|optional, 0, 0 },
5101 { WM_LBUTTONDOWN, wparam, MK_LBUTTON, 0 },
5102 { WM_LBUTTONDOWN, sent|wparam, MK_LBUTTON, 0 },
5103 { WM_LBUTTONUP, wparam, 0, 0 },
5104 { WM_LBUTTONUP, sent|wparam, 0, 0 },
5105 { WM_SYSKEYUP, wparam|lparam, VK_MENU, 0xc0000001 },
5106 { WM_SYSKEYUP, sent|wparam|lparam, VK_MENU, 0xc0000001 },
5109 static const struct message WmF1Seq[] = {
5110 { WM_KEYDOWN, wparam|lparam, VK_F1, 1 },
5111 { WM_KEYDOWN, sent|wparam|lparam, VK_F1, 0x00000001 },
5112 { 0x4d, wparam|lparam, 0, 0 },
5113 { 0x4d, sent|wparam|lparam, 0, 0 },
5114 { WM_HELP, sent|defwinproc },
5115 { WM_KEYUP, wparam|lparam, VK_F1, 0xc0000001 },
5116 { WM_KEYUP, sent|wparam|lparam, VK_F1, 0xc0000001 },
5120 static void pump_msg_loop(HWND hwnd, HACCEL hAccel)
5124 while (PeekMessageA(&msg, 0, 0, 0, PM_REMOVE))
5126 struct message log_msg;
5128 trace("accel: %p, %04x, %08x, %08lx\n", msg.hwnd, msg.message, msg.wParam, msg.lParam);
5130 /* ignore some unwanted messages */
5131 if (msg.message == WM_MOUSEMOVE ||
5132 msg.message == WM_GETICON ||
5133 msg.message == WM_DEVICECHANGE)
5136 log_msg.message = msg.message;
5137 log_msg.flags = wparam|lparam;
5138 log_msg.wParam = msg.wParam;
5139 log_msg.lParam = msg.lParam;
5140 add_message(&log_msg);
5142 if (!hAccel || !TranslateAccelerator(hwnd, hAccel, &msg))
5144 TranslateMessage(&msg);
5145 DispatchMessage(&msg);
5150 static void test_accelerators(void)
5155 HWND hwnd = CreateWindowExA(0, "TestWindowClass", NULL, WS_OVERLAPPEDWINDOW | WS_VISIBLE,
5156 100, 100, 200, 200, 0, 0, 0, NULL);
5163 ok(GetFocus() == hwnd, "wrong focus window %p\n", GetFocus());
5165 state = GetKeyState(VK_SHIFT);
5166 ok(!(state & 0x8000), "wrong Shift state %04x\n", state);
5167 state = GetKeyState(VK_CAPITAL);
5168 ok(state == 0, "wrong CapsLock state %04x\n", state);
5170 hAccel = LoadAccelerators(GetModuleHandleA(0), MAKEINTRESOURCE(1));
5171 assert(hAccel != 0);
5173 pump_msg_loop(hwnd, 0);
5176 trace("testing VK_N press/release\n");
5178 keybd_event('N', 0, 0, 0);
5179 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5180 pump_msg_loop(hwnd, hAccel);
5181 ok_sequence(WmVkN, "VK_N press/release", FALSE);
5183 trace("testing Shift+VK_N press/release\n");
5185 keybd_event(VK_SHIFT, 0, 0, 0);
5186 keybd_event('N', 0, 0, 0);
5187 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5188 keybd_event(VK_SHIFT, 0, KEYEVENTF_KEYUP, 0);
5189 pump_msg_loop(hwnd, hAccel);
5190 ok_sequence(WmShiftVkN, "Shift+VK_N press/release", FALSE);
5192 trace("testing Ctrl+VK_N press/release\n");
5194 keybd_event(VK_CONTROL, 0, 0, 0);
5195 keybd_event('N', 0, 0, 0);
5196 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5197 keybd_event(VK_CONTROL, 0, KEYEVENTF_KEYUP, 0);
5198 pump_msg_loop(hwnd, hAccel);
5199 ok_sequence(WmCtrlVkN, "Ctrl+VK_N press/release", FALSE);
5201 trace("testing Alt+VK_N press/release\n");
5203 keybd_event(VK_MENU, 0, 0, 0);
5204 keybd_event('N', 0, 0, 0);
5205 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5206 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5207 pump_msg_loop(hwnd, hAccel);
5208 ok_sequence(WmAltVkN, "Alt+VK_N press/release", FALSE);
5210 trace("testing Ctrl+Alt+VK_N press/release 1\n");
5212 keybd_event(VK_CONTROL, 0, 0, 0);
5213 keybd_event(VK_MENU, 0, 0, 0);
5214 keybd_event('N', 0, 0, 0);
5215 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5216 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5217 keybd_event(VK_CONTROL, 0, KEYEVENTF_KEYUP, 0);
5218 pump_msg_loop(hwnd, hAccel);
5219 ok_sequence(WmCtrlAltVkN, "Ctrl+Alt+VK_N press/release 1", FALSE);
5221 ret = DestroyAcceleratorTable(hAccel);
5222 ok( ret, "DestroyAcceleratorTable error %d\n", GetLastError());
5224 hAccel = LoadAccelerators(GetModuleHandleA(0), MAKEINTRESOURCE(2));
5225 assert(hAccel != 0);
5227 trace("testing VK_N press/release\n");
5229 keybd_event('N', 0, 0, 0);
5230 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5231 pump_msg_loop(hwnd, hAccel);
5232 ok_sequence(WmVkN, "VK_N press/release", FALSE);
5234 trace("testing Shift+VK_N press/release\n");
5236 keybd_event(VK_SHIFT, 0, 0, 0);
5237 keybd_event('N', 0, 0, 0);
5238 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5239 keybd_event(VK_SHIFT, 0, KEYEVENTF_KEYUP, 0);
5240 pump_msg_loop(hwnd, hAccel);
5241 ok_sequence(WmShiftVkN, "Shift+VK_N press/release", FALSE);
5243 trace("testing Ctrl+VK_N press/release 2\n");
5245 keybd_event(VK_CONTROL, 0, 0, 0);
5246 keybd_event('N', 0, 0, 0);
5247 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5248 keybd_event(VK_CONTROL, 0, KEYEVENTF_KEYUP, 0);
5249 pump_msg_loop(hwnd, hAccel);
5250 ok_sequence(WmCtrlVkN_2, "Ctrl+VK_N press/release 2", FALSE);
5252 trace("testing Alt+VK_N press/release 2\n");
5254 keybd_event(VK_MENU, 0, 0, 0);
5255 keybd_event('N', 0, 0, 0);
5256 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5257 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5258 pump_msg_loop(hwnd, hAccel);
5259 ok_sequence(WmAltVkN_2, "Alt+VK_N press/release 2", FALSE);
5261 trace("testing Ctrl+Alt+VK_N press/release 2\n");
5263 keybd_event(VK_CONTROL, 0, 0, 0);
5264 keybd_event(VK_MENU, 0, 0, 0);
5265 keybd_event('N', 0, 0, 0);
5266 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5267 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5268 keybd_event(VK_CONTROL, 0, KEYEVENTF_KEYUP, 0);
5269 pump_msg_loop(hwnd, hAccel);
5270 ok_sequence(WmCtrlAltVkN, "Ctrl+Alt+VK_N press/release 2", FALSE);
5272 trace("testing Ctrl+Shift+VK_N press/release\n");
5274 keybd_event(VK_CONTROL, 0, 0, 0);
5275 keybd_event(VK_SHIFT, 0, 0, 0);
5276 keybd_event('N', 0, 0, 0);
5277 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5278 keybd_event(VK_SHIFT, 0, KEYEVENTF_KEYUP, 0);
5279 keybd_event(VK_CONTROL, 0, KEYEVENTF_KEYUP, 0);
5280 pump_msg_loop(hwnd, hAccel);
5281 ok_sequence(WmCtrlShiftVkN, "Ctrl+Shift+VK_N press/release", FALSE);
5283 trace("testing Ctrl+Alt+Shift+VK_N press/release\n");
5285 keybd_event(VK_CONTROL, 0, 0, 0);
5286 keybd_event(VK_MENU, 0, 0, 0);
5287 keybd_event(VK_SHIFT, 0, 0, 0);
5288 keybd_event('N', 0, 0, 0);
5289 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
5290 keybd_event(VK_SHIFT, 0, KEYEVENTF_KEYUP, 0);
5291 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5292 keybd_event(VK_CONTROL, 0, KEYEVENTF_KEYUP, 0);
5293 pump_msg_loop(hwnd, hAccel);
5294 ok_sequence(WmCtrlAltShiftVkN, "Ctrl+Alt+Shift+VK_N press/release", FALSE);
5296 ret = DestroyAcceleratorTable(hAccel);
5297 ok( ret, "DestroyAcceleratorTable error %d\n", GetLastError());
5299 trace("testing Alt press/release\n");
5301 keybd_event(VK_MENU, 0, 0, 0);
5302 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5303 keybd_event(VK_MENU, 0, 0, 0);
5304 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5305 pump_msg_loop(hwnd, 0);
5306 /* this test doesn't pass in Wine for managed windows */
5307 ok_sequence(WmAltPressRelease, "Alt press/release", TRUE);
5309 trace("testing Alt+MouseButton press/release\n");
5310 /* first, move mouse pointer inside of the window client area */
5311 GetClientRect(hwnd, &rc);
5312 MapWindowPoints(hwnd, 0, (LPPOINT)&rc, 2);
5313 rc.left += (rc.right - rc.left)/2;
5314 rc.top += (rc.bottom - rc.top)/2;
5315 SetCursorPos(rc.left, rc.top);
5317 pump_msg_loop(hwnd, 0);
5319 keybd_event(VK_MENU, 0, 0, 0);
5320 mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);
5321 mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
5322 keybd_event(VK_MENU, 0, KEYEVENTF_KEYUP, 0);
5323 pump_msg_loop(hwnd, 0);
5324 ok_sequence(WmAltMouseButton, "Alt+MouseButton press/release", FALSE);
5326 keybd_event(VK_F1, 0, 0, 0);
5327 keybd_event(VK_F1, 0, KEYEVENTF_KEYUP, 0);
5328 pump_msg_loop(hwnd, 0);
5329 ok_sequence(WmF1Seq, "F1 press/release", TRUE);
5331 DestroyWindow(hwnd);
5334 /************* window procedures ********************/
5336 static LRESULT MsgCheckProc (BOOL unicode, HWND hwnd, UINT message,
5337 WPARAM wParam, LPARAM lParam)
5339 static long defwndproc_counter = 0;
5340 static long beginpaint_counter = 0;
5344 trace("%p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
5346 /* explicitly ignore WM_GETICON message */
5347 if (message == WM_GETICON) return 0;
5353 LONG style = GetWindowLongA(hwnd, GWL_STYLE);
5354 ok((BOOL)wParam == !(style & WS_DISABLED),
5355 "wrong WS_DISABLED state: %d != %d\n", wParam, !(style & WS_DISABLED));
5359 case WM_CAPTURECHANGED:
5360 if (test_DestroyWindow_flag)
5362 DWORD style = GetWindowLongA(hwnd, GWL_STYLE);
5363 if (style & WS_CHILD)
5364 lParam = GetWindowLongPtrA(hwnd, GWLP_ID);
5365 else if (style & WS_POPUP)
5366 lParam = WND_POPUP_ID;
5368 lParam = WND_PARENT_ID;
5376 ok(!GetWindow(hwnd, GW_CHILD), "children should be unlinked at this point\n");
5377 capture = GetCapture();
5380 ok(capture == hwnd, "capture should NOT be released at this point (capture %p)\n", capture);
5381 trace("current capture %p, releasing...\n", capture);
5388 ok(pGetAncestor(hwnd, GA_PARENT) != 0, "parent should NOT be unlinked at this point\n");
5389 if (test_DestroyWindow_flag)
5391 DWORD style = GetWindowLongA(hwnd, GWL_STYLE);
5392 if (style & WS_CHILD)
5393 lParam = GetWindowLongPtrA(hwnd, GWLP_ID);
5394 else if (style & WS_POPUP)
5395 lParam = WND_POPUP_ID;
5397 lParam = WND_PARENT_ID;
5401 /* test_accelerators() depends on this */
5408 case WM_DEVICECHANGE:
5411 case WM_WINDOWPOSCHANGING:
5412 case WM_WINDOWPOSCHANGED:
5414 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
5416 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
5417 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
5418 winpos->hwnd, winpos->hwndInsertAfter,
5419 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
5421 /* Log only documented flags, win2k uses 0x1000 and 0x2000
5422 * in the high word for internal purposes
5424 wParam = winpos->flags & 0xffff;
5429 msg.message = message;
5430 msg.flags = sent|wparam|lparam;
5431 if (defwndproc_counter) msg.flags |= defwinproc;
5432 if (beginpaint_counter) msg.flags |= beginpaint;
5433 msg.wParam = wParam;
5434 msg.lParam = lParam;
5437 if (message == WM_GETMINMAXINFO && (GetWindowLongA(hwnd, GWL_STYLE) & WS_CHILD))
5439 HWND parent = GetParent(hwnd);
5441 MINMAXINFO *minmax = (MINMAXINFO *)lParam;
5443 GetClientRect(parent, &rc);
5444 trace("parent %p client size = (%d x %d)\n", parent, rc.right, rc.bottom);
5446 trace("ptReserved = (%d,%d)\n"
5447 "ptMaxSize = (%d,%d)\n"
5448 "ptMaxPosition = (%d,%d)\n"
5449 "ptMinTrackSize = (%d,%d)\n"
5450 "ptMaxTrackSize = (%d,%d)\n",
5451 minmax->ptReserved.x, minmax->ptReserved.y,
5452 minmax->ptMaxSize.x, minmax->ptMaxSize.y,
5453 minmax->ptMaxPosition.x, minmax->ptMaxPosition.y,
5454 minmax->ptMinTrackSize.x, minmax->ptMinTrackSize.y,
5455 minmax->ptMaxTrackSize.x, minmax->ptMaxTrackSize.y);
5457 ok(minmax->ptMaxSize.x == rc.right, "default width of maximized child %d != %d\n",
5458 minmax->ptMaxSize.x, rc.right);
5459 ok(minmax->ptMaxSize.y == rc.bottom, "default height of maximized child %d != %d\n",
5460 minmax->ptMaxSize.y, rc.bottom);
5463 if (message == WM_PAINT)
5466 beginpaint_counter++;
5467 BeginPaint( hwnd, &ps );
5468 beginpaint_counter--;
5469 EndPaint( hwnd, &ps );
5473 defwndproc_counter++;
5474 ret = unicode ? DefWindowProcW(hwnd, message, wParam, lParam)
5475 : DefWindowProcA(hwnd, message, wParam, lParam);
5476 defwndproc_counter--;
5481 static LRESULT WINAPI MsgCheckProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
5483 return MsgCheckProc (FALSE, hwnd, message, wParam, lParam);
5486 static LRESULT WINAPI MsgCheckProcW(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
5488 return MsgCheckProc (TRUE, hwnd, message, wParam, lParam);
5491 static LRESULT WINAPI PopupMsgCheckProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
5493 static long defwndproc_counter = 0;
5497 trace("popup: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
5499 /* explicitly ignore WM_GETICON message */
5500 if (message == WM_GETICON) return 0;
5502 msg.message = message;
5503 msg.flags = sent|wparam|lparam;
5504 if (defwndproc_counter) msg.flags |= defwinproc;
5505 msg.wParam = wParam;
5506 msg.lParam = lParam;
5509 if (message == WM_CREATE)
5511 DWORD style = GetWindowLongA(hwnd, GWL_STYLE) | WS_VISIBLE;
5512 SetWindowLongA(hwnd, GWL_STYLE, style);
5515 defwndproc_counter++;
5516 ret = DefWindowProcA(hwnd, message, wParam, lParam);
5517 defwndproc_counter--;
5522 static LRESULT WINAPI ParentMsgCheckProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
5524 static long defwndproc_counter = 0;
5525 static long beginpaint_counter = 0;
5529 trace("parent: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
5531 /* explicitly ignore WM_GETICON message */
5532 if (message == WM_GETICON) return 0;
5534 if (log_all_parent_messages ||
5535 message == WM_PARENTNOTIFY || message == WM_CANCELMODE ||
5536 message == WM_SETFOCUS || message == WM_KILLFOCUS ||
5537 message == WM_ENABLE || message == WM_ENTERIDLE ||
5538 message == WM_IME_SETCONTEXT)
5545 INT ret = GetClipBox((HDC)wParam, &rc);
5547 trace("WM_ERASEBKGND: GetClipBox()=%d, (%d,%d-%d,%d)\n",
5548 ret, rc.left, rc.top, rc.right, rc.bottom);
5552 case WM_WINDOWPOSCHANGING:
5553 case WM_WINDOWPOSCHANGED:
5555 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
5557 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
5558 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
5559 winpos->hwnd, winpos->hwndInsertAfter,
5560 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
5562 /* Log only documented flags, win2k uses 0x1000 and 0x2000
5563 * in the high word for internal purposes
5565 wParam = winpos->flags & 0xffff;
5570 msg.message = message;
5571 msg.flags = sent|parent|wparam|lparam;
5572 if (defwndproc_counter) msg.flags |= defwinproc;
5573 if (beginpaint_counter) msg.flags |= beginpaint;
5574 msg.wParam = wParam;
5575 msg.lParam = lParam;
5579 if (message == WM_PAINT)
5582 beginpaint_counter++;
5583 BeginPaint( hwnd, &ps );
5584 beginpaint_counter--;
5585 EndPaint( hwnd, &ps );
5589 defwndproc_counter++;
5590 ret = DefWindowProcA(hwnd, message, wParam, lParam);
5591 defwndproc_counter--;
5596 static LRESULT WINAPI TestDlgProcA(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
5598 static long defwndproc_counter = 0;
5602 trace("dialog: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
5604 /* explicitly ignore WM_GETICON message */
5605 if (message == WM_GETICON) return 0;
5607 DefDlgProcA(hwnd, DM_SETDEFID, 1, 0);
5608 ret = DefDlgProcA(hwnd, DM_GETDEFID, 0, 0);
5609 if (after_end_dialog)
5610 ok( ret == 0, "DM_GETDEFID should return 0 after EndDialog, got %lx\n", ret );
5612 ok(HIWORD(ret) == DC_HASDEFID, "DM_GETDEFID should return DC_HASDEFID, got %lx\n", ret);
5616 case WM_WINDOWPOSCHANGING:
5617 case WM_WINDOWPOSCHANGED:
5619 WINDOWPOS *winpos = (WINDOWPOS *)lParam;
5621 trace("%s\n", (message == WM_WINDOWPOSCHANGING) ? "WM_WINDOWPOSCHANGING" : "WM_WINDOWPOSCHANGED");
5622 trace("%p after %p, x %d, y %d, cx %d, cy %d flags %08x\n",
5623 winpos->hwnd, winpos->hwndInsertAfter,
5624 winpos->x, winpos->y, winpos->cx, winpos->cy, winpos->flags);
5626 /* Log only documented flags, win2k uses 0x1000 and 0x2000
5627 * in the high word for internal purposes
5629 wParam = winpos->flags & 0xffff;
5634 msg.message = message;
5635 msg.flags = sent|wparam|lparam;
5636 if (defwndproc_counter) msg.flags |= defwinproc;
5637 msg.wParam = wParam;
5638 msg.lParam = lParam;
5641 defwndproc_counter++;
5642 ret = DefDlgProcA(hwnd, message, wParam, lParam);
5643 defwndproc_counter--;
5648 static BOOL RegisterWindowClasses(void)
5653 cls.lpfnWndProc = MsgCheckProcA;
5656 cls.hInstance = GetModuleHandleA(0);
5658 cls.hCursor = LoadCursorA(0, (LPSTR)IDC_ARROW);
5659 cls.hbrBackground = GetStockObject(WHITE_BRUSH);
5660 cls.lpszMenuName = NULL;
5661 cls.lpszClassName = "TestWindowClass";
5662 if(!RegisterClassA(&cls)) return FALSE;
5664 cls.lpfnWndProc = PopupMsgCheckProcA;
5665 cls.lpszClassName = "TestPopupClass";
5666 if(!RegisterClassA(&cls)) return FALSE;
5668 cls.lpfnWndProc = ParentMsgCheckProcA;
5669 cls.lpszClassName = "TestParentClass";
5670 if(!RegisterClassA(&cls)) return FALSE;
5672 cls.lpfnWndProc = DefWindowProcA;
5673 cls.lpszClassName = "SimpleWindowClass";
5674 if(!RegisterClassA(&cls)) return FALSE;
5676 cls.style = CS_NOCLOSE;
5677 cls.lpszClassName = "NoCloseWindowClass";
5678 if(!RegisterClassA(&cls)) return FALSE;
5680 ok(GetClassInfoA(0, "#32770", &cls), "GetClassInfo failed\n");
5682 cls.hInstance = GetModuleHandleA(0);
5683 cls.hbrBackground = 0;
5684 cls.lpfnWndProc = TestDlgProcA;
5685 cls.lpszClassName = "TestDialogClass";
5686 if(!RegisterClassA(&cls)) return FALSE;
5691 static HHOOK hCBT_hook;
5692 static DWORD cbt_hook_thread_id;
5694 static LRESULT CALLBACK cbt_hook_proc(int nCode, WPARAM wParam, LPARAM lParam)
5696 static const char * const CBT_code_name[10] = {
5703 "HCBT_CLICKSKIPPED",
5707 const char *code_name = (nCode >= 0 && nCode <= HCBT_SETFOCUS) ? CBT_code_name[nCode] : "Unknown";
5711 trace("CBT: %d (%s), %08x, %08lx\n", nCode, code_name, wParam, lParam);
5713 ok(cbt_hook_thread_id == GetCurrentThreadId(), "we didn't ask for events from other threads\n");
5715 if (nCode == HCBT_SYSCOMMAND)
5719 msg.message = nCode;
5720 msg.flags = hook|wparam|lparam;
5721 msg.wParam = wParam;
5722 msg.lParam = lParam;
5725 return CallNextHookEx(hCBT_hook, nCode, wParam, lParam);
5728 if (nCode == HCBT_DESTROYWND)
5730 if (test_DestroyWindow_flag)
5732 DWORD style = GetWindowLongA((HWND)wParam, GWL_STYLE);
5733 if (style & WS_CHILD)
5734 lParam = GetWindowLongPtrA((HWND)wParam, GWLP_ID);
5735 else if (style & WS_POPUP)
5736 lParam = WND_POPUP_ID;
5738 lParam = WND_PARENT_ID;
5742 /* Log also SetFocus(0) calls */
5743 hwnd = wParam ? (HWND)wParam : (HWND)lParam;
5745 if (GetClassNameA(hwnd, buf, sizeof(buf)))
5747 if (!lstrcmpiA(buf, "TestWindowClass") ||
5748 !lstrcmpiA(buf, "TestParentClass") ||
5749 !lstrcmpiA(buf, "TestPopupClass") ||
5750 !lstrcmpiA(buf, "SimpleWindowClass") ||
5751 !lstrcmpiA(buf, "TestDialogClass") ||
5752 !lstrcmpiA(buf, "MDI_frame_class") ||
5753 !lstrcmpiA(buf, "MDI_client_class") ||
5754 !lstrcmpiA(buf, "MDI_child_class") ||
5755 !lstrcmpiA(buf, "my_button_class") ||
5756 !lstrcmpiA(buf, "my_edit_class") ||
5757 !lstrcmpiA(buf, "static") ||
5758 !lstrcmpiA(buf, "#32770"))
5762 msg.message = nCode;
5763 msg.flags = hook|wparam|lparam;
5764 msg.wParam = wParam;
5765 msg.lParam = lParam;
5769 return CallNextHookEx(hCBT_hook, nCode, wParam, lParam);
5772 static void CALLBACK win_event_proc(HWINEVENTHOOK hevent,
5782 trace("WEH:%p,event %08x,hwnd %p,obj %08x,id %08x,thread %08x,time %08x\n",
5783 hevent, event, hwnd, object_id, child_id, thread_id, event_time);
5785 ok(thread_id == GetCurrentThreadId(), "we didn't ask for events from other threads\n");
5787 /* ignore mouse cursor events */
5788 if (object_id == OBJID_CURSOR) return;
5790 if (!hwnd || GetClassNameA(hwnd, buf, sizeof(buf)))
5793 !lstrcmpiA(buf, "TestWindowClass") ||
5794 !lstrcmpiA(buf, "TestParentClass") ||
5795 !lstrcmpiA(buf, "TestPopupClass") ||
5796 !lstrcmpiA(buf, "SimpleWindowClass") ||
5797 !lstrcmpiA(buf, "TestDialogClass") ||
5798 !lstrcmpiA(buf, "MDI_frame_class") ||
5799 !lstrcmpiA(buf, "MDI_client_class") ||
5800 !lstrcmpiA(buf, "MDI_child_class") ||
5801 !lstrcmpiA(buf, "my_button_class") ||
5802 !lstrcmpiA(buf, "my_edit_class") ||
5803 !lstrcmpiA(buf, "static") ||
5804 !lstrcmpiA(buf, "#32770"))
5808 msg.message = event;
5809 msg.flags = winevent_hook|wparam|lparam;
5810 msg.wParam = object_id;
5811 msg.lParam = child_id;
5817 static const WCHAR wszUnicode[] = {'U','n','i','c','o','d','e',0};
5818 static const WCHAR wszAnsi[] = {'U',0};
5820 static LRESULT CALLBACK MsgConversionProcW(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
5824 case CB_FINDSTRINGEXACT:
5825 trace("String: %p\n", (LPCWSTR)lParam);
5826 if (!lstrcmpW((LPCWSTR)lParam, wszUnicode))
5828 if (!lstrcmpW((LPCWSTR)lParam, wszAnsi))
5832 return DefWindowProcW(hwnd, uMsg, wParam, lParam);
5835 static const struct message WmGetTextLengthAfromW[] = {
5836 { WM_GETTEXTLENGTH, sent },
5837 { WM_GETTEXT, sent },
5841 static const WCHAR testWindowClassW[] =
5842 { 'T','e','s','t','W','i','n','d','o','w','C','l','a','s','s','W',0 };
5844 static const WCHAR dummy_window_text[] = {'d','u','m','m','y',' ','t','e','x','t',0};
5846 /* dummy window proc for WM_GETTEXTLENGTH test */
5847 static LRESULT CALLBACK get_text_len_proc( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp )
5851 case WM_GETTEXTLENGTH:
5852 return lstrlenW(dummy_window_text) + 37; /* some random length */
5854 lstrcpynW( (LPWSTR)lp, dummy_window_text, wp );
5855 return lstrlenW( (LPWSTR)lp );
5857 return DefWindowProcW( hwnd, msg, wp, lp );
5861 static void test_message_conversion(void)
5863 static const WCHAR wszMsgConversionClass[] =
5864 {'M','s','g','C','o','n','v','e','r','s','i','o','n','C','l','a','s','s',0};
5868 WNDPROC wndproc, newproc;
5872 cls.lpfnWndProc = MsgConversionProcW;
5875 cls.hInstance = GetModuleHandleW(NULL);
5877 cls.hCursor = LoadCursorW(NULL, (LPWSTR)IDC_ARROW);
5878 cls.hbrBackground = (HBRUSH)(COLOR_BTNFACE+1);
5879 cls.lpszMenuName = NULL;
5880 cls.lpszClassName = wszMsgConversionClass;
5881 /* this call will fail on Win9x, but that doesn't matter as this test is
5882 * meaningless on those platforms */
5883 if(!RegisterClassW(&cls)) return;
5886 cls.lpfnWndProc = MsgCheckProcW;
5889 cls.hInstance = GetModuleHandleW(0);
5891 cls.hCursor = LoadCursorW(0, (LPWSTR)IDC_ARROW);
5892 cls.hbrBackground = GetStockObject(WHITE_BRUSH);
5893 cls.lpszMenuName = NULL;
5894 cls.lpszClassName = testWindowClassW;
5895 if(!RegisterClassW(&cls)) return;
5897 hwnd = CreateWindowExW(0, wszMsgConversionClass, NULL, WS_OVERLAPPED,
5898 100, 100, 200, 200, 0, 0, 0, NULL);
5899 ok(hwnd != NULL, "Window creation failed\n");
5903 wndproc = (WNDPROC)GetWindowLongPtrA(hwnd, GWLP_WNDPROC);
5904 lRes = CallWindowProcA(wndproc, hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5905 ok(lRes == 0, "String should have been converted\n");
5906 lRes = CallWindowProcW(wndproc, hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5907 ok(lRes == 1, "String shouldn't have been converted\n");
5911 wndproc = (WNDPROC)GetWindowLongPtrW(hwnd, GWLP_WNDPROC);
5912 lRes = CallWindowProcA(wndproc, hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5913 ok(lRes == 1, "String shouldn't have been converted\n");
5914 lRes = CallWindowProcW(wndproc, hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5915 ok(lRes == 1, "String shouldn't have been converted\n");
5917 /* Synchronous messages */
5919 lRes = SendMessageA(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5920 ok(lRes == 0, "String should have been converted\n");
5921 lRes = SendMessageW(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5922 ok(lRes == 1, "String shouldn't have been converted\n");
5924 /* Asynchronous messages */
5927 lRes = PostMessageA(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5928 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5929 "PostMessage on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5931 lRes = PostMessageW(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5932 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5933 "PostMessage on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5935 lRes = PostThreadMessageA(GetCurrentThreadId(), CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5936 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5937 "PosThreadtMessage on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5939 lRes = PostThreadMessageW(GetCurrentThreadId(), CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5940 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5941 "PosThreadtMessage on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5943 lRes = SendNotifyMessageA(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5944 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5945 "SendNotifyMessage on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5947 lRes = SendNotifyMessageW(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode);
5948 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5949 "SendNotifyMessage on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5951 lRes = SendMessageCallbackA(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode, NULL, 0);
5952 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5953 "SendMessageCallback on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5955 lRes = SendMessageCallbackW(hwnd, CB_FINDSTRINGEXACT, 0, (LPARAM)wszUnicode, NULL, 0);
5956 ok(lRes == 0 && (GetLastError() == ERROR_MESSAGE_SYNC_ONLY || GetLastError() == ERROR_INVALID_PARAMETER),
5957 "SendMessageCallback on sync only message returned %ld, last error %d\n", lRes, GetLastError());
5959 /* Check WM_GETTEXTLENGTH A->W behaviour, whether WM_GETTEXT is also sent or not */
5961 hwnd = CreateWindowW (testWindowClassW, wszUnicode,
5962 WS_OVERLAPPEDWINDOW,
5963 100, 100, 200, 200, 0, 0, 0, NULL);
5966 lRes = SendMessageA (hwnd, WM_GETTEXTLENGTH, 0, 0);
5967 ok_sequence(WmGetTextLengthAfromW, "ANSI WM_GETTEXTLENGTH to Unicode window", FALSE);
5968 ok( lRes == WideCharToMultiByte( CP_ACP, 0, wszUnicode, lstrlenW(wszUnicode), NULL, 0, NULL, NULL ),
5969 "got bad length %ld\n", lRes );
5972 lRes = CallWindowProcA( (WNDPROC)GetWindowLongPtrA( hwnd, GWLP_WNDPROC ),
5973 hwnd, WM_GETTEXTLENGTH, 0, 0);
5974 ok_sequence(WmGetTextLengthAfromW, "ANSI WM_GETTEXTLENGTH to Unicode window", FALSE);
5975 ok( lRes == WideCharToMultiByte( CP_ACP, 0, wszUnicode, lstrlenW(wszUnicode), NULL, 0, NULL, NULL ),
5976 "got bad length %ld\n", lRes );
5978 wndproc = (WNDPROC)SetWindowLongPtrW( hwnd, GWLP_WNDPROC, (LONG_PTR)get_text_len_proc );
5979 newproc = (WNDPROC)GetWindowLongPtrA( hwnd, GWLP_WNDPROC );
5980 lRes = CallWindowProcA( newproc, hwnd, WM_GETTEXTLENGTH, 0, 0 );
5981 ok( lRes == WideCharToMultiByte( CP_ACP, 0, dummy_window_text, lstrlenW(dummy_window_text),
5982 NULL, 0, NULL, NULL ),
5983 "got bad length %ld\n", lRes );
5985 SetWindowLongPtrW( hwnd, GWLP_WNDPROC, (LONG_PTR)wndproc ); /* restore old wnd proc */
5986 lRes = CallWindowProcA( newproc, hwnd, WM_GETTEXTLENGTH, 0, 0 );
5987 ok( lRes == WideCharToMultiByte( CP_ACP, 0, dummy_window_text, lstrlenW(dummy_window_text),
5988 NULL, 0, NULL, NULL ),
5989 "got bad length %ld\n", lRes );
5991 ret = DestroyWindow(hwnd);
5992 ok( ret, "DestroyWindow() error %d\n", GetLastError());
6002 static VOID CALLBACK tfunc(HWND hwnd, UINT uMsg, UINT id, DWORD dwTime)
6006 #define TIMER_ID 0x19
6008 static DWORD WINAPI timer_thread_proc(LPVOID x)
6010 struct timer_info *info = x;
6013 r = KillTimer(info->hWnd, 0x19);
6014 ok(r,"KillTimer failed in thread\n");
6015 r = SetTimer(info->hWnd,TIMER_ID,10000,tfunc);
6016 ok(r,"SetTimer failed in thread\n");
6017 ok(r==TIMER_ID,"SetTimer id different\n");
6018 r = SetEvent(info->handles[0]);
6019 ok(r,"SetEvent failed in thread\n");
6023 static void test_timers(void)
6025 struct timer_info info;
6028 info.hWnd = CreateWindow ("TestWindowClass", NULL,
6029 WS_OVERLAPPEDWINDOW ,
6030 CW_USEDEFAULT, CW_USEDEFAULT, 300, 300, 0,
6033 info.id = SetTimer(info.hWnd,TIMER_ID,10000,tfunc);
6034 ok(info.id, "SetTimer failed\n");
6035 ok(info.id==TIMER_ID, "SetTimer timer ID different\n");
6036 info.handles[0] = CreateEvent(NULL,0,0,NULL);
6037 info.handles[1] = CreateThread(NULL,0,timer_thread_proc,&info,0,&id);
6039 WaitForMultipleObjects(2, info.handles, FALSE, INFINITE);
6041 WaitForSingleObject(info.handles[1], INFINITE);
6043 CloseHandle(info.handles[0]);
6044 CloseHandle(info.handles[1]);
6046 ok( KillTimer(info.hWnd, TIMER_ID), "KillTimer failed\n");
6048 ok(DestroyWindow(info.hWnd), "failed to destroy window\n");
6051 /* Various win events with arbitrary parameters */
6052 static const struct message WmWinEventsSeq[] = {
6053 { EVENT_SYSTEM_SOUND, winevent_hook|wparam|lparam, OBJID_WINDOW, 0 },
6054 { EVENT_SYSTEM_ALERT, winevent_hook|wparam|lparam, OBJID_SYSMENU, 1 },
6055 { EVENT_SYSTEM_FOREGROUND, winevent_hook|wparam|lparam, OBJID_TITLEBAR, 2 },
6056 { EVENT_SYSTEM_MENUSTART, winevent_hook|wparam|lparam, OBJID_MENU, 3 },
6057 { EVENT_SYSTEM_MENUEND, winevent_hook|wparam|lparam, OBJID_CLIENT, 4 },
6058 { EVENT_SYSTEM_MENUPOPUPSTART, winevent_hook|wparam|lparam, OBJID_VSCROLL, 5 },
6059 { EVENT_SYSTEM_MENUPOPUPEND, winevent_hook|wparam|lparam, OBJID_HSCROLL, 6 },
6060 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, OBJID_SIZEGRIP, 7 },
6061 { EVENT_SYSTEM_CAPTUREEND, winevent_hook|wparam|lparam, OBJID_CARET, 8 },
6062 /* our win event hook ignores OBJID_CURSOR events */
6063 /*{ EVENT_SYSTEM_MOVESIZESTART, winevent_hook|wparam|lparam, OBJID_CURSOR, 9 },*/
6064 { EVENT_SYSTEM_MOVESIZEEND, winevent_hook|wparam|lparam, OBJID_ALERT, 10 },
6065 { EVENT_SYSTEM_CONTEXTHELPSTART, winevent_hook|wparam|lparam, OBJID_SOUND, 11 },
6066 { EVENT_SYSTEM_CONTEXTHELPEND, winevent_hook|wparam|lparam, OBJID_QUERYCLASSNAMEIDX, 12 },
6067 { EVENT_SYSTEM_DRAGDROPSTART, winevent_hook|wparam|lparam, OBJID_NATIVEOM, 13 },
6068 { EVENT_SYSTEM_DRAGDROPEND, winevent_hook|wparam|lparam, OBJID_WINDOW, 0 },
6069 { EVENT_SYSTEM_DIALOGSTART, winevent_hook|wparam|lparam, OBJID_SYSMENU, 1 },
6070 { EVENT_SYSTEM_DIALOGEND, winevent_hook|wparam|lparam, OBJID_TITLEBAR, 2 },
6071 { EVENT_SYSTEM_SCROLLINGSTART, winevent_hook|wparam|lparam, OBJID_MENU, 3 },
6072 { EVENT_SYSTEM_SCROLLINGEND, winevent_hook|wparam|lparam, OBJID_CLIENT, 4 },
6073 { EVENT_SYSTEM_SWITCHSTART, winevent_hook|wparam|lparam, OBJID_VSCROLL, 5 },
6074 { EVENT_SYSTEM_SWITCHEND, winevent_hook|wparam|lparam, OBJID_HSCROLL, 6 },
6075 { EVENT_SYSTEM_MINIMIZESTART, winevent_hook|wparam|lparam, OBJID_SIZEGRIP, 7 },
6076 { EVENT_SYSTEM_MINIMIZEEND, winevent_hook|wparam|lparam, OBJID_CARET, 8 },
6079 static const struct message WmWinEventCaretSeq[] = {
6080 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 1 */
6081 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 1 */
6082 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 2 */
6083 { EVENT_OBJECT_NAMECHANGE, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 1 */
6086 static const struct message WmWinEventCaretSeq_2[] = {
6087 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 1/2 */
6088 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 1/2 */
6089 { EVENT_OBJECT_NAMECHANGE, winevent_hook|wparam|lparam, OBJID_CARET, 0 }, /* hook 1/2 */
6092 static const struct message WmWinEventAlertSeq[] = {
6093 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, OBJID_ALERT, 0 },
6096 static const struct message WmWinEventAlertSeq_2[] = {
6097 /* create window in the thread proc */
6098 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_WINDOW, 2 },
6099 /* our test event */
6100 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, OBJID_ALERT, 2 },
6103 static const struct message WmGlobalHookSeq_1[] = {
6104 /* create window in the thread proc */
6105 { HCBT_CREATEWND, hook|lparam, 0, 2 },
6106 /* our test events */
6107 { HCBT_SYSCOMMAND, hook|wparam|lparam, SC_PREVWINDOW, 2 },
6108 { HCBT_SYSCOMMAND, hook|wparam|lparam, SC_NEXTWINDOW, 2 },
6111 static const struct message WmGlobalHookSeq_2[] = {
6112 { HCBT_SYSCOMMAND, hook|wparam|lparam, SC_NEXTWINDOW, 0 }, /* old local hook */
6113 { HCBT_SYSCOMMAND, hook|wparam|lparam, SC_NEXTWINDOW, 2 }, /* new global hook */
6114 { HCBT_SYSCOMMAND, hook|wparam|lparam, SC_PREVWINDOW, 0 }, /* old local hook */
6115 { HCBT_SYSCOMMAND, hook|wparam|lparam, SC_PREVWINDOW, 2 }, /* new global hook */
6119 static const struct message WmMouseLLHookSeq[] = {
6120 { WM_MOUSEMOVE, hook },
6121 { WM_LBUTTONUP, hook },
6122 { WM_MOUSEMOVE, hook },
6126 static void CALLBACK win_event_global_hook_proc(HWINEVENTHOOK hevent,
6136 trace("WEH_2:%p,event %08x,hwnd %p,obj %08x,id %08x,thread %08x,time %08x\n",
6137 hevent, event, hwnd, object_id, child_id, thread_id, event_time);
6139 if (GetClassNameA(hwnd, buf, sizeof(buf)))
6141 if (!lstrcmpiA(buf, "TestWindowClass") ||
6142 !lstrcmpiA(buf, "static"))
6146 msg.message = event;
6147 msg.flags = winevent_hook|wparam|lparam;
6148 msg.wParam = object_id;
6149 msg.lParam = (thread_id == GetCurrentThreadId()) ? child_id : (child_id + 2);
6155 static HHOOK hCBT_global_hook;
6156 static DWORD cbt_global_hook_thread_id;
6158 static LRESULT CALLBACK cbt_global_hook_proc(int nCode, WPARAM wParam, LPARAM lParam)
6163 trace("CBT_2: %d, %08x, %08lx\n", nCode, wParam, lParam);
6165 if (nCode == HCBT_SYSCOMMAND)
6169 msg.message = nCode;
6170 msg.flags = hook|wparam|lparam;
6171 msg.wParam = wParam;
6172 msg.lParam = (cbt_global_hook_thread_id == GetCurrentThreadId()) ? 1 : 2;
6175 return CallNextHookEx(hCBT_global_hook, nCode, wParam, lParam);
6177 /* WH_MOUSE_LL hook */
6178 if (nCode == HC_ACTION)
6181 MSLLHOOKSTRUCT *mhll = (MSLLHOOKSTRUCT *)lParam;
6183 /* we can't test for real mouse events */
6184 if (mhll->flags & LLMHF_INJECTED)
6186 msg.message = wParam;
6190 return CallNextHookEx(hCBT_global_hook, nCode, wParam, lParam);
6193 /* Log also SetFocus(0) calls */
6194 hwnd = wParam ? (HWND)wParam : (HWND)lParam;
6196 if (GetClassNameA(hwnd, buf, sizeof(buf)))
6198 if (!lstrcmpiA(buf, "TestWindowClass") ||
6199 !lstrcmpiA(buf, "static"))
6203 msg.message = nCode;
6204 msg.flags = hook|wparam|lparam;
6205 msg.wParam = wParam;
6206 msg.lParam = (cbt_global_hook_thread_id == GetCurrentThreadId()) ? 1 : 2;
6210 return CallNextHookEx(hCBT_global_hook, nCode, wParam, lParam);
6213 static DWORD WINAPI win_event_global_thread_proc(void *param)
6217 HANDLE hevent = *(HANDLE *)param;
6218 HMODULE user32 = GetModuleHandleA("user32.dll");
6219 FARPROC pNotifyWinEvent = GetProcAddress(user32, "NotifyWinEvent");
6221 assert(pNotifyWinEvent);
6223 hwnd = CreateWindowExA(0, "static", NULL, WS_POPUP, 0,0,0,0,0,0,0, NULL);
6225 trace("created thread window %p\n", hwnd);
6227 *(HWND *)param = hwnd;
6230 /* this event should be received only by our new hook proc,
6231 * an old one does not expect an event from another thread.
6233 pNotifyWinEvent(EVENT_OBJECT_LOCATIONCHANGE, hwnd, OBJID_ALERT, 0);
6236 while (GetMessage(&msg, 0, 0, 0))
6238 TranslateMessage(&msg);
6239 DispatchMessage(&msg);
6244 static DWORD WINAPI cbt_global_hook_thread_proc(void *param)
6248 HANDLE hevent = *(HANDLE *)param;
6251 /* these events should be received only by our new hook proc,
6252 * an old one does not expect an event from another thread.
6255 hwnd = CreateWindowExA(0, "static", NULL, WS_POPUP, 0,0,0,0,0,0,0, NULL);
6257 trace("created thread window %p\n", hwnd);
6259 *(HWND *)param = hwnd;
6261 /* Windows doesn't like when a thread plays games with the focus,
6262 that leads to all kinds of misbehaviours and failures to activate
6263 a window. So, better keep next lines commented out.
6267 DefWindowProcA(hwnd, WM_SYSCOMMAND, SC_PREVWINDOW, 0);
6268 DefWindowProcA(hwnd, WM_SYSCOMMAND, SC_NEXTWINDOW, 0);
6272 while (GetMessage(&msg, 0, 0, 0))
6274 TranslateMessage(&msg);
6275 DispatchMessage(&msg);
6280 static DWORD WINAPI mouse_ll_global_thread_proc(void *param)
6284 HANDLE hevent = *(HANDLE *)param;
6286 hwnd = CreateWindowExA(0, "static", NULL, WS_POPUP, 0,0,0,0,0,0,0, NULL);
6288 trace("created thread window %p\n", hwnd);
6290 *(HWND *)param = hwnd;
6294 /* Windows doesn't like when a thread plays games with the focus,
6295 * that leads to all kinds of misbehaviours and failures to activate
6296 * a window. So, better don't generate a mouse click message below.
6298 mouse_event(MOUSEEVENTF_MOVE, -1, 0, 0, 0);
6299 mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
6300 mouse_event(MOUSEEVENTF_MOVE, 1, 0, 0, 0);
6303 while (GetMessage(&msg, 0, 0, 0))
6305 TranslateMessage(&msg);
6306 DispatchMessage(&msg);
6311 static void test_winevents(void)
6317 HANDLE hthread, hevent;
6319 HWINEVENTHOOK hhook;
6320 const struct message *events = WmWinEventsSeq;
6321 HMODULE user32 = GetModuleHandleA("user32.dll");
6322 FARPROC pSetWinEventHook = GetProcAddress(user32, "SetWinEventHook");
6323 FARPROC pUnhookWinEvent = GetProcAddress(user32, "UnhookWinEvent");
6324 FARPROC pNotifyWinEvent = GetProcAddress(user32, "NotifyWinEvent");
6326 hwnd = CreateWindowExA(0, "TestWindowClass", NULL,
6327 WS_OVERLAPPEDWINDOW,
6328 CW_USEDEFAULT, CW_USEDEFAULT, 300, 300, 0,
6332 /****** start of global hook test *************/
6333 hCBT_global_hook = SetWindowsHookExA(WH_CBT, cbt_global_hook_proc, GetModuleHandleA(0), 0);
6334 assert(hCBT_global_hook);
6336 hevent = CreateEventA(NULL, 0, 0, NULL);
6338 hwnd2 = (HWND)hevent;
6340 hthread = CreateThread(NULL, 0, cbt_global_hook_thread_proc, &hwnd2, 0, &tid);
6341 ok(hthread != NULL, "CreateThread failed, error %d\n", GetLastError());
6343 ok(WaitForSingleObject(hevent, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6345 ok_sequence(WmGlobalHookSeq_1, "global hook 1", FALSE);
6348 /* this one should be received only by old hook proc */
6349 DefWindowProcA(hwnd, WM_SYSCOMMAND, SC_NEXTWINDOW, 0);
6350 /* this one should be received only by old hook proc */
6351 DefWindowProcA(hwnd, WM_SYSCOMMAND, SC_PREVWINDOW, 0);
6353 ok_sequence(WmGlobalHookSeq_2, "global hook 2", FALSE);
6355 ret = UnhookWindowsHookEx(hCBT_global_hook);
6356 ok( ret, "UnhookWindowsHookEx error %d\n", GetLastError());
6358 PostThreadMessageA(tid, WM_QUIT, 0, 0);
6359 ok(WaitForSingleObject(hthread, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6360 CloseHandle(hthread);
6361 CloseHandle(hevent);
6362 ok(!IsWindow(hwnd2), "window should be destroyed on thread exit\n");
6363 /****** end of global hook test *************/
6365 if (!pSetWinEventHook || !pNotifyWinEvent || !pUnhookWinEvent)
6367 ok(DestroyWindow(hwnd), "failed to destroy window\n");
6373 #if 0 /* this test doesn't pass under Win9x */
6374 /* win2k ignores events with hwnd == 0 */
6375 SetLastError(0xdeadbeef);
6376 pNotifyWinEvent(events[0].message, 0, events[0].wParam, events[0].lParam);
6377 ok(GetLastError() == ERROR_INVALID_WINDOW_HANDLE || /* Win2k */
6378 GetLastError() == 0xdeadbeef, /* Win9x */
6379 "unexpected error %ld\n", GetLastError());
6380 ok_sequence(WmEmptySeq, "empty notify winevents", FALSE);
6383 for (i = 0; i < sizeof(WmWinEventsSeq)/sizeof(WmWinEventsSeq[0]); i++)
6384 pNotifyWinEvent(events[i].message, hwnd, events[i].wParam, events[i].lParam);
6386 ok_sequence(WmWinEventsSeq, "notify winevents", FALSE);
6388 /****** start of event filtering test *************/
6389 hhook = (HWINEVENTHOOK)pSetWinEventHook(
6390 EVENT_OBJECT_SHOW, /* 0x8002 */
6391 EVENT_OBJECT_LOCATIONCHANGE, /* 0x800B */
6392 GetModuleHandleA(0), win_event_global_hook_proc,
6393 GetCurrentProcessId(), 0,
6394 WINEVENT_INCONTEXT);
6395 ok(hhook != 0, "SetWinEventHook error %d\n", GetLastError());
6397 hevent = CreateEventA(NULL, 0, 0, NULL);
6399 hwnd2 = (HWND)hevent;
6401 hthread = CreateThread(NULL, 0, win_event_global_thread_proc, &hwnd2, 0, &tid);
6402 ok(hthread != NULL, "CreateThread failed, error %d\n", GetLastError());
6404 ok(WaitForSingleObject(hevent, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6406 ok_sequence(WmWinEventAlertSeq, "alert winevent", FALSE);
6409 /* this one should be received only by old hook proc */
6410 pNotifyWinEvent(EVENT_OBJECT_CREATE, hwnd, OBJID_CARET, 0); /* 0x8000 */
6411 pNotifyWinEvent(EVENT_OBJECT_SHOW, hwnd, OBJID_CARET, 0); /* 0x8002 */
6412 /* this one should be received only by old hook proc */
6413 pNotifyWinEvent(EVENT_OBJECT_NAMECHANGE, hwnd, OBJID_CARET, 0); /* 0x800C */
6415 ok_sequence(WmWinEventCaretSeq, "caret winevent", FALSE);
6417 ret = pUnhookWinEvent(hhook);
6418 ok( ret, "UnhookWinEvent error %d\n", GetLastError());
6420 PostThreadMessageA(tid, WM_QUIT, 0, 0);
6421 ok(WaitForSingleObject(hthread, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6422 CloseHandle(hthread);
6423 CloseHandle(hevent);
6424 ok(!IsWindow(hwnd2), "window should be destroyed on thread exit\n");
6425 /****** end of event filtering test *************/
6427 /****** start of out of context event test *************/
6428 hhook = (HWINEVENTHOOK)pSetWinEventHook(
6429 EVENT_MIN, EVENT_MAX,
6430 0, win_event_global_hook_proc,
6431 GetCurrentProcessId(), 0,
6432 WINEVENT_OUTOFCONTEXT);
6433 ok(hhook != 0, "SetWinEventHook error %d\n", GetLastError());
6435 hevent = CreateEventA(NULL, 0, 0, NULL);
6437 hwnd2 = (HWND)hevent;
6441 hthread = CreateThread(NULL, 0, win_event_global_thread_proc, &hwnd2, 0, &tid);
6442 ok(hthread != NULL, "CreateThread failed, error %d\n", GetLastError());
6444 ok(WaitForSingleObject(hevent, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6446 ok_sequence(WmEmptySeq, "empty notify winevents", FALSE);
6447 /* process pending winevent messages */
6448 ok(!PeekMessageA(&msg, 0, 0, 0, PM_NOREMOVE), "msg queue should be empty\n");
6449 ok_sequence(WmWinEventAlertSeq_2, "alert winevent for out of context proc", FALSE);
6452 /* this one should be received only by old hook proc */
6453 pNotifyWinEvent(EVENT_OBJECT_CREATE, hwnd, OBJID_CARET, 0); /* 0x8000 */
6454 pNotifyWinEvent(EVENT_OBJECT_SHOW, hwnd, OBJID_CARET, 0); /* 0x8002 */
6455 /* this one should be received only by old hook proc */
6456 pNotifyWinEvent(EVENT_OBJECT_NAMECHANGE, hwnd, OBJID_CARET, 0); /* 0x800C */
6458 ok_sequence(WmWinEventCaretSeq_2, "caret winevent for incontext proc", FALSE);
6459 /* process pending winevent messages */
6460 ok(!PeekMessageA(&msg, 0, 0, 0, PM_NOREMOVE), "msg queue should be empty\n");
6461 ok_sequence(WmWinEventCaretSeq_2, "caret winevent for out of context proc", FALSE);
6463 ret = pUnhookWinEvent(hhook);
6464 ok( ret, "UnhookWinEvent error %d\n", GetLastError());
6466 PostThreadMessageA(tid, WM_QUIT, 0, 0);
6467 ok(WaitForSingleObject(hthread, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6468 CloseHandle(hthread);
6469 CloseHandle(hevent);
6470 ok(!IsWindow(hwnd2), "window should be destroyed on thread exit\n");
6471 /****** end of out of context event test *************/
6473 /****** start of MOUSE_LL hook test *************/
6474 hCBT_global_hook = SetWindowsHookExA(WH_MOUSE_LL, cbt_global_hook_proc, GetModuleHandleA(0), 0);
6475 /* WH_MOUSE_LL is not supported on Win9x platforms */
6476 if (!hCBT_global_hook)
6478 trace("Skipping WH_MOUSE_LL test on this platform\n");
6479 goto skip_mouse_ll_hook_test;
6482 hevent = CreateEventA(NULL, 0, 0, NULL);
6484 hwnd2 = (HWND)hevent;
6486 hthread = CreateThread(NULL, 0, mouse_ll_global_thread_proc, &hwnd2, 0, &tid);
6487 ok(hthread != NULL, "CreateThread failed, error %d\n", GetLastError());
6489 while (WaitForSingleObject(hevent, 100) == WAIT_TIMEOUT)
6490 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6492 ok_sequence(WmMouseLLHookSeq, "MOUSE_LL hook other thread", FALSE);
6495 mouse_event(MOUSEEVENTF_MOVE, -1, 0, 0, 0);
6496 mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
6497 mouse_event(MOUSEEVENTF_MOVE, 1, 0, 0, 0);
6499 ok_sequence(WmMouseLLHookSeq, "MOUSE_LL hook same thread", FALSE);
6501 ret = UnhookWindowsHookEx(hCBT_global_hook);
6502 ok( ret, "UnhookWindowsHookEx error %d\n", GetLastError());
6504 PostThreadMessageA(tid, WM_QUIT, 0, 0);
6505 ok(WaitForSingleObject(hthread, INFINITE) == WAIT_OBJECT_0, "WaitForSingleObject failed\n");
6506 CloseHandle(hthread);
6507 CloseHandle(hevent);
6508 ok(!IsWindow(hwnd2), "window should be destroyed on thread exit\n");
6509 /****** end of MOUSE_LL hook test *************/
6510 skip_mouse_ll_hook_test:
6512 ok(DestroyWindow(hwnd), "failed to destroy window\n");
6515 static void test_set_hook(void)
6519 HWINEVENTHOOK hwinevent_hook;
6520 HMODULE user32 = GetModuleHandleA("user32.dll");
6521 FARPROC pSetWinEventHook = GetProcAddress(user32, "SetWinEventHook");
6522 FARPROC pUnhookWinEvent = GetProcAddress(user32, "UnhookWinEvent");
6524 hhook = SetWindowsHookExA(WH_CBT, cbt_hook_proc, GetModuleHandleA(0), GetCurrentThreadId());
6525 ok(hhook != 0, "local hook does not require hModule set to 0\n");
6526 UnhookWindowsHookEx(hhook);
6528 #if 0 /* this test doesn't pass under Win9x: BUG! */
6529 SetLastError(0xdeadbeef);
6530 hhook = SetWindowsHookExA(WH_CBT, cbt_hook_proc, 0, 0);
6531 ok(!hhook, "global hook requires hModule != 0\n");
6532 ok(GetLastError() == ERROR_HOOK_NEEDS_HMOD, "unexpected error %ld\n", GetLastError());
6535 SetLastError(0xdeadbeef);
6536 hhook = SetWindowsHookExA(WH_CBT, 0, GetModuleHandleA(0), GetCurrentThreadId());
6537 ok(!hhook, "SetWinEventHook with invalid proc should fail\n");
6538 ok(GetLastError() == ERROR_INVALID_FILTER_PROC || /* Win2k */
6539 GetLastError() == 0xdeadbeef, /* Win9x */
6540 "unexpected error %d\n", GetLastError());
6542 SetLastError(0xdeadbeef);
6543 ok(!UnhookWindowsHookEx((HHOOK)0xdeadbeef), "UnhookWindowsHookEx succeeded\n");
6544 ok(GetLastError() == ERROR_INVALID_HOOK_HANDLE || /* Win2k */
6545 GetLastError() == 0xdeadbeef, /* Win9x */
6546 "unexpected error %d\n", GetLastError());
6548 if (!pSetWinEventHook || !pUnhookWinEvent) return;
6550 /* even process local incontext hooks require hmodule */
6551 SetLastError(0xdeadbeef);
6552 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(EVENT_MIN, EVENT_MAX,
6553 0, win_event_proc, GetCurrentProcessId(), 0, WINEVENT_INCONTEXT);
6554 ok(!hwinevent_hook, "WINEVENT_INCONTEXT requires hModule != 0\n");
6555 ok(GetLastError() == ERROR_HOOK_NEEDS_HMOD || /* Win2k */
6556 GetLastError() == 0xdeadbeef, /* Win9x */
6557 "unexpected error %d\n", GetLastError());
6559 /* even thread local incontext hooks require hmodule */
6560 SetLastError(0xdeadbeef);
6561 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(EVENT_MIN, EVENT_MAX,
6562 0, win_event_proc, GetCurrentProcessId(), GetCurrentThreadId(), WINEVENT_INCONTEXT);
6563 ok(!hwinevent_hook, "WINEVENT_INCONTEXT requires hModule != 0\n");
6564 ok(GetLastError() == ERROR_HOOK_NEEDS_HMOD || /* Win2k */
6565 GetLastError() == 0xdeadbeef, /* Win9x */
6566 "unexpected error %d\n", GetLastError());
6568 #if 0 /* these 3 tests don't pass under Win9x */
6569 SetLastError(0xdeadbeef);
6570 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(1, 0,
6571 0, win_event_proc, GetCurrentProcessId(), 0, WINEVENT_OUTOFCONTEXT);
6572 ok(!hwinevent_hook, "SetWinEventHook with invalid event range should fail\n");
6573 ok(GetLastError() == ERROR_INVALID_HOOK_FILTER, "unexpected error %ld\n", GetLastError());
6575 SetLastError(0xdeadbeef);
6576 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(-1, 1,
6577 0, win_event_proc, GetCurrentProcessId(), 0, WINEVENT_OUTOFCONTEXT);
6578 ok(!hwinevent_hook, "SetWinEventHook with invalid event range should fail\n");
6579 ok(GetLastError() == ERROR_INVALID_HOOK_FILTER, "unexpected error %ld\n", GetLastError());
6581 SetLastError(0xdeadbeef);
6582 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(EVENT_MIN, EVENT_MAX,
6583 0, win_event_proc, 0, 0xdeadbeef, WINEVENT_OUTOFCONTEXT);
6584 ok(!hwinevent_hook, "SetWinEventHook with invalid tid should fail\n");
6585 ok(GetLastError() == ERROR_INVALID_THREAD_ID, "unexpected error %ld\n", GetLastError());
6588 SetLastError(0xdeadbeef);
6589 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(0, 0,
6590 0, win_event_proc, GetCurrentProcessId(), 0, WINEVENT_OUTOFCONTEXT);
6591 ok(hwinevent_hook != 0, "SetWinEventHook error %d\n", GetLastError());
6592 ok(GetLastError() == 0xdeadbeef, "unexpected error %d\n", GetLastError());
6593 ret = pUnhookWinEvent(hwinevent_hook);
6594 ok( ret, "UnhookWinEvent error %d\n", GetLastError());
6597 /* This call succeeds under win2k SP4, but fails under Wine.
6598 Does win2k test/use passed process id? */
6599 SetLastError(0xdeadbeef);
6600 hwinevent_hook = (HWINEVENTHOOK)pSetWinEventHook(EVENT_MIN, EVENT_MAX,
6601 0, win_event_proc, 0xdeadbeef, 0, WINEVENT_OUTOFCONTEXT);
6602 ok(hwinevent_hook != 0, "SetWinEventHook error %d\n", GetLastError());
6603 ok(GetLastError() == 0xdeadbeef, "unexpected error %d\n", GetLastError());
6604 ret = pUnhookWinEvent(hwinevent_hook);
6605 ok( ret, "UnhookWinEvent error %d\n", GetLastError());
6608 SetLastError(0xdeadbeef);
6609 ok(!pUnhookWinEvent((HWINEVENTHOOK)0xdeadbeef), "UnhookWinEvent succeeded\n");
6610 ok(GetLastError() == ERROR_INVALID_HANDLE || /* Win2k */
6611 GetLastError() == 0xdeadbeef, /* Win9x */
6612 "unexpected error %d\n", GetLastError());
6615 static const struct message ScrollWindowPaint1[] = {
6617 { WM_ERASEBKGND, sent|beginpaint },
6621 static const struct message ScrollWindowPaint2[] = {
6626 static void test_scrollwindowex(void)
6629 RECT rect={0,0,130,130};
6632 hwnd = CreateWindowExA(0, "TestWindowClass", "Test Scroll",
6633 WS_VISIBLE|WS_OVERLAPPEDWINDOW,
6634 100, 100, 200, 200, 0, 0, 0, NULL);
6635 ok (hwnd != 0, "Failed to create overlapped window\n");
6636 hchild = CreateWindowExA(0, "TestWindowClass", "Test child",
6637 WS_VISIBLE|WS_CAPTION|WS_CHILD,
6638 10, 10, 150, 150, hwnd, 0, 0, NULL);
6639 ok (hchild != 0, "Failed to create child\n");
6644 /* scroll without the child window */
6645 trace("start scroll\n");
6646 ScrollWindowEx( hwnd, 10, 10, &rect, NULL, NULL, NULL,
6647 SW_ERASE|SW_INVALIDATE);
6648 ok_sequence(WmEmptySeq, "ScrollWindowEx", 0);
6649 trace("end scroll\n");
6651 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6652 ok_sequence(ScrollWindowPaint1, "ScrollWindowEx", 0);
6654 /* Now without the SW_ERASE flag */
6655 trace("start scroll\n");
6656 ScrollWindowEx( hwnd, 10, 10, &rect, NULL, NULL, NULL, SW_INVALIDATE);
6657 ok_sequence(WmEmptySeq, "ScrollWindowEx", 0);
6658 trace("end scroll\n");
6660 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6661 ok_sequence(ScrollWindowPaint2, "ScrollWindowEx", 0);
6663 /* now scroll the child window as well */
6664 trace("start scroll\n");
6665 ScrollWindowEx( hwnd, 10, 10, &rect, NULL, NULL, NULL,
6666 SW_SCROLLCHILDREN|SW_ERASE|SW_INVALIDATE);
6667 todo_wine { /* wine sends WM_POSCHANGING, WM_POSCHANGED messages */
6668 /* windows sometimes a WM_MOVE */
6669 ok_sequence(WmEmptySeq, "ScrollWindowEx", 0);
6671 trace("end scroll\n");
6673 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6674 ok_sequence(ScrollWindowPaint1, "ScrollWindowEx", 0);
6676 /* now scroll with ScrollWindow() */
6677 trace("start scroll with ScrollWindow\n");
6678 ScrollWindow( hwnd, 5, 5, NULL, NULL);
6679 trace("end scroll\n");
6681 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6682 ok_sequence(ScrollWindowPaint1, "ScrollWindow", 0);
6684 ok(DestroyWindow(hchild), "failed to destroy window\n");
6685 ok(DestroyWindow(hwnd), "failed to destroy window\n");
6689 static const struct message destroy_window_with_children[] = {
6690 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 }, /* popup */
6691 { HCBT_DESTROYWND, hook|lparam, 0, WND_PARENT_ID }, /* parent */
6692 { HCBT_DESTROYWND, hook|lparam, 0, WND_POPUP_ID }, /* popup */
6693 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 }, /* popup */
6694 { WM_DESTROY, sent|wparam|lparam, 0, WND_POPUP_ID }, /* popup */
6695 { WM_CAPTURECHANGED, sent|wparam|lparam, 0, WND_POPUP_ID }, /* popup */
6696 { WM_NCDESTROY, sent|wparam|lparam, 0, WND_POPUP_ID }, /* popup */
6697 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, 0, 0 }, /* parent */
6698 { WM_DESTROY, sent|wparam|lparam, 0, WND_PARENT_ID }, /* parent */
6699 { WM_DESTROY, sent|wparam|lparam, 0, WND_CHILD_ID + 2 }, /* child2 */
6700 { WM_DESTROY, sent|wparam|lparam, 0, WND_CHILD_ID + 1 }, /* child1 */
6701 { WM_DESTROY, sent|wparam|lparam, 0, WND_CHILD_ID + 3 }, /* child3 */
6702 { WM_NCDESTROY, sent|wparam|lparam, 0, WND_CHILD_ID + 2 }, /* child2 */
6703 { WM_NCDESTROY, sent|wparam|lparam, 0, WND_CHILD_ID + 3 }, /* child3 */
6704 { WM_NCDESTROY, sent|wparam|lparam, 0, WND_CHILD_ID + 1 }, /* child1 */
6705 { WM_NCDESTROY, sent|wparam|lparam, 0, WND_PARENT_ID }, /* parent */
6709 static void test_DestroyWindow(void)
6712 HWND parent, child1, child2, child3, child4, test;
6713 UINT child_id = WND_CHILD_ID + 1;
6715 parent = CreateWindowExA(0, "TestWindowClass", NULL, WS_OVERLAPPEDWINDOW,
6716 100, 100, 200, 200, 0, 0, 0, NULL);
6717 assert(parent != 0);
6718 child1 = CreateWindowExA(0, "TestWindowClass", NULL, WS_CHILD,
6719 0, 0, 50, 50, parent, (HMENU)child_id++, 0, NULL);
6720 assert(child1 != 0);
6721 child2 = CreateWindowExA(0, "TestWindowClass", NULL, WS_CHILD,
6722 0, 0, 50, 50, GetDesktopWindow(), (HMENU)child_id++, 0, NULL);
6723 assert(child2 != 0);
6724 child3 = CreateWindowExA(0, "TestWindowClass", NULL, WS_CHILD,
6725 0, 0, 50, 50, child1, (HMENU)child_id++, 0, NULL);
6726 assert(child3 != 0);
6727 child4 = CreateWindowExA(0, "TestWindowClass", NULL, WS_POPUP,
6728 0, 0, 50, 50, parent, 0, 0, NULL);
6729 assert(child4 != 0);
6731 /* test owner/parent of child2 */
6732 test = GetParent(child2);
6733 ok(test == GetDesktopWindow(), "wrong parent %p\n", test);
6734 ok(!IsChild(parent, child2), "wrong parent/child %p/%p\n", parent, child2);
6736 test = pGetAncestor(child2, GA_PARENT);
6737 ok(test == GetDesktopWindow(), "wrong parent %p\n", test);
6739 test = GetWindow(child2, GW_OWNER);
6740 ok(!test, "wrong owner %p\n", test);
6742 test = SetParent(child2, parent);
6743 ok(test == GetDesktopWindow(), "wrong old parent %p\n", test);
6745 /* test owner/parent of the parent */
6746 test = GetParent(parent);
6747 ok(!test, "wrong parent %p\n", test);
6749 ok(!IsChild(GetDesktopWindow(), parent), "wrong parent/child %p/%p\n", GetDesktopWindow(), parent);
6752 test = pGetAncestor(parent, GA_PARENT);
6753 ok(test == GetDesktopWindow(), "wrong parent %p\n", test);
6755 test = GetWindow(parent, GW_OWNER);
6756 ok(!test, "wrong owner %p\n", test);
6758 /* test owner/parent of child1 */
6759 test = GetParent(child1);
6760 ok(test == parent, "wrong parent %p\n", test);
6761 ok(IsChild(parent, child1), "wrong parent/child %p/%p\n", parent, child1);
6763 test = pGetAncestor(child1, GA_PARENT);
6764 ok(test == parent, "wrong parent %p\n", test);
6766 test = GetWindow(child1, GW_OWNER);
6767 ok(!test, "wrong owner %p\n", test);
6769 /* test owner/parent of child2 */
6770 test = GetParent(child2);
6771 ok(test == parent, "wrong parent %p\n", test);
6772 ok(IsChild(parent, child2), "wrong parent/child %p/%p\n", parent, child2);
6774 test = pGetAncestor(child2, GA_PARENT);
6775 ok(test == parent, "wrong parent %p\n", test);
6777 test = GetWindow(child2, GW_OWNER);
6778 ok(!test, "wrong owner %p\n", test);
6780 /* test owner/parent of child3 */
6781 test = GetParent(child3);
6782 ok(test == child1, "wrong parent %p\n", test);
6783 ok(IsChild(parent, child3), "wrong parent/child %p/%p\n", parent, child3);
6785 test = pGetAncestor(child3, GA_PARENT);
6786 ok(test == child1, "wrong parent %p\n", test);
6788 test = GetWindow(child3, GW_OWNER);
6789 ok(!test, "wrong owner %p\n", test);
6791 /* test owner/parent of child4 */
6792 test = GetParent(child4);
6793 ok(test == parent, "wrong parent %p\n", test);
6794 ok(!IsChild(parent, child4), "wrong parent/child %p/%p\n", parent, child4);
6796 test = pGetAncestor(child4, GA_PARENT);
6797 ok(test == GetDesktopWindow(), "wrong parent %p\n", test);
6799 test = GetWindow(child4, GW_OWNER);
6800 ok(test == parent, "wrong owner %p\n", test);
6804 trace("parent %p, child1 %p, child2 %p, child3 %p, child4 %p\n",
6805 parent, child1, child2, child3, child4);
6808 test = GetCapture();
6809 ok(test == child4, "wrong capture window %p\n", test);
6811 test_DestroyWindow_flag = TRUE;
6812 ret = DestroyWindow(parent);
6813 ok( ret, "DestroyWindow() error %d\n", GetLastError());
6814 test_DestroyWindow_flag = FALSE;
6815 ok_sequence(destroy_window_with_children, "destroy window with children", 0);
6817 ok(!IsWindow(parent), "parent still exists\n");
6818 ok(!IsWindow(child1), "child1 still exists\n");
6819 ok(!IsWindow(child2), "child2 still exists\n");
6820 ok(!IsWindow(child3), "child3 still exists\n");
6821 ok(!IsWindow(child4), "child4 still exists\n");
6823 test = GetCapture();
6824 ok(!test, "wrong capture window %p\n", test);
6828 static const struct message WmDispatchPaint[] = {
6829 { WM_NCPAINT, sent },
6830 { WM_GETTEXT, sent|defwinproc|optional },
6831 { WM_GETTEXT, sent|defwinproc|optional },
6832 { WM_ERASEBKGND, sent },
6836 static LRESULT WINAPI DispatchMessageCheckProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
6838 if (message == WM_PAINT) return 0;
6839 return MsgCheckProcA( hwnd, message, wParam, lParam );
6842 static void test_DispatchMessage(void)
6847 HWND hwnd = CreateWindowA( "TestWindowClass", NULL, WS_OVERLAPPEDWINDOW,
6848 100, 100, 200, 200, 0, 0, 0, NULL);
6849 ShowWindow( hwnd, SW_SHOW );
6850 UpdateWindow( hwnd );
6851 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6853 SetWindowLongPtrA( hwnd, GWLP_WNDPROC, (LONG_PTR)DispatchMessageCheckProc );
6855 SetRect( &rect, -5, -5, 5, 5 );
6856 RedrawWindow( hwnd, &rect, 0, RDW_INVALIDATE|RDW_ERASE|RDW_FRAME );
6858 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE ))
6860 if (msg.message != WM_PAINT) DispatchMessage( &msg );
6864 DispatchMessage( &msg );
6865 /* DispatchMessage will send WM_NCPAINT if non client area is still invalid after WM_PAINT */
6866 if (!count) ok_sequence( WmDispatchPaint, "WmDispatchPaint", FALSE );
6867 else ok_sequence( WmEmptySeq, "WmEmpty", FALSE );
6868 if (++count > 10) break;
6871 ok( msg.message == WM_PAINT && count > 10, "WM_PAINT messages stopped\n" );
6873 trace("now without DispatchMessage\n");
6875 RedrawWindow( hwnd, &rect, 0, RDW_INVALIDATE|RDW_ERASE|RDW_FRAME );
6877 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE ))
6879 if (msg.message != WM_PAINT) DispatchMessage( &msg );
6882 HRGN hrgn = CreateRectRgn( 0, 0, 0, 0 );
6884 /* this will send WM_NCCPAINT just like DispatchMessage does */
6885 GetUpdateRgn( hwnd, hrgn, TRUE );
6886 ok_sequence( WmDispatchPaint, "WmDispatchPaint", FALSE );
6887 DeleteObject( hrgn );
6888 GetClientRect( hwnd, &rect );
6889 ValidateRect( hwnd, &rect ); /* this will stop WM_PAINTs */
6890 ok( !count, "Got multiple WM_PAINTs\n" );
6891 if (++count > 10) break;
6894 DestroyWindow(hwnd);
6898 static const struct message WmUser[] = {
6910 static DWORD CALLBACK send_msg_thread( LPVOID arg )
6912 struct sendmsg_info *info = arg;
6913 info->ret = SendMessageTimeoutA( info->hwnd, WM_USER, 0, 0, 0, info->timeout, NULL );
6914 if (!info->ret) ok( GetLastError() == ERROR_TIMEOUT, "unexpected error %d\n", GetLastError());
6918 static void wait_for_thread( HANDLE thread )
6920 while (MsgWaitForMultipleObjects(1, &thread, FALSE, INFINITE, QS_SENDMESSAGE) != WAIT_OBJECT_0)
6923 while (PeekMessageA( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage(&msg);
6927 static LRESULT WINAPI send_msg_delay_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
6929 if (message == WM_USER) Sleep(200);
6930 return MsgCheckProcA( hwnd, message, wParam, lParam );
6933 static void test_SendMessageTimeout(void)
6937 struct sendmsg_info info;
6940 info.hwnd = CreateWindowA( "TestWindowClass", NULL, WS_OVERLAPPEDWINDOW,
6941 100, 100, 200, 200, 0, 0, 0, NULL);
6942 while (PeekMessageA( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
6945 info.timeout = 1000;
6946 info.ret = 0xdeadbeef;
6947 thread = CreateThread( NULL, 0, send_msg_thread, &info, 0, &tid );
6948 wait_for_thread( thread );
6949 CloseHandle( thread );
6950 ok( info.ret == 1, "SendMessageTimeout failed\n" );
6951 ok_sequence( WmUser, "WmUser", FALSE );
6954 info.ret = 0xdeadbeef;
6955 thread = CreateThread( NULL, 0, send_msg_thread, &info, 0, &tid );
6956 Sleep(100); /* SendMessageTimeout should timeout here */
6957 wait_for_thread( thread );
6958 CloseHandle( thread );
6959 ok( info.ret == 0, "SendMessageTimeout succeeded\n" );
6960 ok_sequence( WmEmptySeq, "WmEmptySeq", FALSE );
6962 /* 0 means infinite timeout */
6964 info.ret = 0xdeadbeef;
6965 thread = CreateThread( NULL, 0, send_msg_thread, &info, 0, &tid );
6967 wait_for_thread( thread );
6968 CloseHandle( thread );
6969 ok( info.ret == 1, "SendMessageTimeout failed\n" );
6970 ok_sequence( WmUser, "WmUser", FALSE );
6972 /* timeout is treated as signed despite the prototype */
6973 info.timeout = 0x7fffffff;
6974 info.ret = 0xdeadbeef;
6975 thread = CreateThread( NULL, 0, send_msg_thread, &info, 0, &tid );
6977 wait_for_thread( thread );
6978 CloseHandle( thread );
6979 ok( info.ret == 1, "SendMessageTimeout failed\n" );
6980 ok_sequence( WmUser, "WmUser", FALSE );
6982 info.timeout = 0x80000000;
6983 info.ret = 0xdeadbeef;
6984 thread = CreateThread( NULL, 0, send_msg_thread, &info, 0, &tid );
6986 wait_for_thread( thread );
6987 CloseHandle( thread );
6988 ok( info.ret == 0, "SendMessageTimeout succeeded\n" );
6989 ok_sequence( WmEmptySeq, "WmEmptySeq", FALSE );
6991 /* now check for timeout during message processing */
6992 SetWindowLongPtrA( info.hwnd, GWLP_WNDPROC, (LONG_PTR)send_msg_delay_proc );
6994 info.ret = 0xdeadbeef;
6995 thread = CreateThread( NULL, 0, send_msg_thread, &info, 0, &tid );
6996 wait_for_thread( thread );
6997 CloseHandle( thread );
6998 /* we should timeout but still get the message */
6999 ok( info.ret == 0, "SendMessageTimeout failed\n" );
7000 ok_sequence( WmUser, "WmUser", FALSE );
7002 DestroyWindow( info.hwnd );
7006 /****************** edit message test *************************/
7007 #define ID_EDIT 0x1234
7008 static const struct message sl_edit_setfocus[] =
7010 { HCBT_SETFOCUS, hook },
7011 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
7012 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
7013 { WM_SETFOCUS, sent|wparam, 0 },
7014 { WM_CTLCOLOREDIT, sent|parent },
7015 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7016 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7017 { WM_COMMAND, sent|parent|wparam, MAKEWPARAM(ID_EDIT, EN_SETFOCUS) },
7020 static const struct message ml_edit_setfocus[] =
7022 { HCBT_SETFOCUS, hook },
7023 { WM_IME_SETCONTEXT, sent|wparam|optional, 1 },
7024 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
7025 { WM_SETFOCUS, sent|wparam, 0 },
7026 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7027 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7028 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7029 { WM_COMMAND, sent|parent|wparam, MAKEWPARAM(ID_EDIT, EN_SETFOCUS) },
7032 static const struct message sl_edit_killfocus[] =
7034 { HCBT_SETFOCUS, hook },
7035 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
7036 { WM_KILLFOCUS, sent|wparam, 0 },
7037 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7038 { EVENT_OBJECT_DESTROY, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7039 { WM_COMMAND, sent|parent|wparam, MAKEWPARAM(ID_EDIT, EN_KILLFOCUS) },
7040 { WM_IME_SETCONTEXT, sent|wparam|optional, 0 },
7043 static const struct message sl_edit_lbutton_dblclk[] =
7045 { WM_LBUTTONDBLCLK, sent },
7046 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 },
7049 static const struct message sl_edit_lbutton_down[] =
7051 { WM_LBUTTONDOWN, sent|wparam|lparam, 0, 0 },
7052 { HCBT_SETFOCUS, hook },
7053 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
7054 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
7055 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
7056 { WM_CTLCOLOREDIT, sent|parent },
7057 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7058 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7059 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7060 { WM_COMMAND, sent|parent|wparam, MAKEWPARAM(ID_EDIT, EN_SETFOCUS) },
7061 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 },
7062 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7063 { EVENT_OBJECT_LOCATIONCHANGE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7064 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7067 static const struct message ml_edit_lbutton_down[] =
7069 { WM_LBUTTONDOWN, sent|wparam|lparam, 0, 0 },
7070 { EVENT_SYSTEM_CAPTURESTART, winevent_hook|wparam|lparam, 0, 0 },
7071 { HCBT_SETFOCUS, hook },
7072 { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
7073 { EVENT_OBJECT_FOCUS, winevent_hook|wparam|lparam, OBJID_CLIENT, 0 },
7074 { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
7075 { EVENT_OBJECT_CREATE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7076 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7077 { WM_COMMAND, sent|parent|wparam, MAKEWPARAM(ID_EDIT, EN_SETFOCUS) },
7080 static const struct message sl_edit_lbutton_up[] =
7082 { WM_LBUTTONUP, sent|wparam|lparam, 0, 0 },
7083 { EVENT_OBJECT_HIDE, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7084 { EVENT_SYSTEM_CAPTUREEND, winevent_hook|wparam|lparam, 0, 0 },
7085 { WM_CAPTURECHANGED, sent|defwinproc },
7086 { EVENT_OBJECT_SHOW, winevent_hook|wparam|lparam, OBJID_CARET, 0 },
7089 static const struct message ml_edit_lbutton_up[] =
7091 { WM_LBUTTONUP, sent|wparam|lparam, 0, 0 },
7092 { EVENT_SYSTEM_CAPTUREEND, winevent_hook|wparam|lparam, 0, 0 },
7093 { WM_CAPTURECHANGED, sent|defwinproc },
7097 static WNDPROC old_edit_proc;
7099 static LRESULT CALLBACK edit_hook_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
7101 static long defwndproc_counter = 0;
7105 trace("edit: %p, %04x, %08x, %08lx\n", hwnd, message, wParam, lParam);
7107 /* explicitly ignore WM_GETICON message */
7108 if (message == WM_GETICON) return 0;
7110 msg.message = message;
7111 msg.flags = sent|wparam|lparam;
7112 if (defwndproc_counter) msg.flags |= defwinproc;
7113 msg.wParam = wParam;
7114 msg.lParam = lParam;
7117 defwndproc_counter++;
7118 ret = CallWindowProcA(old_edit_proc, hwnd, message, wParam, lParam);
7119 defwndproc_counter--;
7124 static void subclass_edit(void)
7128 if (!GetClassInfoA(0, "edit", &cls)) assert(0);
7130 old_edit_proc = cls.lpfnWndProc;
7132 cls.hInstance = GetModuleHandle(0);
7133 cls.lpfnWndProc = edit_hook_proc;
7134 cls.lpszClassName = "my_edit_class";
7135 if (!RegisterClassA(&cls)) assert(0);
7138 static void test_edit_messages(void)
7144 log_all_parent_messages++;
7146 parent = CreateWindowExA(0, "TestParentClass", "Test parent", WS_OVERLAPPEDWINDOW | WS_VISIBLE,
7147 100, 100, 200, 200, 0, 0, 0, NULL);
7148 ok (parent != 0, "Failed to create parent window\n");
7150 /* test single line edit */
7151 hwnd = CreateWindowExA(0, "my_edit_class", "test", WS_CHILD,
7152 0, 0, 80, 20, parent, (HMENU)ID_EDIT, 0, NULL);
7153 ok(hwnd != 0, "Failed to create edit window\n");
7155 dlg_code = SendMessageA(hwnd, WM_GETDLGCODE, 0, 0);
7156 ok(dlg_code == (DLGC_WANTCHARS|DLGC_HASSETSEL|DLGC_WANTARROWS), "wrong dlg_code %08x\n", dlg_code);
7158 ShowWindow(hwnd, SW_SHOW);
7164 ok_sequence(sl_edit_setfocus, "SetFocus(hwnd) on an edit", FALSE);
7167 ok_sequence(sl_edit_killfocus, "SetFocus(0) on an edit", FALSE);
7173 SendMessageA(hwnd, WM_LBUTTONDBLCLK, 0, 0);
7174 ok_sequence(sl_edit_lbutton_dblclk, "WM_LBUTTONDBLCLK on an edit", FALSE);
7180 SendMessageA(hwnd, WM_LBUTTONDOWN, 0, 0);
7181 ok_sequence(sl_edit_lbutton_down, "WM_LBUTTONDOWN on an edit", FALSE);
7183 SendMessageA(hwnd, WM_LBUTTONUP, 0, 0);
7184 ok_sequence(sl_edit_lbutton_up, "WM_LBUTTONUP on an edit", FALSE);
7186 DestroyWindow(hwnd);
7188 /* test multiline edit */
7189 hwnd = CreateWindowExA(0, "my_edit_class", "test", WS_CHILD | ES_MULTILINE,
7190 0, 0, 80, 20, parent, (HMENU)ID_EDIT, 0, NULL);
7191 ok(hwnd != 0, "Failed to create edit window\n");
7193 dlg_code = SendMessageA(hwnd, WM_GETDLGCODE, 0, 0);
7194 ok(dlg_code == (DLGC_WANTCHARS|DLGC_HASSETSEL|DLGC_WANTARROWS|DLGC_WANTALLKEYS),
7195 "wrong dlg_code %08x\n", dlg_code);
7197 ShowWindow(hwnd, SW_SHOW);
7203 ok_sequence(ml_edit_setfocus, "SetFocus(hwnd) on multiline edit", FALSE);
7206 ok_sequence(sl_edit_killfocus, "SetFocus(0) on multiline edit", FALSE);
7212 SendMessageA(hwnd, WM_LBUTTONDBLCLK, 0, 0);
7213 ok_sequence(sl_edit_lbutton_dblclk, "WM_LBUTTONDBLCLK on multiline edit", FALSE);
7219 SendMessageA(hwnd, WM_LBUTTONDOWN, 0, 0);
7220 ok_sequence(ml_edit_lbutton_down, "WM_LBUTTONDOWN on multiline edit", FALSE);
7222 SendMessageA(hwnd, WM_LBUTTONUP, 0, 0);
7223 ok_sequence(ml_edit_lbutton_up, "WM_LBUTTONUP on multiline edit", FALSE);
7225 DestroyWindow(hwnd);
7226 DestroyWindow(parent);
7228 log_all_parent_messages--;
7231 /**************************** End of Edit test ******************************/
7233 #define EV_START_STOP 0
7234 #define EV_SENDMSG 1
7240 HANDLE hevent[3]; /* 0 - start/stop, 1 - SendMessage, 2 - ack */
7243 static DWORD CALLBACK send_msg_thread_2(void *param)
7246 struct peekmsg_info *info = param;
7248 trace("thread: waiting for start\n");
7249 WaitForSingleObject(info->hevent[EV_START_STOP], INFINITE);
7250 trace("thread: looping\n");
7254 ret = WaitForMultipleObjects(2, info->hevent, FALSE, INFINITE);
7258 case WAIT_OBJECT_0 + EV_START_STOP:
7259 trace("thread: exiting\n");
7262 case WAIT_OBJECT_0 + EV_SENDMSG:
7263 trace("thread: sending message\n");
7264 SendNotifyMessageA(info->hwnd, WM_USER, 0, 0);
7265 SetEvent(info->hevent[EV_ACK]);
7269 trace("unexpected return: %04x\n", ret);
7277 static void test_PeekMessage(void)
7282 UINT qs_all_input = QS_ALLINPUT;
7283 UINT qs_input = QS_INPUT;
7285 struct peekmsg_info info;
7287 info.hwnd = CreateWindowA("TestWindowClass", NULL, WS_OVERLAPPEDWINDOW,
7288 100, 100, 200, 200, 0, 0, 0, NULL);
7290 ShowWindow(info.hwnd, SW_SHOW);
7291 UpdateWindow(info.hwnd);
7292 SetFocus(info.hwnd);
7294 info.hevent[EV_START_STOP] = CreateEventA(NULL, 0, 0, NULL);
7295 info.hevent[EV_SENDMSG] = CreateEventA(NULL, 0, 0, NULL);
7296 info.hevent[EV_ACK] = CreateEventA(NULL, 0, 0, NULL);
7298 hthread = CreateThread(NULL, 0, send_msg_thread_2, &info, 0, &tid);
7301 trace("signalling to start looping\n");
7302 SetEvent(info.hevent[EV_START_STOP]);
7304 while (PeekMessageA(&msg, 0, 0, 0, PM_REMOVE)) DispatchMessageA(&msg);
7307 SetLastError(0xdeadbeef);
7308 qstatus = GetQueueStatus(qs_all_input);
7309 if (GetLastError() == ERROR_INVALID_FLAGS)
7311 trace("QS_RAWINPUT not supported on this platform\n");
7312 qs_all_input &= ~QS_RAWINPUT;
7313 qs_input &= ~QS_RAWINPUT;
7315 ok(qstatus == 0, "wrong qstatus %08x\n", qstatus);
7317 trace("signalling to send message\n");
7318 SetEvent(info.hevent[EV_SENDMSG]);
7319 WaitForSingleObject(info.hevent[EV_ACK], INFINITE);
7321 /* pass invalid QS_xxxx flags */
7322 SetLastError(0xdeadbeef);
7323 qstatus = GetQueueStatus(0xffffffff);
7324 ok(qstatus == 0, "GetQueueStatus should fail: %08x\n", qstatus);
7325 ok(GetLastError() == ERROR_INVALID_FLAGS, "wrong error %d\n", GetLastError());
7327 qstatus = GetQueueStatus(qs_all_input);
7328 ok(qstatus == MAKELONG(QS_SENDMESSAGE, QS_SENDMESSAGE),
7329 "wrong qstatus %08x\n", qstatus);
7332 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE);
7334 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7336 ok_sequence(WmUser, "WmUser", FALSE);
7338 qstatus = GetQueueStatus(qs_all_input);
7339 ok(qstatus == 0, "wrong qstatus %08x\n", qstatus);
7341 keybd_event('N', 0, 0, 0);
7342 keybd_event('N', 0, KEYEVENTF_KEYUP, 0);
7343 qstatus = GetQueueStatus(qs_all_input);
7344 ok(qstatus == MAKELONG(QS_KEY, QS_KEY),
7345 "wrong qstatus %08x\n", qstatus);
7347 PostMessageA(info.hwnd, WM_CHAR, 'z', 0);
7348 qstatus = GetQueueStatus(qs_all_input);
7349 ok(qstatus == MAKELONG(QS_POSTMESSAGE, QS_POSTMESSAGE|QS_KEY),
7350 "wrong qstatus %08x\n", qstatus);
7352 InvalidateRect(info.hwnd, NULL, FALSE);
7353 qstatus = GetQueueStatus(qs_all_input);
7354 ok(qstatus == MAKELONG(QS_PAINT, QS_PAINT|QS_POSTMESSAGE|QS_KEY),
7355 "wrong qstatus %08x\n", qstatus);
7357 trace("signalling to send message\n");
7358 SetEvent(info.hevent[EV_SENDMSG]);
7359 WaitForSingleObject(info.hevent[EV_ACK], INFINITE);
7361 qstatus = GetQueueStatus(qs_all_input);
7362 ok(qstatus == MAKELONG(QS_SENDMESSAGE, QS_SENDMESSAGE|QS_PAINT|QS_POSTMESSAGE|QS_KEY),
7363 "wrong qstatus %08x\n", qstatus);
7366 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | (qs_input << 16));
7369 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7372 ok_sequence(WmUser, "WmUser", FALSE);
7374 qstatus = GetQueueStatus(qs_all_input);
7376 ok(qstatus == MAKELONG(0, QS_PAINT|QS_POSTMESSAGE|QS_KEY),
7377 "wrong qstatus %08x\n", qstatus);
7380 trace("signalling to send message\n");
7381 SetEvent(info.hevent[EV_SENDMSG]);
7382 WaitForSingleObject(info.hevent[EV_ACK], INFINITE);
7384 qstatus = GetQueueStatus(qs_all_input);
7386 ok(qstatus == MAKELONG(QS_SENDMESSAGE, QS_SENDMESSAGE|QS_PAINT|QS_POSTMESSAGE|QS_KEY),
7387 "wrong qstatus %08x\n", qstatus);
7391 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | PM_QS_POSTMESSAGE);
7394 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7397 ok_sequence(WmUser, "WmUser", FALSE);
7399 qstatus = GetQueueStatus(qs_all_input);
7401 ok(qstatus == MAKELONG(0, QS_PAINT|QS_POSTMESSAGE|QS_KEY),
7402 "wrong qstatus %08x\n", qstatus);
7406 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | PM_QS_POSTMESSAGE);
7408 ok(ret && msg.message == WM_CHAR && msg.wParam == 'z',
7409 "got %d and %04x wParam %08x instead of TRUE and WM_CHAR wParam 'z'\n",
7410 ret, msg.message, msg.wParam);
7412 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7414 qstatus = GetQueueStatus(qs_all_input);
7416 ok(qstatus == MAKELONG(0, QS_PAINT|QS_KEY),
7417 "wrong qstatus %08x\n", qstatus);
7421 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | PM_QS_POSTMESSAGE);
7424 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7427 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7429 qstatus = GetQueueStatus(qs_all_input);
7431 ok(qstatus == MAKELONG(0, QS_PAINT|QS_KEY),
7432 "wrong qstatus %08x\n", qstatus);
7436 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | PM_QS_PAINT);
7437 ok(ret && msg.message == WM_PAINT,
7438 "got %d and %04x instead of TRUE and WM_PAINT\n", ret, msg.message);
7439 DispatchMessageA(&msg);
7440 ok_sequence(WmPaint, "WmPaint", FALSE);
7442 qstatus = GetQueueStatus(qs_all_input);
7444 ok(qstatus == MAKELONG(0, QS_KEY),
7445 "wrong qstatus %08x\n", qstatus);
7449 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | PM_QS_PAINT);
7451 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7453 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7455 qstatus = GetQueueStatus(qs_all_input);
7457 ok(qstatus == MAKELONG(0, QS_KEY),
7458 "wrong qstatus %08x\n", qstatus);
7461 trace("signalling to send message\n");
7462 SetEvent(info.hevent[EV_SENDMSG]);
7463 WaitForSingleObject(info.hevent[EV_ACK], INFINITE);
7465 qstatus = GetQueueStatus(qs_all_input);
7467 ok(qstatus == MAKELONG(QS_SENDMESSAGE, QS_SENDMESSAGE|QS_KEY),
7468 "wrong qstatus %08x\n", qstatus);
7471 PostMessageA(info.hwnd, WM_CHAR, 'z', 0);
7473 qstatus = GetQueueStatus(qs_all_input);
7475 ok(qstatus == MAKELONG(QS_POSTMESSAGE, QS_SENDMESSAGE|QS_POSTMESSAGE|QS_KEY),
7476 "wrong qstatus %08x\n", qstatus);
7480 ret = PeekMessageA(&msg, 0, WM_CHAR, WM_CHAR, PM_REMOVE);
7481 ok(ret && msg.message == WM_CHAR && msg.wParam == 'z',
7482 "got %d and %04x wParam %08x instead of TRUE and WM_CHAR wParam 'z'\n",
7483 ret, msg.message, msg.wParam);
7484 ok_sequence(WmUser, "WmUser", FALSE);
7486 qstatus = GetQueueStatus(qs_all_input);
7488 ok(qstatus == MAKELONG(0, QS_KEY),
7489 "wrong qstatus %08x\n", qstatus);
7493 ret = PeekMessageA(&msg, 0, WM_CHAR, WM_CHAR, PM_REMOVE);
7495 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7497 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7499 qstatus = GetQueueStatus(qs_all_input);
7501 ok(qstatus == MAKELONG(0, QS_KEY),
7502 "wrong qstatus %08x\n", qstatus);
7505 PostMessageA(info.hwnd, WM_CHAR, 'z', 0);
7507 qstatus = GetQueueStatus(qs_all_input);
7509 ok(qstatus == MAKELONG(QS_POSTMESSAGE, QS_POSTMESSAGE|QS_KEY),
7510 "wrong qstatus %08x\n", qstatus);
7513 trace("signalling to send message\n");
7514 SetEvent(info.hevent[EV_SENDMSG]);
7515 WaitForSingleObject(info.hevent[EV_ACK], INFINITE);
7517 qstatus = GetQueueStatus(qs_all_input);
7519 ok(qstatus == MAKELONG(QS_SENDMESSAGE, QS_SENDMESSAGE|QS_POSTMESSAGE|QS_KEY),
7520 "wrong qstatus %08x\n", qstatus);
7524 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | (QS_KEY << 16));
7527 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7530 ok_sequence(WmUser, "WmUser", FALSE);
7532 qstatus = GetQueueStatus(qs_all_input);
7534 ok(qstatus == MAKELONG(0, QS_POSTMESSAGE|QS_KEY),
7535 "wrong qstatus %08x\n", qstatus);
7539 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | (QS_RAWINPUT << 16));
7541 ok(ret && msg.message == WM_KEYDOWN && msg.wParam == 'N',
7542 "got %d and %04x wParam %08x instead of TRUE and WM_KEYDOWN wParam 'N'\n",
7543 ret, msg.message, msg.wParam);
7545 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7547 qstatus = GetQueueStatus(qs_all_input);
7549 ok(qstatus == MAKELONG(0, QS_POSTMESSAGE|QS_KEY),
7550 "wrong qstatus %08x\n", qstatus);
7554 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | (QS_RAWINPUT << 16));
7556 ok(ret && msg.message == WM_KEYUP && msg.wParam == 'N',
7557 "got %d and %04x wParam %08x instead of TRUE and WM_KEYUP wParam 'N'\n",
7558 ret, msg.message, msg.wParam);
7560 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7562 qstatus = GetQueueStatus(qs_all_input);
7564 ok(qstatus == MAKELONG(0, QS_POSTMESSAGE),
7565 "wrong qstatus %08x\n", qstatus);
7569 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE | PM_QS_SENDMESSAGE);
7571 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7573 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7575 qstatus = GetQueueStatus(qs_all_input);
7577 ok(qstatus == MAKELONG(0, QS_POSTMESSAGE),
7578 "wrong qstatus %08x\n", qstatus);
7582 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE);
7584 ok(ret && msg.message == WM_CHAR && msg.wParam == 'z',
7585 "got %d and %04x wParam %08x instead of TRUE and WM_CHAR wParam 'z'\n",
7586 ret, msg.message, msg.wParam);
7588 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7590 qstatus = GetQueueStatus(qs_all_input);
7592 "wrong qstatus %08x\n", qstatus);
7595 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE);
7597 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7599 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7601 qstatus = GetQueueStatus(qs_all_input);
7603 "wrong qstatus %08x\n", qstatus);
7605 /* test whether presence of the quit flag in the queue affects
7608 PostQuitMessage(0x1234abcd);
7610 qstatus = GetQueueStatus(qs_all_input);
7611 ok(qstatus == MAKELONG(QS_POSTMESSAGE, QS_POSTMESSAGE),
7612 "wrong qstatus %08x\n", qstatus);
7614 PostMessageA(info.hwnd, WM_USER, 0, 0);
7616 qstatus = GetQueueStatus(qs_all_input);
7617 ok(qstatus == MAKELONG(QS_POSTMESSAGE, QS_POSTMESSAGE),
7618 "wrong qstatus %08x\n", qstatus);
7621 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE);
7622 ok(ret && msg.message == WM_USER,
7623 "got %d and %04x instead of TRUE and WM_USER\n", ret, msg.message);
7624 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7626 qstatus = GetQueueStatus(qs_all_input);
7627 ok(qstatus == MAKELONG(0, QS_POSTMESSAGE),
7628 "wrong qstatus %08x\n", qstatus);
7631 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE);
7632 ok(ret && msg.message == WM_QUIT,
7633 "got %d and %04x instead of TRUE and WM_QUIT\n", ret, msg.message);
7634 ok(msg.wParam == 0x1234abcd, "got wParam %08x instead of 0x1234abcd\n", msg.wParam);
7635 ok(msg.lParam == 0, "got lParam %08lx instead of 0\n", msg.lParam);
7636 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7638 qstatus = GetQueueStatus(qs_all_input);
7640 ok(qstatus == MAKELONG(0, QS_POSTMESSAGE),
7641 "wrong qstatus %08x\n", qstatus);
7645 ret = PeekMessageA(&msg, 0, 0, 0, PM_REMOVE);
7647 "PeekMessageA should have returned FALSE instead of msg %04x\n",
7649 ok_sequence(WmEmptySeq, "WmEmptySeq", FALSE);
7651 qstatus = GetQueueStatus(qs_all_input);
7653 "wrong qstatus %08x\n", qstatus);
7655 trace("signalling to exit\n");
7656 SetEvent(info.hevent[EV_START_STOP]);
7658 WaitForSingleObject(hthread, INFINITE);
7660 CloseHandle(hthread);
7661 CloseHandle(info.hevent[0]);
7662 CloseHandle(info.hevent[1]);
7663 CloseHandle(info.hevent[2]);
7665 DestroyWindow(info.hwnd);
7669 static void test_quit_message(void)
7674 /* test using PostQuitMessage */
7675 PostQuitMessage(0xbeef);
7677 ret = PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE);
7678 ok(ret, "PeekMessage failed with error %d\n", GetLastError());
7679 ok(msg.message == WM_QUIT, "Received message 0x%04x instead of WM_QUIT\n", msg.message);
7680 ok(msg.wParam == 0xbeef, "wParam was 0x%x instead of 0xbeef\n", msg.wParam);
7682 ret = PostThreadMessage(GetCurrentThreadId(), WM_USER, 0, 0);
7683 ok(ret, "PostMessage failed with error %d\n", GetLastError());
7685 ret = GetMessage(&msg, NULL, 0, 0);
7686 ok(ret > 0, "GetMessage failed with error %d\n", GetLastError());
7687 ok(msg.message == WM_USER, "Received message 0x%04x instead of WM_USER\n", msg.message);
7689 /* note: WM_QUIT message received after WM_USER message */
7690 ret = GetMessage(&msg, NULL, 0, 0);
7691 ok(!ret, "GetMessage return %d with error %d instead of FALSE\n", ret, GetLastError());
7692 ok(msg.message == WM_QUIT, "Received message 0x%04x instead of WM_QUIT\n", msg.message);
7693 ok(msg.wParam == 0xbeef, "wParam was 0x%x instead of 0xbeef\n", msg.wParam);
7695 ret = PeekMessage(&msg, NULL, 0, 0, PM_REMOVE);
7696 ok( !ret || msg.message != WM_QUIT, "Received WM_QUIT again\n" );
7698 /* now test with PostThreadMessage - different behaviour! */
7699 PostThreadMessage(GetCurrentThreadId(), WM_QUIT, 0xdead, 0);
7701 ret = PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE);
7702 ok(ret, "PeekMessage failed with error %d\n", GetLastError());
7703 ok(msg.message == WM_QUIT, "Received message 0x%04x instead of WM_QUIT\n", msg.message);
7704 ok(msg.wParam == 0xdead, "wParam was 0x%x instead of 0xdead\n", msg.wParam);
7706 ret = PostThreadMessage(GetCurrentThreadId(), WM_USER, 0, 0);
7707 ok(ret, "PostMessage failed with error %d\n", GetLastError());
7709 /* note: we receive the WM_QUIT message first this time */
7710 ret = GetMessage(&msg, NULL, 0, 0);
7711 ok(!ret, "GetMessage return %d with error %d instead of FALSE\n", ret, GetLastError());
7712 ok(msg.message == WM_QUIT, "Received message 0x%04x instead of WM_QUIT\n", msg.message);
7713 ok(msg.wParam == 0xdead, "wParam was 0x%x instead of 0xdead\n", msg.wParam);
7715 ret = GetMessage(&msg, NULL, 0, 0);
7716 ok(ret > 0, "GetMessage failed with error %d\n", GetLastError());
7717 ok(msg.message == WM_USER, "Received message 0x%04x instead of WM_USER\n", msg.message);
7720 static const struct message WmMouseHoverSeq[] = {
7721 { WM_SYSTIMER, sent },
7722 { WM_MOUSEHOVER, sent|wparam, 0 },
7726 static void pump_msg_loop_timeout(DWORD timeout, BOOL inject_mouse_move)
7729 DWORD start_ticks, end_ticks;
7731 start_ticks = GetTickCount();
7732 /* add some deviation (5%) to cover not expected delays */
7733 start_ticks += timeout / 20;
7737 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE))
7739 /* Timer proc messages are not dispatched to the window proc,
7740 * and therefore not logged.
7742 if (msg.message == WM_TIMER || msg.message == WM_SYSTIMER)
7744 struct message s_msg;
7746 s_msg.message = msg.message;
7747 s_msg.flags = sent|wparam|lparam;
7748 s_msg.wParam = msg.wParam;
7749 s_msg.lParam = msg.lParam;
7750 add_message(&s_msg);
7752 DispatchMessage(&msg);
7755 end_ticks = GetTickCount();
7757 /* inject WM_MOUSEMOVE to see how it changes tracking */
7758 if (inject_mouse_move && start_ticks + timeout / 2 >= end_ticks)
7760 mouse_event(MOUSEEVENTF_MOVE, -1, 0, 0, 0);
7761 mouse_event(MOUSEEVENTF_MOVE, 1, 0, 0, 0);
7763 inject_mouse_move = FALSE;
7765 } while (start_ticks + timeout >= end_ticks);
7768 static void test_TrackMouseEvent(void)
7771 TRACKMOUSEEVENT tme;
7774 RECT rc_parent, rc_child;
7775 UINT default_hover_time, hover_width = 0, hover_height = 0;
7777 #define track_hover(track_hwnd, track_hover_time) \
7778 tme.cbSize = sizeof(tme); \
7779 tme.dwFlags = TME_HOVER; \
7780 tme.hwndTrack = track_hwnd; \
7781 tme.dwHoverTime = track_hover_time; \
7782 SetLastError(0xdeadbeef); \
7783 ret = TrackMouseEvent(&tme); \
7784 ok(ret, "TrackMouseEvent(TME_HOVER) error %d\n", GetLastError())
7786 #define track_query(expected_track_flags, expected_track_hwnd, expected_hover_time) \
7787 tme.cbSize = sizeof(tme); \
7788 tme.dwFlags = TME_QUERY; \
7789 tme.hwndTrack = (HWND)0xdeadbeef; \
7790 tme.dwHoverTime = 0xdeadbeef; \
7791 SetLastError(0xdeadbeef); \
7792 ret = TrackMouseEvent(&tme); \
7793 ok(ret, "TrackMouseEvent(TME_QUERY) error %d\n", GetLastError());\
7794 ok(tme.cbSize == sizeof(tme), "wrong tme.cbSize %u\n", tme.cbSize); \
7795 ok(tme.dwFlags == (expected_track_flags), \
7796 "wrong tme.dwFlags %08x, expected %08x\n", tme.dwFlags, (expected_track_flags)); \
7797 ok(tme.hwndTrack == (expected_track_hwnd), \
7798 "wrong tme.hwndTrack %p, expected %p\n", tme.hwndTrack, (expected_track_hwnd)); \
7799 ok(tme.dwHoverTime == (expected_hover_time), \
7800 "wrong tme.dwHoverTime %u, expected %u\n", tme.dwHoverTime, (expected_hover_time))
7802 #define track_hover_cancel(track_hwnd) \
7803 tme.cbSize = sizeof(tme); \
7804 tme.dwFlags = TME_HOVER | TME_CANCEL; \
7805 tme.hwndTrack = track_hwnd; \
7806 tme.dwHoverTime = 0xdeadbeef; \
7807 SetLastError(0xdeadbeef); \
7808 ret = TrackMouseEvent(&tme); \
7809 ok(ret, "TrackMouseEvent(TME_HOVER | TME_CANCEL) error %d\n", GetLastError())
7811 default_hover_time = 0xdeadbeef;
7812 ret = SystemParametersInfo(SPI_GETMOUSEHOVERTIME, 0, &default_hover_time, 0);
7813 ok(ret, "SystemParametersInfo(SPI_GETMOUSEHOVERTIME) failed\n");
7814 trace("SPI_GETMOUSEHOVERTIME returned %u ms\n", default_hover_time);
7816 SystemParametersInfo(SPI_GETMOUSEHOVERWIDTH, 0, &hover_width, 0);
7817 SystemParametersInfo(SPI_GETMOUSEHOVERHEIGHT, 0, &hover_height, 0);
7818 trace("hover rect is %u x %d\n", hover_width, hover_height);
7820 hwnd = CreateWindowEx(0, "TestWindowClass", NULL,
7821 WS_OVERLAPPEDWINDOW | WS_VISIBLE,
7822 CW_USEDEFAULT, CW_USEDEFAULT, 300, 300, 0,
7826 hchild = CreateWindowEx(0, "TestWindowClass", NULL,
7827 WS_CHILD | WS_BORDER | WS_VISIBLE,
7828 50, 50, 200, 200, hwnd,
7836 tme.dwFlags = TME_QUERY;
7837 tme.hwndTrack = (HWND)0xdeadbeef;
7838 tme.dwHoverTime = 0xdeadbeef;
7839 SetLastError(0xdeadbeef);
7840 ret = TrackMouseEvent(&tme);
7841 ok(!ret, "TrackMouseEvent should fail\n");
7842 ok(GetLastError() == ERROR_INVALID_PARAMETER, "not expected error %d\n", GetLastError());
7844 tme.cbSize = sizeof(tme);
7845 tme.dwFlags = TME_HOVER;
7846 tme.hwndTrack = (HWND)0xdeadbeef;
7847 tme.dwHoverTime = 0xdeadbeef;
7848 SetLastError(0xdeadbeef);
7849 ret = TrackMouseEvent(&tme);
7850 ok(!ret, "TrackMouseEvent should fail\n");
7851 ok(GetLastError() == ERROR_INVALID_WINDOW_HANDLE, "not expected error %d\n", GetLastError());
7853 tme.cbSize = sizeof(tme);
7854 tme.dwFlags = TME_HOVER | TME_CANCEL;
7855 tme.hwndTrack = (HWND)0xdeadbeef;
7856 tme.dwHoverTime = 0xdeadbeef;
7857 SetLastError(0xdeadbeef);
7858 ret = TrackMouseEvent(&tme);
7859 ok(!ret, "TrackMouseEvent should fail\n");
7860 ok(GetLastError() == ERROR_INVALID_WINDOW_HANDLE, "not expected error %d\n", GetLastError());
7862 GetWindowRect(hwnd, &rc_parent);
7863 GetWindowRect(hchild, &rc_child);
7864 SetCursorPos(rc_child.left - 10, rc_child.top - 10);
7866 /* Process messages so that the system updates its internal current
7867 * window and hittest, otherwise TrackMouseEvent calls don't have any
7870 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) DispatchMessage(&msg);
7873 track_query(0, NULL, 0);
7874 track_hover(hchild, 0);
7875 track_query(0, NULL, 0);
7877 while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) DispatchMessage(&msg);
7880 track_hover(hwnd, 0);
7881 track_query(TME_HOVER, hwnd, default_hover_time);
7883 pump_msg_loop_timeout(default_hover_time, FALSE);
7884 ok_sequence(WmMouseHoverSeq, "WmMouseHoverSeq", FALSE);
7886 track_query(0, NULL, 0);
7888 track_hover(hwnd, HOVER_DEFAULT);
7889 track_query(TME_HOVER, hwnd, default_hover_time);
7891 Sleep(default_hover_time / 2);
7892 mouse_event(MOUSEEVENTF_MOVE, -1, 0, 0, 0);
7893 mouse_event(MOUSEEVENTF_MOVE, 1, 0, 0, 0);
7895 track_query(TME_HOVER, hwnd, default_hover_time);
7897 pump_msg_loop_timeout(default_hover_time / 2, FALSE);
7898 ok_sequence(WmMouseHoverSeq, "WmMouseHoverSeq", FALSE);
7900 track_query(0, NULL, 0);
7902 track_hover(hwnd, HOVER_DEFAULT);
7903 track_query(TME_HOVER, hwnd, default_hover_time);
7905 pump_msg_loop_timeout(default_hover_time, TRUE);
7906 ok_sequence(WmMouseHoverSeq, "WmMouseHoverSeq", FALSE);
7908 track_query(0, NULL, 0);
7910 track_hover(hwnd, HOVER_DEFAULT);
7911 track_query(TME_HOVER, hwnd, default_hover_time);
7912 track_hover_cancel(hwnd);
7914 DestroyWindow(hwnd);
7918 #undef track_hover_cancel
7924 HMODULE user32 = GetModuleHandleA("user32.dll");
7925 FARPROC pSetWinEventHook = GetProcAddress(user32, "SetWinEventHook");
7926 FARPROC pUnhookWinEvent = GetProcAddress(user32, "UnhookWinEvent");
7927 FARPROC pIsWinEventHookInstalled = 0;/*GetProcAddress(user32, "IsWinEventHookInstalled");*/
7928 pGetAncestor = (void*) GetProcAddress(user32, "GetAncestor");
7930 if (!RegisterWindowClasses()) assert(0);
7932 if (pSetWinEventHook)
7934 hEvent_hook = (HWINEVENTHOOK)pSetWinEventHook(EVENT_MIN, EVENT_MAX,
7935 GetModuleHandleA(0),
7938 GetCurrentThreadId(),
7939 WINEVENT_INCONTEXT);
7940 assert(hEvent_hook);
7942 if (pIsWinEventHookInstalled)
7945 for (event = EVENT_MIN; event <= EVENT_MAX; event++)
7946 ok(pIsWinEventHookInstalled(event), "IsWinEventHookInstalled(%u) failed\n", event);
7950 cbt_hook_thread_id = GetCurrentThreadId();
7951 hCBT_hook = SetWindowsHookExA(WH_CBT, cbt_hook_proc, 0, GetCurrentThreadId());
7956 /* Fix message sequences before removing 4 lines below */
7958 if (pUnhookWinEvent && hEvent_hook)
7960 ret = pUnhookWinEvent(hEvent_hook);
7961 ok( ret, "UnhookWinEvent error %d\n", GetLastError());
7962 pUnhookWinEvent = 0;
7968 test_scrollwindowex();
7970 invisible_parent_tests();
7971 test_mdi_messages();
7972 test_button_messages();
7973 test_paint_messages();
7974 test_interthread_messages();
7975 test_message_conversion();
7976 test_accelerators();
7979 test_DestroyWindow();
7980 test_DispatchMessage();
7981 test_SendMessageTimeout();
7982 test_edit_messages();
7983 test_quit_message();
7984 test_TrackMouseEvent();
7986 UnhookWindowsHookEx(hCBT_hook);
7987 if (pUnhookWinEvent)
7989 ret = pUnhookWinEvent(hEvent_hook);
7990 ok( ret, "UnhookWinEvent error %d\n", GetLastError());
7991 SetLastError(0xdeadbeef);
7992 ok(!pUnhookWinEvent(hEvent_hook), "UnhookWinEvent succeeded\n");
7993 ok(GetLastError() == ERROR_INVALID_HANDLE || /* Win2k */
7994 GetLastError() == 0xdeadbeef, /* Win9x */
7995 "unexpected error %d\n", GetLastError());