dpnet/tests: Add a trailing '\n' to some ok() calls.
[wine.git] / dlls / user32 / win.c
blob73cad0c372ad0c46a18f8d2168f3f2162a5d1c85
1 /*
2 * Window related functions
4 * Copyright 1993, 1994 Alexandre Julliard
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "config.h"
22 #include "wine/port.h"
24 #include <assert.h>
25 #include <stdarg.h>
26 #include <stdlib.h>
27 #include <string.h>
28 #include "windef.h"
29 #include "winbase.h"
30 #include "winver.h"
31 #include "wine/server.h"
32 #include "wine/unicode.h"
33 #include "win.h"
34 #include "user_private.h"
35 #include "controls.h"
36 #include "winerror.h"
37 #include "wine/gdi_driver.h"
38 #include "wine/debug.h"
40 WINE_DEFAULT_DEBUG_CHANNEL(win);
42 #define NB_USER_HANDLES ((LAST_USER_HANDLE - FIRST_USER_HANDLE + 1) >> 1)
43 #define USER_HANDLE_TO_INDEX(hwnd) ((LOWORD(hwnd) - FIRST_USER_HANDLE) >> 1)
45 static DWORD process_layout = ~0u;
47 static struct list window_surfaces = LIST_INIT( window_surfaces );
49 static CRITICAL_SECTION surfaces_section;
50 static CRITICAL_SECTION_DEBUG critsect_debug =
52 0, 0, &surfaces_section,
53 { &critsect_debug.ProcessLocksList, &critsect_debug.ProcessLocksList },
54 0, 0, { (DWORD_PTR)(__FILE__ ": surfaces_section") }
56 static CRITICAL_SECTION surfaces_section = { &critsect_debug, -1, 0, 0, 0, 0 };
58 /**********************************************************************/
60 /* helper for Get/SetWindowLong */
61 static inline LONG_PTR get_win_data( const void *ptr, UINT size )
63 if (size == sizeof(WORD))
65 WORD ret;
66 memcpy( &ret, ptr, sizeof(ret) );
67 return ret;
69 else if (size == sizeof(DWORD))
71 DWORD ret;
72 memcpy( &ret, ptr, sizeof(ret) );
73 return ret;
75 else
77 LONG_PTR ret;
78 memcpy( &ret, ptr, sizeof(ret) );
79 return ret;
83 /* helper for Get/SetWindowLong */
84 static inline void set_win_data( void *ptr, LONG_PTR val, UINT size )
86 if (size == sizeof(WORD))
88 WORD newval = val;
89 memcpy( ptr, &newval, sizeof(newval) );
91 else if (size == sizeof(DWORD))
93 DWORD newval = val;
94 memcpy( ptr, &newval, sizeof(newval) );
96 else
98 memcpy( ptr, &val, sizeof(val) );
103 static void *user_handles[NB_USER_HANDLES];
105 /***********************************************************************
106 * alloc_user_handle
108 HANDLE alloc_user_handle( struct user_object *ptr, enum user_obj_type type )
110 HANDLE handle = 0;
112 SERVER_START_REQ( alloc_user_handle )
114 if (!wine_server_call_err( req )) handle = wine_server_ptr_handle( reply->handle );
116 SERVER_END_REQ;
118 if (handle)
120 UINT index = USER_HANDLE_TO_INDEX( handle );
122 assert( index < NB_USER_HANDLES );
123 ptr->handle = handle;
124 ptr->type = type;
125 InterlockedExchangePointer( &user_handles[index], ptr );
127 return handle;
131 /***********************************************************************
132 * get_user_handle_ptr
134 void *get_user_handle_ptr( HANDLE handle, enum user_obj_type type )
136 struct user_object *ptr;
137 WORD index = USER_HANDLE_TO_INDEX( handle );
139 if (index >= NB_USER_HANDLES) return NULL;
141 USER_Lock();
142 if ((ptr = user_handles[index]))
144 if (ptr->type == type &&
145 ((UINT)(UINT_PTR)ptr->handle == (UINT)(UINT_PTR)handle ||
146 !HIWORD(handle) || HIWORD(handle) == 0xffff))
147 return ptr;
148 ptr = NULL;
150 else ptr = OBJ_OTHER_PROCESS;
151 USER_Unlock();
152 return ptr;
156 /***********************************************************************
157 * release_user_handle_ptr
159 void release_user_handle_ptr( void *ptr )
161 assert( ptr && ptr != OBJ_OTHER_PROCESS );
162 USER_Unlock();
166 /***********************************************************************
167 * free_user_handle
169 void *free_user_handle( HANDLE handle, enum user_obj_type type )
171 struct user_object *ptr;
172 WORD index = USER_HANDLE_TO_INDEX( handle );
174 if ((ptr = get_user_handle_ptr( handle, type )) && ptr != OBJ_OTHER_PROCESS)
176 SERVER_START_REQ( free_user_handle )
178 req->handle = wine_server_user_handle( handle );
179 if (wine_server_call( req )) ptr = NULL;
180 else InterlockedCompareExchangePointer( &user_handles[index], NULL, ptr );
182 SERVER_END_REQ;
183 USER_Unlock();
185 return ptr;
189 /***********************************************************************
190 * create_window_handle
192 * Create a window handle with the server.
194 static WND *create_window_handle( HWND parent, HWND owner, LPCWSTR name,
195 HINSTANCE instance, BOOL unicode )
197 WORD index;
198 WND *win;
199 HWND handle = 0, full_parent = 0, full_owner = 0;
200 struct tagCLASS *class = NULL;
201 int extra_bytes = 0;
203 SERVER_START_REQ( create_window )
205 req->parent = wine_server_user_handle( parent );
206 req->owner = wine_server_user_handle( owner );
207 req->instance = wine_server_client_ptr( instance );
208 if (!(req->atom = get_int_atom_value( name )) && name)
209 wine_server_add_data( req, name, strlenW(name)*sizeof(WCHAR) );
210 if (!wine_server_call_err( req ))
212 handle = wine_server_ptr_handle( reply->handle );
213 full_parent = wine_server_ptr_handle( reply->parent );
214 full_owner = wine_server_ptr_handle( reply->owner );
215 extra_bytes = reply->extra;
216 class = wine_server_get_ptr( reply->class_ptr );
219 SERVER_END_REQ;
221 if (!handle)
223 WARN( "error %d creating window\n", GetLastError() );
224 return NULL;
227 if (!(win = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
228 sizeof(WND) + extra_bytes - sizeof(win->wExtra) )))
230 SERVER_START_REQ( destroy_window )
232 req->handle = wine_server_user_handle( handle );
233 wine_server_call( req );
235 SERVER_END_REQ;
236 SetLastError( ERROR_NOT_ENOUGH_MEMORY );
237 return NULL;
240 if (!parent) /* if parent is 0 we don't have a desktop window yet */
242 struct user_thread_info *thread_info = get_user_thread_info();
244 if (name == (LPCWSTR)DESKTOP_CLASS_ATOM)
246 if (!thread_info->top_window) thread_info->top_window = full_parent ? full_parent : handle;
247 else assert( full_parent == thread_info->top_window );
248 if (full_parent && !USER_Driver->pCreateDesktopWindow( thread_info->top_window ))
249 ERR( "failed to create desktop window\n" );
251 else /* HWND_MESSAGE parent */
253 if (!thread_info->msg_window && !full_parent) thread_info->msg_window = handle;
257 USER_Lock();
259 index = USER_HANDLE_TO_INDEX(handle);
260 assert( index < NB_USER_HANDLES );
261 win->obj.handle = handle;
262 win->obj.type = USER_WINDOW;
263 win->parent = full_parent;
264 win->owner = full_owner;
265 win->class = class;
266 win->winproc = get_class_winproc( class );
267 win->cbWndExtra = extra_bytes;
268 InterlockedExchangePointer( &user_handles[index], win );
269 if (WINPROC_IsUnicode( win->winproc, unicode )) win->flags |= WIN_ISUNICODE;
270 return win;
274 /***********************************************************************
275 * free_window_handle
277 * Free a window handle.
279 static void free_window_handle( HWND hwnd )
281 struct user_object *ptr;
282 WORD index = USER_HANDLE_TO_INDEX(hwnd);
284 if ((ptr = get_user_handle_ptr( hwnd, USER_WINDOW )) && ptr != OBJ_OTHER_PROCESS)
286 SERVER_START_REQ( destroy_window )
288 req->handle = wine_server_user_handle( hwnd );
289 if (wine_server_call_err( req )) ptr = NULL;
290 else InterlockedCompareExchangePointer( &user_handles[index], NULL, ptr );
292 SERVER_END_REQ;
293 USER_Unlock();
294 HeapFree( GetProcessHeap(), 0, ptr );
299 /*******************************************************************
300 * list_window_children
302 * Build an array of the children of a given window. The array must be
303 * freed with HeapFree. Returns NULL when no windows are found.
305 static HWND *list_window_children( HDESK desktop, HWND hwnd, LPCWSTR class, DWORD tid )
307 HWND *list;
308 int i, size = 128;
309 ATOM atom = get_int_atom_value( class );
311 /* empty class is not the same as NULL class */
312 if (!atom && class && !class[0]) return NULL;
314 for (;;)
316 int count = 0;
318 if (!(list = HeapAlloc( GetProcessHeap(), 0, size * sizeof(HWND) ))) break;
320 SERVER_START_REQ( get_window_children )
322 req->desktop = wine_server_obj_handle( desktop );
323 req->parent = wine_server_user_handle( hwnd );
324 req->tid = tid;
325 req->atom = atom;
326 if (!atom && class) wine_server_add_data( req, class, strlenW(class)*sizeof(WCHAR) );
327 wine_server_set_reply( req, list, (size-1) * sizeof(user_handle_t) );
328 if (!wine_server_call( req )) count = reply->count;
330 SERVER_END_REQ;
331 if (count && count < size)
333 /* start from the end since HWND is potentially larger than user_handle_t */
334 for (i = count - 1; i >= 0; i--)
335 list[i] = wine_server_ptr_handle( ((user_handle_t *)list)[i] );
336 list[count] = 0;
337 return list;
339 HeapFree( GetProcessHeap(), 0, list );
340 if (!count) break;
341 size = count + 1; /* restart with a large enough buffer */
343 return NULL;
347 /*******************************************************************
348 * list_window_parents
350 * Build an array of all parents of a given window, starting with
351 * the immediate parent. The array must be freed with HeapFree.
353 static HWND *list_window_parents( HWND hwnd )
355 WND *win;
356 HWND current, *list;
357 int i, pos = 0, size = 16, count = 0;
359 if (!(list = HeapAlloc( GetProcessHeap(), 0, size * sizeof(HWND) ))) return NULL;
361 current = hwnd;
362 for (;;)
364 if (!(win = WIN_GetPtr( current ))) goto empty;
365 if (win == WND_OTHER_PROCESS) break; /* need to do it the hard way */
366 if (win == WND_DESKTOP)
368 if (!pos) goto empty;
369 list[pos] = 0;
370 return list;
372 list[pos] = current = win->parent;
373 WIN_ReleasePtr( win );
374 if (!current) return list;
375 if (++pos == size - 1)
377 /* need to grow the list */
378 HWND *new_list = HeapReAlloc( GetProcessHeap(), 0, list, (size+16) * sizeof(HWND) );
379 if (!new_list) goto empty;
380 list = new_list;
381 size += 16;
385 /* at least one parent belongs to another process, have to query the server */
387 for (;;)
389 count = 0;
390 SERVER_START_REQ( get_window_parents )
392 req->handle = wine_server_user_handle( hwnd );
393 wine_server_set_reply( req, list, (size-1) * sizeof(user_handle_t) );
394 if (!wine_server_call( req )) count = reply->count;
396 SERVER_END_REQ;
397 if (!count) goto empty;
398 if (size > count)
400 /* start from the end since HWND is potentially larger than user_handle_t */
401 for (i = count - 1; i >= 0; i--)
402 list[i] = wine_server_ptr_handle( ((user_handle_t *)list)[i] );
403 list[count] = 0;
404 return list;
406 HeapFree( GetProcessHeap(), 0, list );
407 size = count + 1;
408 if (!(list = HeapAlloc( GetProcessHeap(), 0, size * sizeof(HWND) ))) return NULL;
411 empty:
412 HeapFree( GetProcessHeap(), 0, list );
413 return NULL;
417 /*******************************************************************
418 * send_parent_notify
420 static void send_parent_notify( HWND hwnd, UINT msg )
422 if ((GetWindowLongW( hwnd, GWL_STYLE ) & (WS_CHILD | WS_POPUP)) == WS_CHILD &&
423 !(GetWindowLongW( hwnd, GWL_EXSTYLE ) & WS_EX_NOPARENTNOTIFY))
425 HWND parent = GetParent(hwnd);
426 if (parent && parent != GetDesktopWindow())
427 SendMessageW( parent, WM_PARENTNOTIFY,
428 MAKEWPARAM( msg, GetWindowLongPtrW( hwnd, GWLP_ID )), (LPARAM)hwnd );
433 /*******************************************************************
434 * update_window_state
436 * Trigger an update of the window's driver state and surface.
438 static void update_window_state( HWND hwnd )
440 RECT window_rect, client_rect;
442 WIN_GetRectangles( hwnd, COORDS_PARENT, &window_rect, &client_rect );
443 set_window_pos( hwnd, 0, SWP_NOSIZE | SWP_NOMOVE | SWP_NOCLIENTSIZE | SWP_NOCLIENTMOVE |
444 SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOREDRAW, &window_rect, &client_rect, NULL );
448 /*******************************************************************
449 * get_server_window_text
451 * Retrieve the window text from the server.
453 static void get_server_window_text( HWND hwnd, LPWSTR text, INT count )
455 size_t len = 0;
457 SERVER_START_REQ( get_window_text )
459 req->handle = wine_server_user_handle( hwnd );
460 wine_server_set_reply( req, text, (count - 1) * sizeof(WCHAR) );
461 if (!wine_server_call_err( req )) len = wine_server_reply_size(reply);
463 SERVER_END_REQ;
464 text[len / sizeof(WCHAR)] = 0;
468 /*******************************************************************
469 * get_hwnd_message_parent
471 * Return the parent for HWND_MESSAGE windows.
473 HWND get_hwnd_message_parent(void)
475 struct user_thread_info *thread_info = get_user_thread_info();
477 if (!thread_info->msg_window) GetDesktopWindow(); /* trigger creation */
478 return thread_info->msg_window;
482 /*******************************************************************
483 * is_desktop_window
485 * Check if window is the desktop or the HWND_MESSAGE top parent.
487 BOOL is_desktop_window( HWND hwnd )
489 struct user_thread_info *thread_info = get_user_thread_info();
491 if (!hwnd) return FALSE;
492 if (hwnd == thread_info->top_window) return TRUE;
493 if (hwnd == thread_info->msg_window) return TRUE;
495 if (!HIWORD(hwnd) || HIWORD(hwnd) == 0xffff)
497 if (LOWORD(thread_info->top_window) == LOWORD(hwnd)) return TRUE;
498 if (LOWORD(thread_info->msg_window) == LOWORD(hwnd)) return TRUE;
500 return FALSE;
504 /*******************************************************************
505 * Dummy window surface for windows that shouldn't get painted.
508 static void dummy_surface_lock( struct window_surface *window_surface )
510 /* nothing to do */
513 static void dummy_surface_unlock( struct window_surface *window_surface )
515 /* nothing to do */
518 static void *dummy_surface_get_bitmap_info( struct window_surface *window_surface, BITMAPINFO *info )
520 static DWORD dummy_data;
522 info->bmiHeader.biSize = sizeof( info->bmiHeader );
523 info->bmiHeader.biWidth = dummy_surface.rect.right;
524 info->bmiHeader.biHeight = dummy_surface.rect.bottom;
525 info->bmiHeader.biPlanes = 1;
526 info->bmiHeader.biBitCount = 32;
527 info->bmiHeader.biCompression = BI_RGB;
528 info->bmiHeader.biSizeImage = 0;
529 info->bmiHeader.biXPelsPerMeter = 0;
530 info->bmiHeader.biYPelsPerMeter = 0;
531 info->bmiHeader.biClrUsed = 0;
532 info->bmiHeader.biClrImportant = 0;
533 return &dummy_data;
536 static RECT *dummy_surface_get_bounds( struct window_surface *window_surface )
538 static RECT dummy_bounds;
539 return &dummy_bounds;
542 static void dummy_surface_set_region( struct window_surface *window_surface, HRGN region )
544 /* nothing to do */
547 static void dummy_surface_flush( struct window_surface *window_surface )
549 /* nothing to do */
552 static void dummy_surface_destroy( struct window_surface *window_surface )
554 /* nothing to do */
557 static const struct window_surface_funcs dummy_surface_funcs =
559 dummy_surface_lock,
560 dummy_surface_unlock,
561 dummy_surface_get_bitmap_info,
562 dummy_surface_get_bounds,
563 dummy_surface_set_region,
564 dummy_surface_flush,
565 dummy_surface_destroy
568 struct window_surface dummy_surface = { &dummy_surface_funcs, { NULL, NULL }, 1, { 0, 0, 1, 1 } };
571 /*******************************************************************
572 * register_window_surface
574 * Register a window surface in the global list, possibly replacing another one.
576 void register_window_surface( struct window_surface *old, struct window_surface *new )
578 if (old == new) return;
579 EnterCriticalSection( &surfaces_section );
580 if (old && old != &dummy_surface) list_remove( &old->entry );
581 if (new && new != &dummy_surface) list_add_tail( &window_surfaces, &new->entry );
582 LeaveCriticalSection( &surfaces_section );
586 /*******************************************************************
587 * flush_window_surfaces
589 * Flush pending output from all window surfaces.
591 void flush_window_surfaces( BOOL idle )
593 static DWORD last_idle;
594 DWORD now;
595 struct window_surface *surface;
597 EnterCriticalSection( &surfaces_section );
598 now = GetTickCount();
599 if (idle) last_idle = now;
600 /* if not idle, we only flush if there's evidence that the app never goes idle */
601 else if ((int)(now - last_idle) < 1000) goto done;
603 LIST_FOR_EACH_ENTRY( surface, &window_surfaces, struct window_surface, entry )
604 surface->funcs->flush( surface );
605 done:
606 LeaveCriticalSection( &surfaces_section );
610 /***********************************************************************
611 * WIN_GetPtr
613 * Return a pointer to the WND structure if local to the process,
614 * or WND_OTHER_PROCESS if handle may be valid in other process.
615 * If ret value is a valid pointer, it must be released with WIN_ReleasePtr.
617 WND *WIN_GetPtr( HWND hwnd )
619 WND *ptr;
621 if ((ptr = get_user_handle_ptr( hwnd, USER_WINDOW )) == WND_OTHER_PROCESS)
623 if (is_desktop_window( hwnd )) ptr = WND_DESKTOP;
625 return ptr;
629 /***********************************************************************
630 * WIN_IsCurrentProcess
632 * Check whether a given window belongs to the current process (and return the full handle).
634 HWND WIN_IsCurrentProcess( HWND hwnd )
636 WND *ptr;
637 HWND ret;
639 if (!(ptr = WIN_GetPtr( hwnd )) || ptr == WND_OTHER_PROCESS || ptr == WND_DESKTOP) return 0;
640 ret = ptr->obj.handle;
641 WIN_ReleasePtr( ptr );
642 return ret;
646 /***********************************************************************
647 * WIN_IsCurrentThread
649 * Check whether a given window belongs to the current thread (and return the full handle).
651 HWND WIN_IsCurrentThread( HWND hwnd )
653 WND *ptr;
654 HWND ret = 0;
656 if (!(ptr = WIN_GetPtr( hwnd )) || ptr == WND_OTHER_PROCESS || ptr == WND_DESKTOP) return 0;
657 if (ptr->tid == GetCurrentThreadId()) ret = ptr->obj.handle;
658 WIN_ReleasePtr( ptr );
659 return ret;
663 /***********************************************************************
664 * WIN_GetFullHandle
666 * Convert a possibly truncated window handle to a full 32-bit handle.
668 HWND WIN_GetFullHandle( HWND hwnd )
670 WND *ptr;
672 if (!hwnd || (ULONG_PTR)hwnd >> 16) return hwnd;
673 if (LOWORD(hwnd) <= 1 || LOWORD(hwnd) == 0xffff) return hwnd;
674 /* do sign extension for -2 and -3 */
675 if (LOWORD(hwnd) >= (WORD)-3) return (HWND)(LONG_PTR)(INT16)LOWORD(hwnd);
677 if (!(ptr = WIN_GetPtr( hwnd ))) return hwnd;
679 if (ptr == WND_DESKTOP)
681 if (LOWORD(hwnd) == LOWORD(GetDesktopWindow())) return GetDesktopWindow();
682 else return get_hwnd_message_parent();
685 if (ptr != WND_OTHER_PROCESS)
687 hwnd = ptr->obj.handle;
688 WIN_ReleasePtr( ptr );
690 else /* may belong to another process */
692 SERVER_START_REQ( get_window_info )
694 req->handle = wine_server_user_handle( hwnd );
695 if (!wine_server_call_err( req )) hwnd = wine_server_ptr_handle( reply->full_handle );
697 SERVER_END_REQ;
699 return hwnd;
703 /***********************************************************************
704 * WIN_SetOwner
706 * Change the owner of a window.
708 HWND WIN_SetOwner( HWND hwnd, HWND owner )
710 WND *win = WIN_GetPtr( hwnd );
711 HWND ret = 0;
713 if (!win || win == WND_DESKTOP) return 0;
714 if (win == WND_OTHER_PROCESS)
716 if (IsWindow(hwnd)) ERR( "cannot set owner %p on other process window %p\n", owner, hwnd );
717 return 0;
719 SERVER_START_REQ( set_window_owner )
721 req->handle = wine_server_user_handle( hwnd );
722 req->owner = wine_server_user_handle( owner );
723 if (!wine_server_call( req ))
725 win->owner = wine_server_ptr_handle( reply->full_owner );
726 ret = wine_server_ptr_handle( reply->prev_owner );
729 SERVER_END_REQ;
730 WIN_ReleasePtr( win );
731 return ret;
735 /***********************************************************************
736 * WIN_SetStyle
738 * Change the style of a window.
740 ULONG WIN_SetStyle( HWND hwnd, ULONG set_bits, ULONG clear_bits )
742 BOOL ok, made_visible = FALSE;
743 STYLESTRUCT style;
744 WND *win = WIN_GetPtr( hwnd );
746 if (!win || win == WND_DESKTOP) return 0;
747 if (win == WND_OTHER_PROCESS)
749 if (IsWindow(hwnd))
750 ERR( "cannot set style %x/%x on other process window %p\n",
751 set_bits, clear_bits, hwnd );
752 return 0;
754 style.styleOld = win->dwStyle;
755 style.styleNew = (win->dwStyle | set_bits) & ~clear_bits;
756 if (style.styleNew == style.styleOld)
758 WIN_ReleasePtr( win );
759 return style.styleNew;
761 SERVER_START_REQ( set_window_info )
763 req->handle = wine_server_user_handle( hwnd );
764 req->flags = SET_WIN_STYLE;
765 req->style = style.styleNew;
766 req->extra_offset = -1;
767 if ((ok = !wine_server_call( req )))
769 style.styleOld = reply->old_style;
770 win->dwStyle = style.styleNew;
773 SERVER_END_REQ;
775 if (ok && ((style.styleOld ^ style.styleNew) & WS_VISIBLE))
777 /* Some apps try to make their window visible through WM_SETREDRAW.
778 * Only do that if the window was never explicitly hidden,
779 * because Steam messes with WM_SETREDRAW after hiding its windows. */
780 made_visible = !(win->flags & WIN_HIDDEN) && (style.styleNew & WS_VISIBLE);
781 invalidate_dce( win, NULL );
783 WIN_ReleasePtr( win );
785 if (!ok) return 0;
787 USER_Driver->pSetWindowStyle( hwnd, GWL_STYLE, &style );
788 if (made_visible) update_window_state( hwnd );
790 return style.styleOld;
794 /***********************************************************************
795 * WIN_GetRectangles
797 * Get the window and client rectangles.
799 BOOL WIN_GetRectangles( HWND hwnd, enum coords_relative relative, RECT *rectWindow, RECT *rectClient )
801 WND *win = WIN_GetPtr( hwnd );
802 BOOL ret = TRUE;
804 if (!win)
806 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
807 return FALSE;
809 if (win == WND_DESKTOP)
811 RECT rect;
812 rect.left = rect.top = 0;
813 if (hwnd == get_hwnd_message_parent())
815 rect.right = 100;
816 rect.bottom = 100;
818 else
820 rect.right = GetSystemMetrics(SM_CXSCREEN);
821 rect.bottom = GetSystemMetrics(SM_CYSCREEN);
823 if (rectWindow) *rectWindow = rect;
824 if (rectClient) *rectClient = rect;
825 return TRUE;
827 if (win != WND_OTHER_PROCESS)
829 RECT window_rect = win->rectWindow, client_rect = win->rectClient;
831 switch (relative)
833 case COORDS_CLIENT:
834 OffsetRect( &window_rect, -win->rectClient.left, -win->rectClient.top );
835 OffsetRect( &client_rect, -win->rectClient.left, -win->rectClient.top );
836 if (win->dwExStyle & WS_EX_LAYOUTRTL)
837 mirror_rect( &win->rectClient, &window_rect );
838 break;
839 case COORDS_WINDOW:
840 OffsetRect( &window_rect, -win->rectWindow.left, -win->rectWindow.top );
841 OffsetRect( &client_rect, -win->rectWindow.left, -win->rectWindow.top );
842 if (win->dwExStyle & WS_EX_LAYOUTRTL)
843 mirror_rect( &win->rectWindow, &client_rect );
844 break;
845 case COORDS_PARENT:
846 if (win->parent)
848 WND *parent = WIN_GetPtr( win->parent );
849 if (parent == WND_DESKTOP) break;
850 if (!parent || parent == WND_OTHER_PROCESS)
852 WIN_ReleasePtr( win );
853 goto other_process;
855 if (parent->flags & WIN_CHILDREN_MOVED)
857 WIN_ReleasePtr( parent );
858 WIN_ReleasePtr( win );
859 goto other_process;
861 if (parent->dwExStyle & WS_EX_LAYOUTRTL)
863 mirror_rect( &parent->rectClient, &window_rect );
864 mirror_rect( &parent->rectClient, &client_rect );
866 WIN_ReleasePtr( parent );
868 break;
869 case COORDS_SCREEN:
870 while (win->parent)
872 WND *parent = WIN_GetPtr( win->parent );
873 if (parent == WND_DESKTOP) break;
874 if (!parent || parent == WND_OTHER_PROCESS)
876 WIN_ReleasePtr( win );
877 goto other_process;
879 WIN_ReleasePtr( win );
880 if (parent->flags & WIN_CHILDREN_MOVED)
882 WIN_ReleasePtr( parent );
883 goto other_process;
885 win = parent;
886 if (win->parent)
888 OffsetRect( &window_rect, win->rectClient.left, win->rectClient.top );
889 OffsetRect( &client_rect, win->rectClient.left, win->rectClient.top );
892 break;
894 if (rectWindow) *rectWindow = window_rect;
895 if (rectClient) *rectClient = client_rect;
896 WIN_ReleasePtr( win );
897 return TRUE;
900 other_process:
901 SERVER_START_REQ( get_window_rectangles )
903 req->handle = wine_server_user_handle( hwnd );
904 req->relative = relative;
905 if ((ret = !wine_server_call_err( req )))
907 if (rectWindow)
909 rectWindow->left = reply->window.left;
910 rectWindow->top = reply->window.top;
911 rectWindow->right = reply->window.right;
912 rectWindow->bottom = reply->window.bottom;
914 if (rectClient)
916 rectClient->left = reply->client.left;
917 rectClient->top = reply->client.top;
918 rectClient->right = reply->client.right;
919 rectClient->bottom = reply->client.bottom;
923 SERVER_END_REQ;
924 return ret;
928 /***********************************************************************
929 * WIN_DestroyWindow
931 * Destroy storage associated to a window. "Internals" p.358
933 LRESULT WIN_DestroyWindow( HWND hwnd )
935 WND *wndPtr;
936 HWND *list;
937 HMENU menu = 0, sys_menu;
938 HWND icon_title;
939 struct window_surface *surface;
941 TRACE("%p\n", hwnd );
943 /* free child windows */
944 if ((list = WIN_ListChildren( hwnd )))
946 int i;
947 for (i = 0; list[i]; i++)
949 if (WIN_IsCurrentThread( list[i] )) WIN_DestroyWindow( list[i] );
950 else SendMessageW( list[i], WM_WINE_DESTROYWINDOW, 0, 0 );
952 HeapFree( GetProcessHeap(), 0, list );
955 /* Unlink now so we won't bother with the children later on */
956 SERVER_START_REQ( set_parent )
958 req->handle = wine_server_user_handle( hwnd );
959 req->parent = 0;
960 wine_server_call( req );
962 SERVER_END_REQ;
965 * Send the WM_NCDESTROY to the window being destroyed.
967 SendMessageW( hwnd, WM_NCDESTROY, 0, 0 );
969 /* FIXME: do we need to fake QS_MOUSEMOVE wakebit? */
971 /* free resources associated with the window */
973 if (!(wndPtr = WIN_GetPtr( hwnd )) || wndPtr == WND_OTHER_PROCESS) return 0;
974 if ((wndPtr->dwStyle & (WS_CHILD | WS_POPUP)) != WS_CHILD)
975 menu = (HMENU)wndPtr->wIDmenu;
976 sys_menu = wndPtr->hSysMenu;
977 free_dce( wndPtr->dce, hwnd );
978 wndPtr->dce = NULL;
979 icon_title = wndPtr->icon_title;
980 HeapFree( GetProcessHeap(), 0, wndPtr->text );
981 wndPtr->text = NULL;
982 HeapFree( GetProcessHeap(), 0, wndPtr->pScroll );
983 wndPtr->pScroll = NULL;
984 surface = wndPtr->surface;
985 wndPtr->surface = NULL;
986 WIN_ReleasePtr( wndPtr );
988 if (icon_title) DestroyWindow( icon_title );
989 if (menu) DestroyMenu( menu );
990 if (sys_menu) DestroyMenu( sys_menu );
991 if (surface)
993 register_window_surface( surface, NULL );
994 window_surface_release( surface );
997 USER_Driver->pDestroyWindow( hwnd );
999 free_window_handle( hwnd );
1000 return 0;
1004 /***********************************************************************
1005 * destroy_thread_window
1007 * Destroy a window upon exit of its thread.
1009 static void destroy_thread_window( HWND hwnd )
1011 WND *wndPtr;
1012 HWND *list;
1013 HMENU menu = 0, sys_menu = 0;
1014 struct window_surface *surface = NULL;
1015 WORD index;
1017 /* free child windows */
1019 if ((list = WIN_ListChildren( hwnd )))
1021 int i;
1022 for (i = 0; list[i]; i++)
1024 if (WIN_IsCurrentThread( list[i] )) destroy_thread_window( list[i] );
1025 else SendMessageW( list[i], WM_WINE_DESTROYWINDOW, 0, 0 );
1027 HeapFree( GetProcessHeap(), 0, list );
1030 /* destroy the client-side storage */
1032 index = USER_HANDLE_TO_INDEX(hwnd);
1033 if (index >= NB_USER_HANDLES) return;
1034 USER_Lock();
1035 if ((wndPtr = user_handles[index]))
1037 if ((wndPtr->dwStyle & (WS_CHILD | WS_POPUP)) != WS_CHILD) menu = (HMENU)wndPtr->wIDmenu;
1038 sys_menu = wndPtr->hSysMenu;
1039 free_dce( wndPtr->dce, hwnd );
1040 surface = wndPtr->surface;
1041 wndPtr->surface = NULL;
1042 InterlockedCompareExchangePointer( &user_handles[index], NULL, wndPtr );
1044 USER_Unlock();
1046 HeapFree( GetProcessHeap(), 0, wndPtr );
1047 if (menu) DestroyMenu( menu );
1048 if (sys_menu) DestroyMenu( sys_menu );
1049 if (surface)
1051 register_window_surface( surface, NULL );
1052 window_surface_release( surface );
1057 /***********************************************************************
1058 * destroy_thread_child_windows
1060 * Destroy child windows upon exit of its thread.
1062 static void destroy_thread_child_windows( HWND hwnd )
1064 HWND *list;
1065 int i;
1067 if (WIN_IsCurrentThread( hwnd ))
1069 destroy_thread_window( hwnd );
1071 else if ((list = WIN_ListChildren( hwnd )))
1073 for (i = 0; list[i]; i++) destroy_thread_child_windows( list[i] );
1074 HeapFree( GetProcessHeap(), 0, list );
1079 /***********************************************************************
1080 * WIN_DestroyThreadWindows
1082 * Destroy all children of 'wnd' owned by the current thread.
1084 void WIN_DestroyThreadWindows( HWND hwnd )
1086 HWND *list;
1087 int i;
1089 if (!(list = WIN_ListChildren( hwnd ))) return;
1091 /* reset owners of top-level windows */
1092 for (i = 0; list[i]; i++)
1094 if (!WIN_IsCurrentThread( list[i] ))
1096 HWND owner = GetWindow( list[i], GW_OWNER );
1097 if (owner && WIN_IsCurrentThread( owner )) WIN_SetOwner( list[i], 0 );
1101 for (i = 0; list[i]; i++) destroy_thread_child_windows( list[i] );
1102 HeapFree( GetProcessHeap(), 0, list );
1106 /***********************************************************************
1107 * WIN_FixCoordinates
1109 * Fix the coordinates - Helper for WIN_CreateWindowEx.
1110 * returns default show mode in sw.
1112 static void WIN_FixCoordinates( CREATESTRUCTW *cs, INT *sw)
1114 #define IS_DEFAULT(x) ((x) == CW_USEDEFAULT || (x) == (SHORT)0x8000)
1115 POINT pos[2];
1117 if (cs->dwExStyle & WS_EX_MDICHILD)
1119 UINT id = 0;
1121 MDI_CalcDefaultChildPos(cs->hwndParent, -1, pos, 0, &id);
1122 if (!(cs->style & WS_POPUP)) cs->hMenu = ULongToHandle(id);
1124 TRACE("MDI child id %04x\n", id);
1127 if (cs->style & (WS_CHILD | WS_POPUP))
1129 if (cs->dwExStyle & WS_EX_MDICHILD)
1131 if (IS_DEFAULT(cs->x))
1133 cs->x = pos[0].x;
1134 cs->y = pos[0].y;
1136 if (IS_DEFAULT(cs->cx) || !cs->cx) cs->cx = pos[1].x;
1137 if (IS_DEFAULT(cs->cy) || !cs->cy) cs->cy = pos[1].y;
1139 else
1141 if (IS_DEFAULT(cs->x)) cs->x = cs->y = 0;
1142 if (IS_DEFAULT(cs->cx)) cs->cx = cs->cy = 0;
1145 else /* overlapped window */
1147 HMONITOR monitor;
1148 MONITORINFO mon_info;
1149 STARTUPINFOW info;
1151 if (!IS_DEFAULT(cs->x) && !IS_DEFAULT(cs->cx) && !IS_DEFAULT(cs->cy)) return;
1153 monitor = MonitorFromWindow( cs->hwndParent, MONITOR_DEFAULTTOPRIMARY );
1154 mon_info.cbSize = sizeof(mon_info);
1155 GetMonitorInfoW( monitor, &mon_info );
1156 GetStartupInfoW( &info );
1158 if (IS_DEFAULT(cs->x))
1160 if (!IS_DEFAULT(cs->y)) *sw = cs->y;
1161 cs->x = (info.dwFlags & STARTF_USEPOSITION) ? info.dwX : mon_info.rcWork.left;
1162 cs->y = (info.dwFlags & STARTF_USEPOSITION) ? info.dwY : mon_info.rcWork.top;
1165 if (IS_DEFAULT(cs->cx))
1167 if (info.dwFlags & STARTF_USESIZE)
1169 cs->cx = info.dwXSize;
1170 cs->cy = info.dwYSize;
1172 else
1174 cs->cx = (mon_info.rcWork.right - mon_info.rcWork.left) * 3 / 4 - cs->x;
1175 cs->cy = (mon_info.rcWork.bottom - mon_info.rcWork.top) * 3 / 4 - cs->y;
1178 /* neither x nor cx are default. Check the y values .
1179 * In the trace we see Outlook and Outlook Express using
1180 * cy set to CW_USEDEFAULT when opening the address book.
1182 else if (IS_DEFAULT(cs->cy))
1184 FIXME("Strange use of CW_USEDEFAULT in nHeight\n");
1185 cs->cy = (mon_info.rcWork.bottom - mon_info.rcWork.top) * 3 / 4 - cs->y;
1188 #undef IS_DEFAULT
1191 /***********************************************************************
1192 * dump_window_styles
1194 static void dump_window_styles( DWORD style, DWORD exstyle )
1196 TRACE( "style:" );
1197 if(style & WS_POPUP) TRACE(" WS_POPUP");
1198 if(style & WS_CHILD) TRACE(" WS_CHILD");
1199 if(style & WS_MINIMIZE) TRACE(" WS_MINIMIZE");
1200 if(style & WS_VISIBLE) TRACE(" WS_VISIBLE");
1201 if(style & WS_DISABLED) TRACE(" WS_DISABLED");
1202 if(style & WS_CLIPSIBLINGS) TRACE(" WS_CLIPSIBLINGS");
1203 if(style & WS_CLIPCHILDREN) TRACE(" WS_CLIPCHILDREN");
1204 if(style & WS_MAXIMIZE) TRACE(" WS_MAXIMIZE");
1205 if((style & WS_CAPTION) == WS_CAPTION) TRACE(" WS_CAPTION");
1206 else
1208 if(style & WS_BORDER) TRACE(" WS_BORDER");
1209 if(style & WS_DLGFRAME) TRACE(" WS_DLGFRAME");
1211 if(style & WS_VSCROLL) TRACE(" WS_VSCROLL");
1212 if(style & WS_HSCROLL) TRACE(" WS_HSCROLL");
1213 if(style & WS_SYSMENU) TRACE(" WS_SYSMENU");
1214 if(style & WS_THICKFRAME) TRACE(" WS_THICKFRAME");
1215 if (style & WS_CHILD)
1217 if(style & WS_GROUP) TRACE(" WS_GROUP");
1218 if(style & WS_TABSTOP) TRACE(" WS_TABSTOP");
1220 else
1222 if(style & WS_MINIMIZEBOX) TRACE(" WS_MINIMIZEBOX");
1223 if(style & WS_MAXIMIZEBOX) TRACE(" WS_MAXIMIZEBOX");
1226 /* FIXME: Add dumping of BS_/ES_/SBS_/LBS_/CBS_/DS_/etc. styles */
1227 #define DUMPED_STYLES \
1228 ((DWORD)(WS_POPUP | \
1229 WS_CHILD | \
1230 WS_MINIMIZE | \
1231 WS_VISIBLE | \
1232 WS_DISABLED | \
1233 WS_CLIPSIBLINGS | \
1234 WS_CLIPCHILDREN | \
1235 WS_MAXIMIZE | \
1236 WS_BORDER | \
1237 WS_DLGFRAME | \
1238 WS_VSCROLL | \
1239 WS_HSCROLL | \
1240 WS_SYSMENU | \
1241 WS_THICKFRAME | \
1242 WS_GROUP | \
1243 WS_TABSTOP | \
1244 WS_MINIMIZEBOX | \
1245 WS_MAXIMIZEBOX))
1247 if(style & ~DUMPED_STYLES) TRACE(" %08x", style & ~DUMPED_STYLES);
1248 TRACE("\n");
1249 #undef DUMPED_STYLES
1251 TRACE( "exstyle:" );
1252 if(exstyle & WS_EX_DLGMODALFRAME) TRACE(" WS_EX_DLGMODALFRAME");
1253 if(exstyle & WS_EX_DRAGDETECT) TRACE(" WS_EX_DRAGDETECT");
1254 if(exstyle & WS_EX_NOPARENTNOTIFY) TRACE(" WS_EX_NOPARENTNOTIFY");
1255 if(exstyle & WS_EX_TOPMOST) TRACE(" WS_EX_TOPMOST");
1256 if(exstyle & WS_EX_ACCEPTFILES) TRACE(" WS_EX_ACCEPTFILES");
1257 if(exstyle & WS_EX_TRANSPARENT) TRACE(" WS_EX_TRANSPARENT");
1258 if(exstyle & WS_EX_MDICHILD) TRACE(" WS_EX_MDICHILD");
1259 if(exstyle & WS_EX_TOOLWINDOW) TRACE(" WS_EX_TOOLWINDOW");
1260 if(exstyle & WS_EX_WINDOWEDGE) TRACE(" WS_EX_WINDOWEDGE");
1261 if(exstyle & WS_EX_CLIENTEDGE) TRACE(" WS_EX_CLIENTEDGE");
1262 if(exstyle & WS_EX_CONTEXTHELP) TRACE(" WS_EX_CONTEXTHELP");
1263 if(exstyle & WS_EX_RIGHT) TRACE(" WS_EX_RIGHT");
1264 if(exstyle & WS_EX_RTLREADING) TRACE(" WS_EX_RTLREADING");
1265 if(exstyle & WS_EX_LEFTSCROLLBAR) TRACE(" WS_EX_LEFTSCROLLBAR");
1266 if(exstyle & WS_EX_CONTROLPARENT) TRACE(" WS_EX_CONTROLPARENT");
1267 if(exstyle & WS_EX_STATICEDGE) TRACE(" WS_EX_STATICEDGE");
1268 if(exstyle & WS_EX_APPWINDOW) TRACE(" WS_EX_APPWINDOW");
1269 if(exstyle & WS_EX_LAYERED) TRACE(" WS_EX_LAYERED");
1270 if(exstyle & WS_EX_LAYOUTRTL) TRACE(" WS_EX_LAYOUTRTL");
1272 #define DUMPED_EX_STYLES \
1273 ((DWORD)(WS_EX_DLGMODALFRAME | \
1274 WS_EX_DRAGDETECT | \
1275 WS_EX_NOPARENTNOTIFY | \
1276 WS_EX_TOPMOST | \
1277 WS_EX_ACCEPTFILES | \
1278 WS_EX_TRANSPARENT | \
1279 WS_EX_MDICHILD | \
1280 WS_EX_TOOLWINDOW | \
1281 WS_EX_WINDOWEDGE | \
1282 WS_EX_CLIENTEDGE | \
1283 WS_EX_CONTEXTHELP | \
1284 WS_EX_RIGHT | \
1285 WS_EX_RTLREADING | \
1286 WS_EX_LEFTSCROLLBAR | \
1287 WS_EX_CONTROLPARENT | \
1288 WS_EX_STATICEDGE | \
1289 WS_EX_APPWINDOW | \
1290 WS_EX_LAYERED | \
1291 WS_EX_LAYOUTRTL))
1293 if(exstyle & ~DUMPED_EX_STYLES) TRACE(" %08x", exstyle & ~DUMPED_EX_STYLES);
1294 TRACE("\n");
1295 #undef DUMPED_EX_STYLES
1299 /***********************************************************************
1300 * WIN_CreateWindowEx
1302 * Implementation of CreateWindowEx().
1304 HWND WIN_CreateWindowEx( CREATESTRUCTW *cs, LPCWSTR className, HINSTANCE module, BOOL unicode )
1306 INT cx, cy, style, sw = SW_SHOW;
1307 LRESULT result;
1308 RECT rect;
1309 WND *wndPtr;
1310 HWND hwnd, parent, owner, top_child = 0;
1311 MDICREATESTRUCTW mdi_cs;
1312 CBT_CREATEWNDW cbtc;
1313 CREATESTRUCTW cbcs;
1315 TRACE("%s %s ex=%08x style=%08x %d,%d %dx%d parent=%p menu=%p inst=%p params=%p\n",
1316 unicode ? debugstr_w(cs->lpszName) : debugstr_a((LPCSTR)cs->lpszName),
1317 debugstr_w(className),
1318 cs->dwExStyle, cs->style, cs->x, cs->y, cs->cx, cs->cy,
1319 cs->hwndParent, cs->hMenu, cs->hInstance, cs->lpCreateParams );
1320 if(TRACE_ON(win)) dump_window_styles( cs->style, cs->dwExStyle );
1322 /* Fix the styles for MDI children */
1323 if (cs->dwExStyle & WS_EX_MDICHILD)
1325 UINT flags = 0;
1327 wndPtr = WIN_GetPtr(cs->hwndParent);
1328 if (wndPtr && wndPtr != WND_OTHER_PROCESS && wndPtr != WND_DESKTOP)
1330 flags = wndPtr->flags;
1331 WIN_ReleasePtr(wndPtr);
1334 if (!(flags & WIN_ISMDICLIENT))
1336 WARN("WS_EX_MDICHILD, but parent %p is not MDIClient\n", cs->hwndParent);
1337 return 0;
1340 /* cs->lpCreateParams of WM_[NC]CREATE is different for MDI children.
1341 * MDICREATESTRUCT members have the originally passed values.
1343 * Note: we rely on the fact that MDICREATESTRUCTA and MDICREATESTRUCTW
1344 * have the same layout.
1346 mdi_cs.szClass = cs->lpszClass;
1347 mdi_cs.szTitle = cs->lpszName;
1348 mdi_cs.hOwner = cs->hInstance;
1349 mdi_cs.x = cs->x;
1350 mdi_cs.y = cs->y;
1351 mdi_cs.cx = cs->cx;
1352 mdi_cs.cy = cs->cy;
1353 mdi_cs.style = cs->style;
1354 mdi_cs.lParam = (LPARAM)cs->lpCreateParams;
1356 cs->lpCreateParams = &mdi_cs;
1358 if (GetWindowLongW(cs->hwndParent, GWL_STYLE) & MDIS_ALLCHILDSTYLES)
1360 if (cs->style & WS_POPUP)
1362 TRACE("WS_POPUP with MDIS_ALLCHILDSTYLES is not allowed\n");
1363 return 0;
1365 cs->style |= WS_CHILD | WS_CLIPSIBLINGS;
1367 else
1369 cs->style &= ~WS_POPUP;
1370 cs->style |= WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CAPTION |
1371 WS_SYSMENU | WS_THICKFRAME | WS_MINIMIZEBOX | WS_MAXIMIZEBOX;
1374 top_child = GetWindow(cs->hwndParent, GW_CHILD);
1376 if (top_child)
1378 /* Restore current maximized child */
1379 if((cs->style & WS_VISIBLE) && IsZoomed(top_child))
1381 TRACE("Restoring current maximized child %p\n", top_child);
1382 if (cs->style & WS_MAXIMIZE)
1384 /* if the new window is maximized don't bother repainting */
1385 SendMessageW( top_child, WM_SETREDRAW, FALSE, 0 );
1386 ShowWindow( top_child, SW_SHOWNORMAL );
1387 SendMessageW( top_child, WM_SETREDRAW, TRUE, 0 );
1389 else ShowWindow( top_child, SW_SHOWNORMAL );
1394 /* Find the parent window */
1396 parent = cs->hwndParent;
1397 owner = 0;
1399 if (cs->hwndParent == HWND_MESSAGE)
1401 cs->hwndParent = parent = get_hwnd_message_parent();
1403 else if (cs->hwndParent)
1405 if ((cs->style & (WS_CHILD|WS_POPUP)) != WS_CHILD)
1407 parent = GetDesktopWindow();
1408 owner = cs->hwndParent;
1410 else
1412 DWORD parent_style = GetWindowLongW( parent, GWL_EXSTYLE );
1413 if ((parent_style & WS_EX_LAYOUTRTL) && !(parent_style & WS_EX_NOINHERITLAYOUT))
1414 cs->dwExStyle |= WS_EX_LAYOUTRTL;
1417 else
1419 static const WCHAR messageW[] = {'M','e','s','s','a','g','e',0};
1421 if ((cs->style & (WS_CHILD|WS_POPUP)) == WS_CHILD)
1423 WARN("No parent for child window\n" );
1424 SetLastError(ERROR_TLW_WITH_WSCHILD);
1425 return 0; /* WS_CHILD needs a parent, but WS_POPUP doesn't */
1428 /* are we creating the desktop or HWND_MESSAGE parent itself? */
1429 if (className != (LPCWSTR)DESKTOP_CLASS_ATOM &&
1430 (IS_INTRESOURCE(className) || strcmpiW( className, messageW )))
1432 DWORD layout;
1433 GetProcessDefaultLayout( &layout );
1434 if (layout & LAYOUT_RTL) cs->dwExStyle |= WS_EX_LAYOUTRTL;
1435 parent = GetDesktopWindow();
1439 WIN_FixCoordinates(cs, &sw); /* fix default coordinates */
1441 if ((cs->dwExStyle & WS_EX_DLGMODALFRAME) ||
1442 ((!(cs->dwExStyle & WS_EX_STATICEDGE)) &&
1443 (cs->style & (WS_DLGFRAME | WS_THICKFRAME))))
1444 cs->dwExStyle |= WS_EX_WINDOWEDGE;
1445 else
1446 cs->dwExStyle &= ~WS_EX_WINDOWEDGE;
1448 /* Create the window structure */
1450 if (!(wndPtr = create_window_handle( parent, owner, className, module, unicode )))
1452 WNDCLASSW wc;
1453 /* if it's a comctl32 class, GetClassInfo will load it, then we can retry */
1454 if (GetLastError() != ERROR_INVALID_HANDLE ||
1455 !GetClassInfoW( 0, className, &wc ) ||
1456 !(wndPtr = create_window_handle( parent, owner, className, module, unicode )))
1457 return 0;
1459 hwnd = wndPtr->obj.handle;
1461 /* Fill the window structure */
1463 wndPtr->tid = GetCurrentThreadId();
1464 wndPtr->hInstance = cs->hInstance;
1465 wndPtr->text = NULL;
1466 wndPtr->dwStyle = cs->style & ~WS_VISIBLE;
1467 wndPtr->dwExStyle = cs->dwExStyle;
1468 wndPtr->wIDmenu = 0;
1469 wndPtr->helpContext = 0;
1470 wndPtr->pScroll = NULL;
1471 wndPtr->userdata = 0;
1472 wndPtr->hIcon = 0;
1473 wndPtr->hIconSmall = 0;
1474 wndPtr->hSysMenu = 0;
1476 wndPtr->min_pos.x = wndPtr->min_pos.y = -1;
1477 wndPtr->max_pos.x = wndPtr->max_pos.y = -1;
1478 SetRect( &wndPtr->normal_rect, cs->x, cs->y, cs->x + cs->cx, cs->y + cs->cy );
1480 if (wndPtr->dwStyle & WS_SYSMENU) SetSystemMenu( hwnd, 0 );
1483 * Correct the window styles.
1485 * It affects only the style loaded into the WIN structure.
1488 if ((wndPtr->dwStyle & (WS_CHILD | WS_POPUP)) != WS_CHILD)
1490 wndPtr->dwStyle |= WS_CLIPSIBLINGS;
1491 if (!(wndPtr->dwStyle & WS_POPUP))
1492 wndPtr->dwStyle |= WS_CAPTION;
1495 /* WS_EX_WINDOWEDGE depends on some other styles */
1496 if (wndPtr->dwExStyle & WS_EX_DLGMODALFRAME)
1497 wndPtr->dwExStyle |= WS_EX_WINDOWEDGE;
1498 else if (wndPtr->dwStyle & (WS_DLGFRAME | WS_THICKFRAME))
1500 if (!((wndPtr->dwExStyle & WS_EX_STATICEDGE) &&
1501 (wndPtr->dwStyle & (WS_CHILD | WS_POPUP))))
1502 wndPtr->dwExStyle |= WS_EX_WINDOWEDGE;
1504 else
1505 wndPtr->dwExStyle &= ~WS_EX_WINDOWEDGE;
1507 if (!(wndPtr->dwStyle & (WS_CHILD | WS_POPUP)))
1508 wndPtr->flags |= WIN_NEED_SIZE;
1510 SERVER_START_REQ( set_window_info )
1512 req->handle = wine_server_user_handle( hwnd );
1513 req->flags = SET_WIN_STYLE | SET_WIN_EXSTYLE | SET_WIN_INSTANCE | SET_WIN_UNICODE;
1514 req->style = wndPtr->dwStyle;
1515 req->ex_style = wndPtr->dwExStyle;
1516 req->instance = wine_server_client_ptr( wndPtr->hInstance );
1517 req->is_unicode = (wndPtr->flags & WIN_ISUNICODE) != 0;
1518 req->extra_offset = -1;
1519 wine_server_call( req );
1521 SERVER_END_REQ;
1523 /* Set the window menu */
1525 if ((wndPtr->dwStyle & (WS_CHILD | WS_POPUP)) != WS_CHILD)
1527 if (cs->hMenu)
1529 if (!MENU_SetMenu(hwnd, cs->hMenu))
1531 WIN_ReleasePtr( wndPtr );
1532 free_window_handle( hwnd );
1533 return 0;
1536 else
1538 LPCWSTR menuName = (LPCWSTR)GetClassLongPtrW( hwnd, GCLP_MENUNAME );
1539 if (menuName)
1541 cs->hMenu = LoadMenuW( cs->hInstance, menuName );
1542 if (cs->hMenu) MENU_SetMenu( hwnd, cs->hMenu );
1546 else SetWindowLongPtrW( hwnd, GWLP_ID, (ULONG_PTR)cs->hMenu );
1548 /* call the WH_CBT hook */
1550 /* the window style passed to the hook must be the real window style,
1551 * rather than just the window style that the caller to CreateWindowEx
1552 * passed in, so we have to copy the original CREATESTRUCT and get the
1553 * the real style. */
1554 cbcs = *cs;
1555 cbcs.style = wndPtr->dwStyle;
1556 cbtc.lpcs = &cbcs;
1557 cbtc.hwndInsertAfter = HWND_TOP;
1558 WIN_ReleasePtr( wndPtr );
1559 if (HOOK_CallHooks( WH_CBT, HCBT_CREATEWND, (WPARAM)hwnd, (LPARAM)&cbtc, unicode )) goto failed;
1561 /* send the WM_GETMINMAXINFO message and fix the size if needed */
1563 cx = cs->cx;
1564 cy = cs->cy;
1565 if ((cs->style & WS_THICKFRAME) || !(cs->style & (WS_POPUP | WS_CHILD)))
1567 POINT maxSize, maxPos, minTrack, maxTrack;
1568 WINPOS_GetMinMaxInfo( hwnd, &maxSize, &maxPos, &minTrack, &maxTrack);
1569 if (maxTrack.x < cx) cx = maxTrack.x;
1570 if (maxTrack.y < cy) cy = maxTrack.y;
1571 if (minTrack.x > cx) cx = minTrack.x;
1572 if (minTrack.y > cy) cy = minTrack.y;
1575 if (cx < 0) cx = 0;
1576 if (cy < 0) cy = 0;
1577 SetRect( &rect, cs->x, cs->y, cs->x + cx, cs->y + cy );
1578 /* check for wraparound */
1579 if (cs->x + cx < cs->x) rect.right = 0x7fffffff;
1580 if (cs->y + cy < cs->y) rect.bottom = 0x7fffffff;
1581 if (!set_window_pos( hwnd, 0, SWP_NOZORDER | SWP_NOACTIVATE, &rect, &rect, NULL )) goto failed;
1583 /* send WM_NCCREATE */
1585 TRACE( "hwnd %p cs %d,%d %dx%d\n", hwnd, cs->x, cs->y, cx, cy );
1586 if (unicode)
1587 result = SendMessageW( hwnd, WM_NCCREATE, 0, (LPARAM)cs );
1588 else
1589 result = SendMessageA( hwnd, WM_NCCREATE, 0, (LPARAM)cs );
1590 if (!result)
1592 WARN( "%p: aborted by WM_NCCREATE\n", hwnd );
1593 goto failed;
1596 /* send WM_NCCALCSIZE */
1598 if (WIN_GetRectangles( hwnd, COORDS_PARENT, &rect, NULL ))
1600 /* yes, even if the CBT hook was called with HWND_TOP */
1601 HWND insert_after = (GetWindowLongW( hwnd, GWL_STYLE ) & WS_CHILD) ? HWND_BOTTOM : HWND_TOP;
1602 RECT client_rect = rect;
1604 /* the rectangle is in screen coords for WM_NCCALCSIZE when wparam is FALSE */
1605 MapWindowPoints( parent, 0, (POINT *)&client_rect, 2 );
1606 SendMessageW( hwnd, WM_NCCALCSIZE, FALSE, (LPARAM)&client_rect );
1607 MapWindowPoints( 0, parent, (POINT *)&client_rect, 2 );
1608 set_window_pos( hwnd, insert_after, SWP_NOACTIVATE, &rect, &client_rect, NULL );
1610 else return 0;
1612 /* send WM_CREATE */
1614 if (unicode)
1615 result = SendMessageW( hwnd, WM_CREATE, 0, (LPARAM)cs );
1616 else
1617 result = SendMessageA( hwnd, WM_CREATE, 0, (LPARAM)cs );
1618 if (result == -1) goto failed;
1620 /* call the driver */
1622 if (!USER_Driver->pCreateWindow( hwnd )) goto failed;
1624 NotifyWinEvent(EVENT_OBJECT_CREATE, hwnd, OBJID_WINDOW, 0);
1626 /* send the size messages */
1628 if (!(wndPtr = WIN_GetPtr( hwnd )) ||
1629 wndPtr == WND_OTHER_PROCESS || wndPtr == WND_DESKTOP) return 0;
1630 if (!(wndPtr->flags & WIN_NEED_SIZE))
1632 WIN_ReleasePtr( wndPtr );
1633 WIN_GetRectangles( hwnd, COORDS_PARENT, NULL, &rect );
1634 SendMessageW( hwnd, WM_SIZE, SIZE_RESTORED,
1635 MAKELONG(rect.right-rect.left, rect.bottom-rect.top));
1636 SendMessageW( hwnd, WM_MOVE, 0, MAKELONG( rect.left, rect.top ) );
1638 else WIN_ReleasePtr( wndPtr );
1640 /* Show the window, maximizing or minimizing if needed */
1642 style = WIN_SetStyle( hwnd, 0, WS_MAXIMIZE | WS_MINIMIZE );
1643 if (style & (WS_MINIMIZE | WS_MAXIMIZE))
1645 RECT newPos;
1646 UINT swFlag = (style & WS_MINIMIZE) ? SW_MINIMIZE : SW_MAXIMIZE;
1648 swFlag = WINPOS_MinMaximize( hwnd, swFlag, &newPos );
1649 swFlag |= SWP_FRAMECHANGED; /* Frame always gets changed */
1650 if (!(style & WS_VISIBLE) || (style & WS_CHILD) || GetActiveWindow()) swFlag |= SWP_NOACTIVATE;
1651 SetWindowPos( hwnd, 0, newPos.left, newPos.top, newPos.right - newPos.left,
1652 newPos.bottom - newPos.top, swFlag );
1655 /* Notify the parent window only */
1657 send_parent_notify( hwnd, WM_CREATE );
1658 if (!IsWindow( hwnd )) return 0;
1660 if (cs->style & WS_VISIBLE)
1662 if (cs->style & WS_MAXIMIZE)
1663 sw = SW_SHOW;
1664 else if (cs->style & WS_MINIMIZE)
1665 sw = SW_SHOWMINIMIZED;
1667 ShowWindow( hwnd, sw );
1668 if (cs->dwExStyle & WS_EX_MDICHILD)
1670 SendMessageW(cs->hwndParent, WM_MDIREFRESHMENU, 0, 0);
1671 /* ShowWindow won't activate child windows */
1672 SetWindowPos( hwnd, HWND_TOP, 0, 0, 0, 0, SWP_SHOWWINDOW | SWP_NOMOVE | SWP_NOSIZE );
1676 /* Call WH_SHELL hook */
1678 if (!(GetWindowLongW( hwnd, GWL_STYLE ) & WS_CHILD) && !GetWindow( hwnd, GW_OWNER ))
1679 HOOK_CallHooks( WH_SHELL, HSHELL_WINDOWCREATED, (WPARAM)hwnd, 0, TRUE );
1681 TRACE("created window %p\n", hwnd);
1682 return hwnd;
1684 failed:
1685 WIN_DestroyWindow( hwnd );
1686 return 0;
1690 /***********************************************************************
1691 * CreateWindowExA (USER32.@)
1693 HWND WINAPI DECLSPEC_HOTPATCH CreateWindowExA( DWORD exStyle, LPCSTR className,
1694 LPCSTR windowName, DWORD style, INT x,
1695 INT y, INT width, INT height,
1696 HWND parent, HMENU menu,
1697 HINSTANCE instance, LPVOID data )
1699 CREATESTRUCTA cs;
1701 cs.lpCreateParams = data;
1702 cs.hInstance = instance;
1703 cs.hMenu = menu;
1704 cs.hwndParent = parent;
1705 cs.x = x;
1706 cs.y = y;
1707 cs.cx = width;
1708 cs.cy = height;
1709 cs.style = style;
1710 cs.lpszName = windowName;
1711 cs.lpszClass = className;
1712 cs.dwExStyle = exStyle;
1714 if (!IS_INTRESOURCE(className))
1716 WCHAR bufferW[256];
1717 if (!MultiByteToWideChar( CP_ACP, 0, className, -1, bufferW, sizeof(bufferW)/sizeof(WCHAR) ))
1718 return 0;
1719 return wow_handlers.create_window( (CREATESTRUCTW *)&cs, bufferW, instance, FALSE );
1721 /* Note: we rely on the fact that CREATESTRUCTA and */
1722 /* CREATESTRUCTW have the same layout. */
1723 return wow_handlers.create_window( (CREATESTRUCTW *)&cs, (LPCWSTR)className, instance, FALSE );
1727 /***********************************************************************
1728 * CreateWindowExW (USER32.@)
1730 HWND WINAPI DECLSPEC_HOTPATCH CreateWindowExW( DWORD exStyle, LPCWSTR className,
1731 LPCWSTR windowName, DWORD style, INT x,
1732 INT y, INT width, INT height,
1733 HWND parent, HMENU menu,
1734 HINSTANCE instance, LPVOID data )
1736 CREATESTRUCTW cs;
1738 cs.lpCreateParams = data;
1739 cs.hInstance = instance;
1740 cs.hMenu = menu;
1741 cs.hwndParent = parent;
1742 cs.x = x;
1743 cs.y = y;
1744 cs.cx = width;
1745 cs.cy = height;
1746 cs.style = style;
1747 cs.lpszName = windowName;
1748 cs.lpszClass = className;
1749 cs.dwExStyle = exStyle;
1751 return wow_handlers.create_window( &cs, className, instance, TRUE );
1755 /***********************************************************************
1756 * WIN_SendDestroyMsg
1758 static void WIN_SendDestroyMsg( HWND hwnd )
1760 GUITHREADINFO info;
1762 info.cbSize = sizeof(info);
1763 if (GetGUIThreadInfo( GetCurrentThreadId(), &info ))
1765 if (hwnd == info.hwndCaret) DestroyCaret();
1766 if (hwnd == info.hwndActive) WINPOS_ActivateOtherWindow( hwnd );
1770 * Send the WM_DESTROY to the window.
1772 SendMessageW( hwnd, WM_DESTROY, 0, 0);
1775 * This WM_DESTROY message can trigger re-entrant calls to DestroyWindow
1776 * make sure that the window still exists when we come back.
1778 if (IsWindow(hwnd))
1780 HWND* pWndArray;
1781 int i;
1783 if (!(pWndArray = WIN_ListChildren( hwnd ))) return;
1785 for (i = 0; pWndArray[i]; i++)
1787 if (IsWindow( pWndArray[i] )) WIN_SendDestroyMsg( pWndArray[i] );
1789 HeapFree( GetProcessHeap(), 0, pWndArray );
1791 else
1792 WARN("\tdestroyed itself while in WM_DESTROY!\n");
1796 /***********************************************************************
1797 * DestroyWindow (USER32.@)
1799 BOOL WINAPI DestroyWindow( HWND hwnd )
1801 BOOL is_child;
1803 if (!(hwnd = WIN_IsCurrentThread( hwnd )) || is_desktop_window( hwnd ))
1805 SetLastError( ERROR_ACCESS_DENIED );
1806 return FALSE;
1809 TRACE("(%p)\n", hwnd);
1811 /* Call hooks */
1813 if (HOOK_CallHooks( WH_CBT, HCBT_DESTROYWND, (WPARAM)hwnd, 0, TRUE )) return FALSE;
1815 if (MENU_IsMenuActive() == hwnd)
1816 EndMenu();
1818 is_child = (GetWindowLongW( hwnd, GWL_STYLE ) & WS_CHILD) != 0;
1820 if (is_child)
1822 if (!USER_IsExitingThread( GetCurrentThreadId() ))
1823 send_parent_notify( hwnd, WM_DESTROY );
1825 else if (!GetWindow( hwnd, GW_OWNER ))
1827 HOOK_CallHooks( WH_SHELL, HSHELL_WINDOWDESTROYED, (WPARAM)hwnd, 0L, TRUE );
1828 /* FIXME: clean up palette - see "Internals" p.352 */
1831 if (!IsWindow(hwnd)) return TRUE;
1833 /* Hide the window */
1834 if (GetWindowLongW( hwnd, GWL_STYLE ) & WS_VISIBLE)
1836 /* Only child windows receive WM_SHOWWINDOW in DestroyWindow() */
1837 if (is_child)
1838 ShowWindow( hwnd, SW_HIDE );
1839 else
1840 SetWindowPos( hwnd, 0, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE |
1841 SWP_NOZORDER | SWP_NOACTIVATE | SWP_HIDEWINDOW );
1844 if (!IsWindow(hwnd)) return TRUE;
1846 /* Recursively destroy owned windows */
1848 if (!is_child)
1850 for (;;)
1852 int i;
1853 BOOL got_one = FALSE;
1854 HWND *list = WIN_ListChildren( GetDesktopWindow() );
1855 if (list)
1857 for (i = 0; list[i]; i++)
1859 if (GetWindow( list[i], GW_OWNER ) != hwnd) continue;
1860 if (WIN_IsCurrentThread( list[i] ))
1862 DestroyWindow( list[i] );
1863 got_one = TRUE;
1864 continue;
1866 WIN_SetOwner( list[i], 0 );
1868 HeapFree( GetProcessHeap(), 0, list );
1870 if (!got_one) break;
1874 /* Send destroy messages */
1876 WIN_SendDestroyMsg( hwnd );
1877 if (!IsWindow( hwnd )) return TRUE;
1879 if (GetClipboardOwner() == hwnd)
1880 CLIPBOARD_ReleaseOwner();
1882 /* Destroy the window storage */
1884 WIN_DestroyWindow( hwnd );
1885 return TRUE;
1889 /***********************************************************************
1890 * CloseWindow (USER32.@)
1892 BOOL WINAPI CloseWindow( HWND hwnd )
1894 if (GetWindowLongW( hwnd, GWL_STYLE ) & WS_CHILD) return FALSE;
1895 ShowWindow( hwnd, SW_MINIMIZE );
1896 return TRUE;
1900 /***********************************************************************
1901 * OpenIcon (USER32.@)
1903 BOOL WINAPI OpenIcon( HWND hwnd )
1905 if (!IsIconic( hwnd )) return FALSE;
1906 ShowWindow( hwnd, SW_SHOWNORMAL );
1907 return TRUE;
1911 /***********************************************************************
1912 * FindWindowExW (USER32.@)
1914 HWND WINAPI FindWindowExW( HWND parent, HWND child, LPCWSTR className, LPCWSTR title )
1916 HWND *list = NULL;
1917 HWND retvalue = 0;
1918 int i = 0, len = 0;
1919 WCHAR *buffer = NULL;
1921 if (!parent && child) parent = GetDesktopWindow();
1922 else if (parent == HWND_MESSAGE) parent = get_hwnd_message_parent();
1924 if (title)
1926 len = strlenW(title) + 1; /* one extra char to check for chars beyond the end */
1927 if (!(buffer = HeapAlloc( GetProcessHeap(), 0, (len + 1) * sizeof(WCHAR) ))) return 0;
1930 if (!(list = list_window_children( 0, parent, className, 0 ))) goto done;
1932 if (child)
1934 child = WIN_GetFullHandle( child );
1935 while (list[i] && list[i] != child) i++;
1936 if (!list[i]) goto done;
1937 i++; /* start from next window */
1940 if (title)
1942 while (list[i])
1944 if (GetWindowTextW( list[i], buffer, len + 1 ))
1946 if (!strcmpiW( buffer, title )) break;
1948 else
1950 if (!title[0]) break;
1952 i++;
1955 retvalue = list[i];
1957 done:
1958 HeapFree( GetProcessHeap(), 0, list );
1959 HeapFree( GetProcessHeap(), 0, buffer );
1960 return retvalue;
1965 /***********************************************************************
1966 * FindWindowA (USER32.@)
1968 HWND WINAPI FindWindowA( LPCSTR className, LPCSTR title )
1970 HWND ret = FindWindowExA( 0, 0, className, title );
1971 if (!ret) SetLastError (ERROR_CANNOT_FIND_WND_CLASS);
1972 return ret;
1976 /***********************************************************************
1977 * FindWindowExA (USER32.@)
1979 HWND WINAPI FindWindowExA( HWND parent, HWND child, LPCSTR className, LPCSTR title )
1981 LPWSTR titleW = NULL;
1982 HWND hwnd = 0;
1984 if (title)
1986 DWORD len = MultiByteToWideChar( CP_ACP, 0, title, -1, NULL, 0 );
1987 if (!(titleW = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) ))) return 0;
1988 MultiByteToWideChar( CP_ACP, 0, title, -1, titleW, len );
1991 if (!IS_INTRESOURCE(className))
1993 WCHAR classW[256];
1994 if (MultiByteToWideChar( CP_ACP, 0, className, -1, classW, sizeof(classW)/sizeof(WCHAR) ))
1995 hwnd = FindWindowExW( parent, child, classW, titleW );
1997 else
1999 hwnd = FindWindowExW( parent, child, (LPCWSTR)className, titleW );
2002 HeapFree( GetProcessHeap(), 0, titleW );
2003 return hwnd;
2007 /***********************************************************************
2008 * FindWindowW (USER32.@)
2010 HWND WINAPI FindWindowW( LPCWSTR className, LPCWSTR title )
2012 return FindWindowExW( 0, 0, className, title );
2016 /**********************************************************************
2017 * GetDesktopWindow (USER32.@)
2019 HWND WINAPI GetDesktopWindow(void)
2021 struct user_thread_info *thread_info = get_user_thread_info();
2023 if (thread_info->top_window) return thread_info->top_window;
2025 SERVER_START_REQ( get_desktop_window )
2027 req->force = 0;
2028 if (!wine_server_call( req ))
2030 thread_info->top_window = wine_server_ptr_handle( reply->top_window );
2031 thread_info->msg_window = wine_server_ptr_handle( reply->msg_window );
2034 SERVER_END_REQ;
2036 if (!thread_info->top_window)
2038 USEROBJECTFLAGS flags;
2039 if (!GetUserObjectInformationW( GetProcessWindowStation(), UOI_FLAGS, &flags,
2040 sizeof(flags), NULL ) || (flags.dwFlags & WSF_VISIBLE))
2042 static const WCHAR explorer[] = {'\\','e','x','p','l','o','r','e','r','.','e','x','e',0};
2043 static const WCHAR args[] = {' ','/','d','e','s','k','t','o','p',0};
2044 STARTUPINFOW si;
2045 PROCESS_INFORMATION pi;
2046 WCHAR windir[MAX_PATH];
2047 WCHAR app[MAX_PATH + sizeof(explorer)/sizeof(WCHAR)];
2048 WCHAR cmdline[MAX_PATH + (sizeof(explorer) + sizeof(args))/sizeof(WCHAR)];
2049 void *redir;
2051 memset( &si, 0, sizeof(si) );
2052 si.cb = sizeof(si);
2053 si.dwFlags = STARTF_USESTDHANDLES;
2054 si.hStdInput = 0;
2055 si.hStdOutput = 0;
2056 si.hStdError = GetStdHandle( STD_ERROR_HANDLE );
2058 GetSystemDirectoryW( windir, MAX_PATH );
2059 strcpyW( app, windir );
2060 strcatW( app, explorer );
2061 strcpyW( cmdline, app );
2062 strcatW( cmdline, args );
2064 Wow64DisableWow64FsRedirection( &redir );
2065 if (CreateProcessW( app, cmdline, NULL, NULL, FALSE, DETACHED_PROCESS,
2066 NULL, windir, &si, &pi ))
2068 TRACE( "started explorer pid %04x tid %04x\n", pi.dwProcessId, pi.dwThreadId );
2069 WaitForInputIdle( pi.hProcess, 10000 );
2070 CloseHandle( pi.hThread );
2071 CloseHandle( pi.hProcess );
2073 else WARN( "failed to start explorer, err %d\n", GetLastError() );
2074 Wow64RevertWow64FsRedirection( redir );
2076 else TRACE( "not starting explorer since winstation is not visible\n" );
2078 SERVER_START_REQ( get_desktop_window )
2080 req->force = 1;
2081 if (!wine_server_call( req ))
2083 thread_info->top_window = wine_server_ptr_handle( reply->top_window );
2084 thread_info->msg_window = wine_server_ptr_handle( reply->msg_window );
2087 SERVER_END_REQ;
2090 if (!thread_info->top_window || !USER_Driver->pCreateDesktopWindow( thread_info->top_window ))
2091 ERR( "failed to create desktop window\n" );
2093 return thread_info->top_window;
2097 /*******************************************************************
2098 * EnableWindow (USER32.@)
2100 BOOL WINAPI EnableWindow( HWND hwnd, BOOL enable )
2102 BOOL retvalue;
2103 HWND full_handle;
2105 if (is_broadcast(hwnd))
2107 SetLastError( ERROR_INVALID_PARAMETER );
2108 return FALSE;
2111 if (!(full_handle = WIN_IsCurrentThread( hwnd )))
2112 return SendMessageW( hwnd, WM_WINE_ENABLEWINDOW, enable, 0 );
2114 hwnd = full_handle;
2116 TRACE("( %p, %d )\n", hwnd, enable);
2118 retvalue = !IsWindowEnabled( hwnd );
2120 if (enable && retvalue)
2122 WIN_SetStyle( hwnd, 0, WS_DISABLED );
2123 SendMessageW( hwnd, WM_ENABLE, TRUE, 0 );
2125 else if (!enable && !retvalue)
2127 HWND capture_wnd;
2129 SendMessageW( hwnd, WM_CANCELMODE, 0, 0);
2131 WIN_SetStyle( hwnd, WS_DISABLED, 0 );
2133 if (hwnd == GetFocus())
2134 SetFocus( 0 ); /* A disabled window can't have the focus */
2136 capture_wnd = GetCapture();
2137 if (hwnd == capture_wnd || IsChild(hwnd, capture_wnd))
2138 ReleaseCapture(); /* A disabled window can't capture the mouse */
2140 SendMessageW( hwnd, WM_ENABLE, FALSE, 0 );
2142 return retvalue;
2146 /***********************************************************************
2147 * IsWindowEnabled (USER32.@)
2149 BOOL WINAPI IsWindowEnabled(HWND hWnd)
2151 return !(GetWindowLongW( hWnd, GWL_STYLE ) & WS_DISABLED);
2155 /***********************************************************************
2156 * IsWindowUnicode (USER32.@)
2158 BOOL WINAPI IsWindowUnicode( HWND hwnd )
2160 WND * wndPtr;
2161 BOOL retvalue = FALSE;
2163 if (!(wndPtr = WIN_GetPtr(hwnd))) return FALSE;
2165 if (wndPtr == WND_DESKTOP) return TRUE;
2167 if (wndPtr != WND_OTHER_PROCESS)
2169 retvalue = (wndPtr->flags & WIN_ISUNICODE) != 0;
2170 WIN_ReleasePtr( wndPtr );
2172 else
2174 SERVER_START_REQ( get_window_info )
2176 req->handle = wine_server_user_handle( hwnd );
2177 if (!wine_server_call_err( req )) retvalue = reply->is_unicode;
2179 SERVER_END_REQ;
2181 return retvalue;
2185 /**********************************************************************
2186 * WIN_GetWindowLong
2188 * Helper function for GetWindowLong().
2190 static LONG_PTR WIN_GetWindowLong( HWND hwnd, INT offset, UINT size, BOOL unicode )
2192 LONG_PTR retvalue = 0;
2193 WND *wndPtr;
2195 if (offset == GWLP_HWNDPARENT)
2197 HWND parent = GetAncestor( hwnd, GA_PARENT );
2198 if (parent == GetDesktopWindow()) parent = GetWindow( hwnd, GW_OWNER );
2199 return (ULONG_PTR)parent;
2202 if (!(wndPtr = WIN_GetPtr( hwnd )))
2204 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
2205 return 0;
2208 if (wndPtr == WND_OTHER_PROCESS || wndPtr == WND_DESKTOP)
2210 if (offset == GWLP_WNDPROC)
2212 SetLastError( ERROR_ACCESS_DENIED );
2213 return 0;
2215 SERVER_START_REQ( set_window_info )
2217 req->handle = wine_server_user_handle( hwnd );
2218 req->flags = 0; /* don't set anything, just retrieve */
2219 req->extra_offset = (offset >= 0) ? offset : -1;
2220 req->extra_size = (offset >= 0) ? size : 0;
2221 if (!wine_server_call_err( req ))
2223 switch(offset)
2225 case GWL_STYLE: retvalue = reply->old_style; break;
2226 case GWL_EXSTYLE: retvalue = reply->old_ex_style; break;
2227 case GWLP_ID: retvalue = reply->old_id; break;
2228 case GWLP_HINSTANCE: retvalue = (ULONG_PTR)wine_server_get_ptr( reply->old_instance ); break;
2229 case GWLP_USERDATA: retvalue = reply->old_user_data; break;
2230 default:
2231 if (offset >= 0) retvalue = get_win_data( &reply->old_extra_value, size );
2232 else SetLastError( ERROR_INVALID_INDEX );
2233 break;
2237 SERVER_END_REQ;
2238 return retvalue;
2241 /* now we have a valid wndPtr */
2243 if (offset >= 0)
2245 if (offset > (int)(wndPtr->cbWndExtra - size))
2247 WARN("Invalid offset %d\n", offset );
2248 WIN_ReleasePtr( wndPtr );
2249 SetLastError( ERROR_INVALID_INDEX );
2250 return 0;
2252 retvalue = get_win_data( (char *)wndPtr->wExtra + offset, size );
2254 /* Special case for dialog window procedure */
2255 if ((offset == DWLP_DLGPROC) && (size == sizeof(LONG_PTR)) && wndPtr->dlgInfo)
2256 retvalue = (LONG_PTR)WINPROC_GetProc( (WNDPROC)retvalue, unicode );
2257 WIN_ReleasePtr( wndPtr );
2258 return retvalue;
2261 switch(offset)
2263 case GWLP_USERDATA: retvalue = wndPtr->userdata; break;
2264 case GWL_STYLE: retvalue = wndPtr->dwStyle; break;
2265 case GWL_EXSTYLE: retvalue = wndPtr->dwExStyle; break;
2266 case GWLP_ID: retvalue = wndPtr->wIDmenu; break;
2267 case GWLP_HINSTANCE: retvalue = (ULONG_PTR)wndPtr->hInstance; break;
2268 case GWLP_WNDPROC:
2269 /* This looks like a hack only for the edit control (see tests). This makes these controls
2270 * more tolerant to A/W mismatches. The lack of W->A->W conversion for such a mismatch suggests
2271 * that the hack is in GetWindowLongPtr[AW], not in winprocs.
2273 if (wndPtr->winproc == BUILTIN_WINPROC(WINPROC_EDIT) && (!unicode != !(wndPtr->flags & WIN_ISUNICODE)))
2274 retvalue = (ULONG_PTR)wndPtr->winproc;
2275 else
2276 retvalue = (ULONG_PTR)WINPROC_GetProc( wndPtr->winproc, unicode );
2277 break;
2278 default:
2279 WARN("Unknown offset %d\n", offset );
2280 SetLastError( ERROR_INVALID_INDEX );
2281 break;
2283 WIN_ReleasePtr(wndPtr);
2284 return retvalue;
2288 /**********************************************************************
2289 * WIN_SetWindowLong
2291 * Helper function for SetWindowLong().
2293 * 0 is the failure code. However, in the case of failure SetLastError
2294 * must be set to distinguish between a 0 return value and a failure.
2296 LONG_PTR WIN_SetWindowLong( HWND hwnd, INT offset, UINT size, LONG_PTR newval, BOOL unicode )
2298 STYLESTRUCT style;
2299 BOOL ok, made_visible = FALSE;
2300 LONG_PTR retval = 0;
2301 WND *wndPtr;
2303 TRACE( "%p %d %lx %c\n", hwnd, offset, newval, unicode ? 'W' : 'A' );
2305 if (is_broadcast(hwnd))
2307 SetLastError( ERROR_INVALID_PARAMETER );
2308 return FALSE;
2311 if (!(wndPtr = WIN_GetPtr( hwnd )))
2313 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
2314 return 0;
2316 if (wndPtr == WND_DESKTOP)
2318 /* can't change anything on the desktop window */
2319 SetLastError( ERROR_ACCESS_DENIED );
2320 return 0;
2322 if (wndPtr == WND_OTHER_PROCESS)
2324 if (offset == GWLP_WNDPROC)
2326 SetLastError( ERROR_ACCESS_DENIED );
2327 return 0;
2329 if (offset > 32767 || offset < -32767)
2331 SetLastError( ERROR_INVALID_INDEX );
2332 return 0;
2334 return SendMessageW( hwnd, WM_WINE_SETWINDOWLONG, MAKEWPARAM( offset, size ), newval );
2337 /* first some special cases */
2338 switch( offset )
2340 case GWL_STYLE:
2341 style.styleOld = wndPtr->dwStyle;
2342 style.styleNew = newval;
2343 WIN_ReleasePtr( wndPtr );
2344 SendMessageW( hwnd, WM_STYLECHANGING, GWL_STYLE, (LPARAM)&style );
2345 if (!(wndPtr = WIN_GetPtr( hwnd )) || wndPtr == WND_OTHER_PROCESS) return 0;
2346 newval = style.styleNew;
2347 /* WS_CLIPSIBLINGS can't be reset on top-level windows */
2348 if (wndPtr->parent == GetDesktopWindow()) newval |= WS_CLIPSIBLINGS;
2349 /* FIXME: changing WS_DLGFRAME | WS_THICKFRAME is supposed to change
2350 WS_EX_WINDOWEDGE too */
2351 break;
2352 case GWL_EXSTYLE:
2353 style.styleOld = wndPtr->dwExStyle;
2354 style.styleNew = newval;
2355 WIN_ReleasePtr( wndPtr );
2356 SendMessageW( hwnd, WM_STYLECHANGING, GWL_EXSTYLE, (LPARAM)&style );
2357 if (!(wndPtr = WIN_GetPtr( hwnd )) || wndPtr == WND_OTHER_PROCESS) return 0;
2358 /* WS_EX_TOPMOST can only be changed through SetWindowPos */
2359 newval = (style.styleNew & ~WS_EX_TOPMOST) | (wndPtr->dwExStyle & WS_EX_TOPMOST);
2360 /* WS_EX_WINDOWEDGE depends on some other styles */
2361 if (newval & WS_EX_DLGMODALFRAME)
2362 newval |= WS_EX_WINDOWEDGE;
2363 else if (!(newval & WS_EX_STATICEDGE) && (wndPtr->dwStyle & (WS_DLGFRAME | WS_THICKFRAME)))
2364 newval |= WS_EX_WINDOWEDGE;
2365 else
2366 newval &= ~WS_EX_WINDOWEDGE;
2367 break;
2368 case GWLP_HWNDPARENT:
2369 if (wndPtr->parent == GetDesktopWindow())
2371 WIN_ReleasePtr( wndPtr );
2372 return (ULONG_PTR)WIN_SetOwner( hwnd, (HWND)newval );
2374 else
2376 WIN_ReleasePtr( wndPtr );
2377 return (ULONG_PTR)SetParent( hwnd, (HWND)newval );
2379 case GWLP_WNDPROC:
2381 WNDPROC proc;
2382 UINT old_flags = wndPtr->flags;
2383 retval = WIN_GetWindowLong( hwnd, offset, size, unicode );
2384 proc = WINPROC_AllocProc( (WNDPROC)newval, unicode );
2385 if (proc) wndPtr->winproc = proc;
2386 if (WINPROC_IsUnicode( proc, unicode )) wndPtr->flags |= WIN_ISUNICODE;
2387 else wndPtr->flags &= ~WIN_ISUNICODE;
2388 if (!((old_flags ^ wndPtr->flags) & WIN_ISUNICODE))
2390 WIN_ReleasePtr( wndPtr );
2391 return retval;
2393 /* update is_unicode flag on the server side */
2394 break;
2396 case GWLP_ID:
2397 case GWLP_HINSTANCE:
2398 case GWLP_USERDATA:
2399 break;
2400 case DWLP_DLGPROC:
2401 if ((wndPtr->cbWndExtra - sizeof(LONG_PTR) >= DWLP_DLGPROC) &&
2402 (size == sizeof(LONG_PTR)) && wndPtr->dlgInfo)
2404 WNDPROC *ptr = (WNDPROC *)((char *)wndPtr->wExtra + DWLP_DLGPROC);
2405 retval = (ULONG_PTR)WINPROC_GetProc( *ptr, unicode );
2406 *ptr = WINPROC_AllocProc( (WNDPROC)newval, unicode );
2407 WIN_ReleasePtr( wndPtr );
2408 return retval;
2410 /* fall through */
2411 default:
2412 if (offset < 0 || offset > (int)(wndPtr->cbWndExtra - size))
2414 WARN("Invalid offset %d\n", offset );
2415 WIN_ReleasePtr( wndPtr );
2416 SetLastError( ERROR_INVALID_INDEX );
2417 return 0;
2419 else if (get_win_data( (char *)wndPtr->wExtra + offset, size ) == newval)
2421 /* already set to the same value */
2422 WIN_ReleasePtr( wndPtr );
2423 return newval;
2425 break;
2428 SERVER_START_REQ( set_window_info )
2430 req->handle = wine_server_user_handle( hwnd );
2431 req->extra_offset = -1;
2432 switch(offset)
2434 case GWL_STYLE:
2435 req->flags = SET_WIN_STYLE;
2436 req->style = newval;
2437 break;
2438 case GWL_EXSTYLE:
2439 req->flags = SET_WIN_EXSTYLE;
2440 req->ex_style = newval;
2441 break;
2442 case GWLP_ID:
2443 req->flags = SET_WIN_ID;
2444 req->id = newval;
2445 break;
2446 case GWLP_HINSTANCE:
2447 req->flags = SET_WIN_INSTANCE;
2448 req->instance = wine_server_client_ptr( (void *)newval );
2449 break;
2450 case GWLP_WNDPROC:
2451 req->flags = SET_WIN_UNICODE;
2452 req->is_unicode = (wndPtr->flags & WIN_ISUNICODE) != 0;
2453 break;
2454 case GWLP_USERDATA:
2455 req->flags = SET_WIN_USERDATA;
2456 req->user_data = newval;
2457 break;
2458 default:
2459 req->flags = SET_WIN_EXTRA;
2460 req->extra_offset = offset;
2461 req->extra_size = size;
2462 set_win_data( &req->extra_value, newval, size );
2464 if ((ok = !wine_server_call_err( req )))
2466 switch(offset)
2468 case GWL_STYLE:
2469 wndPtr->dwStyle = newval;
2470 retval = reply->old_style;
2471 break;
2472 case GWL_EXSTYLE:
2473 wndPtr->dwExStyle = newval;
2474 retval = reply->old_ex_style;
2475 break;
2476 case GWLP_ID:
2477 wndPtr->wIDmenu = newval;
2478 retval = reply->old_id;
2479 break;
2480 case GWLP_HINSTANCE:
2481 wndPtr->hInstance = (HINSTANCE)newval;
2482 retval = (ULONG_PTR)wine_server_get_ptr( reply->old_instance );
2483 break;
2484 case GWLP_WNDPROC:
2485 break;
2486 case GWLP_USERDATA:
2487 wndPtr->userdata = newval;
2488 retval = reply->old_user_data;
2489 break;
2490 default:
2491 retval = get_win_data( (char *)wndPtr->wExtra + offset, size );
2492 set_win_data( (char *)wndPtr->wExtra + offset, newval, size );
2493 break;
2497 SERVER_END_REQ;
2499 if ((offset == GWL_STYLE && ((style.styleOld ^ style.styleNew) & WS_VISIBLE)) ||
2500 (offset == GWL_EXSTYLE && ((style.styleOld ^ style.styleNew) & WS_EX_LAYERED)))
2502 made_visible = !(wndPtr->flags & WIN_HIDDEN) && (wndPtr->dwStyle & WS_VISIBLE);
2503 invalidate_dce( wndPtr, NULL );
2505 WIN_ReleasePtr( wndPtr );
2507 if (!ok) return 0;
2509 if (offset == GWL_STYLE || offset == GWL_EXSTYLE)
2511 style.styleOld = retval;
2512 style.styleNew = newval;
2513 USER_Driver->pSetWindowStyle( hwnd, offset, &style );
2514 if (made_visible) update_window_state( hwnd );
2515 SendMessageW( hwnd, WM_STYLECHANGED, offset, (LPARAM)&style );
2518 return retval;
2522 /**********************************************************************
2523 * GetWindowWord (USER32.@)
2525 WORD WINAPI GetWindowWord( HWND hwnd, INT offset )
2527 switch(offset)
2529 case GWLP_ID:
2530 case GWLP_HINSTANCE:
2531 case GWLP_HWNDPARENT:
2532 break;
2533 default:
2534 if (offset < 0)
2536 WARN("Invalid offset %d\n", offset );
2537 SetLastError( ERROR_INVALID_INDEX );
2538 return 0;
2540 break;
2542 return WIN_GetWindowLong( hwnd, offset, sizeof(WORD), FALSE );
2546 /**********************************************************************
2547 * GetWindowLongA (USER32.@)
2549 LONG WINAPI GetWindowLongA( HWND hwnd, INT offset )
2551 return WIN_GetWindowLong( hwnd, offset, sizeof(LONG), FALSE );
2555 /**********************************************************************
2556 * GetWindowLongW (USER32.@)
2558 LONG WINAPI GetWindowLongW( HWND hwnd, INT offset )
2560 return WIN_GetWindowLong( hwnd, offset, sizeof(LONG), TRUE );
2564 /**********************************************************************
2565 * SetWindowWord (USER32.@)
2567 WORD WINAPI SetWindowWord( HWND hwnd, INT offset, WORD newval )
2569 switch(offset)
2571 case GWLP_ID:
2572 case GWLP_HINSTANCE:
2573 case GWLP_HWNDPARENT:
2574 break;
2575 default:
2576 if (offset < 0)
2578 WARN("Invalid offset %d\n", offset );
2579 SetLastError( ERROR_INVALID_INDEX );
2580 return 0;
2582 break;
2584 return WIN_SetWindowLong( hwnd, offset, sizeof(WORD), newval, FALSE );
2588 /**********************************************************************
2589 * SetWindowLongA (USER32.@)
2591 * See SetWindowLongW.
2593 LONG WINAPI DECLSPEC_HOTPATCH SetWindowLongA( HWND hwnd, INT offset, LONG newval )
2595 return WIN_SetWindowLong( hwnd, offset, sizeof(LONG), newval, FALSE );
2599 /**********************************************************************
2600 * SetWindowLongW (USER32.@) Set window attribute
2602 * SetWindowLong() alters one of a window's attributes or sets a 32-bit (long)
2603 * value in a window's extra memory.
2605 * The _hwnd_ parameter specifies the handle to a window that
2606 * has extra memory. The _newval_ parameter contains the new
2607 * attribute or extra memory value. If positive, the _offset_
2608 * parameter is the byte-addressed location in the window's extra
2609 * memory to set. If negative, _offset_ specifies the window
2610 * attribute to set, and should be one of the following values:
2612 * GWL_EXSTYLE The window's extended window style
2614 * GWL_STYLE The window's window style.
2616 * GWLP_WNDPROC Pointer to the window's window procedure.
2618 * GWLP_HINSTANCE The window's application instance handle.
2620 * GWLP_ID The window's identifier.
2622 * GWLP_USERDATA The window's user-specified data.
2624 * If the window is a dialog box, the _offset_ parameter can be one of
2625 * the following values:
2627 * DWLP_DLGPROC The address of the window's dialog box procedure.
2629 * DWLP_MSGRESULT The return value of a message
2630 * that the dialog box procedure processed.
2632 * DWLP_USER Application specific information.
2634 * RETURNS
2636 * If successful, returns the previous value located at _offset_. Otherwise,
2637 * returns 0.
2639 * NOTES
2641 * Extra memory for a window class is specified by a nonzero cbWndExtra
2642 * parameter of the WNDCLASS structure passed to RegisterClass() at the
2643 * time of class creation.
2645 * Using GWL_WNDPROC to set a new window procedure effectively creates
2646 * a window subclass. Use CallWindowProc() in the new windows procedure
2647 * to pass messages to the superclass's window procedure.
2649 * The user data is reserved for use by the application which created
2650 * the window.
2652 * Do not use GWL_STYLE to change the window's WS_DISABLED style;
2653 * instead, call the EnableWindow() function to change the window's
2654 * disabled state.
2656 * Do not use GWL_HWNDPARENT to reset the window's parent, use
2657 * SetParent() instead.
2659 * Win95:
2660 * When offset is GWL_STYLE and the calling app's ver is 4.0,
2661 * it sends WM_STYLECHANGING before changing the settings
2662 * and WM_STYLECHANGED afterwards.
2663 * App ver 4.0 can't use SetWindowLong to change WS_EX_TOPMOST.
2665 LONG WINAPI SetWindowLongW(
2666 HWND hwnd, /* [in] window to alter */
2667 INT offset, /* [in] offset, in bytes, of location to alter */
2668 LONG newval /* [in] new value of location */
2670 return WIN_SetWindowLong( hwnd, offset, sizeof(LONG), newval, TRUE );
2674 /*******************************************************************
2675 * GetWindowTextA (USER32.@)
2677 INT WINAPI GetWindowTextA( HWND hwnd, LPSTR lpString, INT nMaxCount )
2679 WCHAR *buffer;
2681 if (!lpString) return 0;
2683 if (WIN_IsCurrentProcess( hwnd ))
2684 return (INT)SendMessageA( hwnd, WM_GETTEXT, nMaxCount, (LPARAM)lpString );
2686 /* when window belongs to other process, don't send a message */
2687 if (nMaxCount <= 0) return 0;
2688 if (!(buffer = HeapAlloc( GetProcessHeap(), 0, nMaxCount * sizeof(WCHAR) ))) return 0;
2689 get_server_window_text( hwnd, buffer, nMaxCount );
2690 if (!WideCharToMultiByte( CP_ACP, 0, buffer, -1, lpString, nMaxCount, NULL, NULL ))
2691 lpString[nMaxCount-1] = 0;
2692 HeapFree( GetProcessHeap(), 0, buffer );
2693 return strlen(lpString);
2697 /*******************************************************************
2698 * InternalGetWindowText (USER32.@)
2700 INT WINAPI InternalGetWindowText(HWND hwnd,LPWSTR lpString,INT nMaxCount )
2702 WND *win;
2704 if (nMaxCount <= 0) return 0;
2705 if (!(win = WIN_GetPtr( hwnd ))) return 0;
2706 if (win == WND_DESKTOP) lpString[0] = 0;
2707 else if (win != WND_OTHER_PROCESS)
2709 if (win->text) lstrcpynW( lpString, win->text, nMaxCount );
2710 else lpString[0] = 0;
2711 WIN_ReleasePtr( win );
2713 else
2715 get_server_window_text( hwnd, lpString, nMaxCount );
2717 return strlenW(lpString);
2721 /*******************************************************************
2722 * GetWindowTextW (USER32.@)
2724 INT WINAPI GetWindowTextW( HWND hwnd, LPWSTR lpString, INT nMaxCount )
2726 if (!lpString) return 0;
2728 if (WIN_IsCurrentProcess( hwnd ))
2729 return (INT)SendMessageW( hwnd, WM_GETTEXT, nMaxCount, (LPARAM)lpString );
2731 /* when window belongs to other process, don't send a message */
2732 if (nMaxCount <= 0) return 0;
2733 get_server_window_text( hwnd, lpString, nMaxCount );
2734 return strlenW(lpString);
2738 /*******************************************************************
2739 * SetWindowTextA (USER32.@)
2740 * SetWindowText (USER32.@)
2742 BOOL WINAPI DECLSPEC_HOTPATCH SetWindowTextA( HWND hwnd, LPCSTR lpString )
2744 if (is_broadcast(hwnd))
2746 SetLastError( ERROR_INVALID_PARAMETER );
2747 return FALSE;
2749 if (!WIN_IsCurrentProcess( hwnd ))
2750 WARN( "setting text %s of other process window %p should not use SendMessage\n",
2751 debugstr_a(lpString), hwnd );
2752 return (BOOL)SendMessageA( hwnd, WM_SETTEXT, 0, (LPARAM)lpString );
2756 /*******************************************************************
2757 * SetWindowTextW (USER32.@)
2759 BOOL WINAPI DECLSPEC_HOTPATCH SetWindowTextW( HWND hwnd, LPCWSTR lpString )
2761 if (is_broadcast(hwnd))
2763 SetLastError( ERROR_INVALID_PARAMETER );
2764 return FALSE;
2766 if (!WIN_IsCurrentProcess( hwnd ))
2767 WARN( "setting text %s of other process window %p should not use SendMessage\n",
2768 debugstr_w(lpString), hwnd );
2769 return (BOOL)SendMessageW( hwnd, WM_SETTEXT, 0, (LPARAM)lpString );
2773 /*******************************************************************
2774 * GetWindowTextLengthA (USER32.@)
2776 INT WINAPI GetWindowTextLengthA( HWND hwnd )
2778 return SendMessageA( hwnd, WM_GETTEXTLENGTH, 0, 0 );
2781 /*******************************************************************
2782 * GetWindowTextLengthW (USER32.@)
2784 INT WINAPI GetWindowTextLengthW( HWND hwnd )
2786 return SendMessageW( hwnd, WM_GETTEXTLENGTH, 0, 0 );
2790 /*******************************************************************
2791 * IsWindow (USER32.@)
2793 BOOL WINAPI IsWindow( HWND hwnd )
2795 WND *ptr;
2796 BOOL ret;
2798 if (!(ptr = WIN_GetPtr( hwnd ))) return FALSE;
2799 if (ptr == WND_DESKTOP) return TRUE;
2801 if (ptr != WND_OTHER_PROCESS)
2803 WIN_ReleasePtr( ptr );
2804 return TRUE;
2807 /* check other processes */
2808 SERVER_START_REQ( get_window_info )
2810 req->handle = wine_server_user_handle( hwnd );
2811 ret = !wine_server_call_err( req );
2813 SERVER_END_REQ;
2814 return ret;
2818 /***********************************************************************
2819 * GetWindowThreadProcessId (USER32.@)
2821 DWORD WINAPI GetWindowThreadProcessId( HWND hwnd, LPDWORD process )
2823 WND *ptr;
2824 DWORD tid = 0;
2826 if (!(ptr = WIN_GetPtr( hwnd )))
2828 SetLastError( ERROR_INVALID_WINDOW_HANDLE);
2829 return 0;
2832 if (ptr != WND_OTHER_PROCESS && ptr != WND_DESKTOP)
2834 /* got a valid window */
2835 tid = ptr->tid;
2836 if (process) *process = GetCurrentProcessId();
2837 WIN_ReleasePtr( ptr );
2838 return tid;
2841 /* check other processes */
2842 SERVER_START_REQ( get_window_info )
2844 req->handle = wine_server_user_handle( hwnd );
2845 if (!wine_server_call_err( req ))
2847 tid = (DWORD)reply->tid;
2848 if (process) *process = (DWORD)reply->pid;
2851 SERVER_END_REQ;
2852 return tid;
2856 /*****************************************************************
2857 * GetParent (USER32.@)
2859 HWND WINAPI GetParent( HWND hwnd )
2861 WND *wndPtr;
2862 HWND retvalue = 0;
2864 if (!(wndPtr = WIN_GetPtr( hwnd )))
2866 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
2867 return 0;
2869 if (wndPtr == WND_DESKTOP) return 0;
2870 if (wndPtr == WND_OTHER_PROCESS)
2872 LONG style = GetWindowLongW( hwnd, GWL_STYLE );
2873 if (style & (WS_POPUP | WS_CHILD))
2875 SERVER_START_REQ( get_window_tree )
2877 req->handle = wine_server_user_handle( hwnd );
2878 if (!wine_server_call_err( req ))
2880 if (style & WS_POPUP) retvalue = wine_server_ptr_handle( reply->owner );
2881 else if (style & WS_CHILD) retvalue = wine_server_ptr_handle( reply->parent );
2884 SERVER_END_REQ;
2887 else
2889 if (wndPtr->dwStyle & WS_POPUP) retvalue = wndPtr->owner;
2890 else if (wndPtr->dwStyle & WS_CHILD) retvalue = wndPtr->parent;
2891 WIN_ReleasePtr( wndPtr );
2893 return retvalue;
2897 /*****************************************************************
2898 * GetAncestor (USER32.@)
2900 HWND WINAPI GetAncestor( HWND hwnd, UINT type )
2902 WND *win;
2903 HWND *list, ret = 0;
2905 switch(type)
2907 case GA_PARENT:
2908 if (!(win = WIN_GetPtr( hwnd )))
2910 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
2911 return 0;
2913 if (win == WND_DESKTOP) return 0;
2914 if (win != WND_OTHER_PROCESS)
2916 ret = win->parent;
2917 WIN_ReleasePtr( win );
2919 else /* need to query the server */
2921 SERVER_START_REQ( get_window_tree )
2923 req->handle = wine_server_user_handle( hwnd );
2924 if (!wine_server_call_err( req )) ret = wine_server_ptr_handle( reply->parent );
2926 SERVER_END_REQ;
2928 break;
2930 case GA_ROOT:
2931 if (!(list = list_window_parents( hwnd ))) return 0;
2933 if (!list[0] || !list[1]) ret = WIN_GetFullHandle( hwnd ); /* top-level window */
2934 else
2936 int count = 2;
2937 while (list[count]) count++;
2938 ret = list[count - 2]; /* get the one before the desktop */
2940 HeapFree( GetProcessHeap(), 0, list );
2941 break;
2943 case GA_ROOTOWNER:
2944 if (is_desktop_window( hwnd )) return 0;
2945 ret = WIN_GetFullHandle( hwnd );
2946 for (;;)
2948 HWND parent = GetParent( ret );
2949 if (!parent) break;
2950 ret = parent;
2952 break;
2954 return ret;
2958 /*****************************************************************
2959 * SetParent (USER32.@)
2961 HWND WINAPI SetParent( HWND hwnd, HWND parent )
2963 HWND full_handle;
2964 HWND old_parent = 0;
2965 BOOL was_visible;
2966 WND *wndPtr;
2967 POINT pt;
2968 BOOL ret;
2970 if (is_broadcast(hwnd) || is_broadcast(parent))
2972 SetLastError(ERROR_INVALID_PARAMETER);
2973 return 0;
2976 if (!parent) parent = GetDesktopWindow();
2977 else if (parent == HWND_MESSAGE) parent = get_hwnd_message_parent();
2978 else parent = WIN_GetFullHandle( parent );
2980 if (!IsWindow( parent ))
2982 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
2983 return 0;
2986 /* Some applications try to set a child as a parent */
2987 if (IsChild(hwnd, parent))
2989 SetLastError( ERROR_INVALID_PARAMETER );
2990 return 0;
2993 if (!(full_handle = WIN_IsCurrentThread( hwnd )))
2994 return (HWND)SendMessageW( hwnd, WM_WINE_SETPARENT, (WPARAM)parent, 0 );
2996 if (full_handle == parent)
2998 SetLastError( ERROR_INVALID_PARAMETER );
2999 return 0;
3002 /* Windows hides the window first, then shows it again
3003 * including the WM_SHOWWINDOW messages and all */
3004 was_visible = ShowWindow( hwnd, SW_HIDE );
3006 wndPtr = WIN_GetPtr( hwnd );
3007 if (!wndPtr || wndPtr == WND_OTHER_PROCESS || wndPtr == WND_DESKTOP) return 0;
3009 pt.x = wndPtr->rectWindow.left;
3010 pt.y = wndPtr->rectWindow.top;
3012 SERVER_START_REQ( set_parent )
3014 req->handle = wine_server_user_handle( hwnd );
3015 req->parent = wine_server_user_handle( parent );
3016 if ((ret = !wine_server_call( req )))
3018 old_parent = wine_server_ptr_handle( reply->old_parent );
3019 wndPtr->parent = parent = wine_server_ptr_handle( reply->full_parent );
3023 SERVER_END_REQ;
3024 WIN_ReleasePtr( wndPtr );
3025 if (!ret) return 0;
3027 USER_Driver->pSetParent( full_handle, parent, old_parent );
3029 /* SetParent additionally needs to make hwnd the topmost window
3030 in the x-order and send the expected WM_WINDOWPOSCHANGING and
3031 WM_WINDOWPOSCHANGED notification messages.
3033 SetWindowPos( hwnd, HWND_TOP, pt.x, pt.y, 0, 0, SWP_NOSIZE );
3035 if (was_visible) ShowWindow( hwnd, SW_SHOW );
3037 return old_parent;
3041 /*******************************************************************
3042 * IsChild (USER32.@)
3044 BOOL WINAPI IsChild( HWND parent, HWND child )
3046 HWND *list;
3047 int i;
3048 BOOL ret = FALSE;
3050 if (!(GetWindowLongW( child, GWL_STYLE ) & WS_CHILD)) return FALSE;
3051 if (!(list = list_window_parents( child ))) return FALSE;
3052 parent = WIN_GetFullHandle( parent );
3053 for (i = 0; list[i]; i++)
3055 if (list[i] == parent)
3057 ret = list[i] && list[i+1];
3058 break;
3060 if (!(GetWindowLongW( list[i], GWL_STYLE ) & WS_CHILD)) break;
3062 HeapFree( GetProcessHeap(), 0, list );
3063 return ret;
3067 /***********************************************************************
3068 * IsWindowVisible (USER32.@)
3070 BOOL WINAPI IsWindowVisible( HWND hwnd )
3072 HWND *list;
3073 BOOL retval = TRUE;
3074 int i;
3076 if (!(GetWindowLongW( hwnd, GWL_STYLE ) & WS_VISIBLE)) return FALSE;
3077 if (!(list = list_window_parents( hwnd ))) return TRUE;
3078 if (list[0])
3080 for (i = 0; list[i+1]; i++)
3081 if (!(GetWindowLongW( list[i], GWL_STYLE ) & WS_VISIBLE)) break;
3082 retval = !list[i+1] && (list[i] == GetDesktopWindow()); /* top message window isn't visible */
3084 HeapFree( GetProcessHeap(), 0, list );
3085 return retval;
3089 /***********************************************************************
3090 * WIN_IsWindowDrawable
3092 * hwnd is drawable when it is visible, all parents are not
3093 * minimized, and it is itself not minimized unless we are
3094 * trying to draw its default class icon.
3096 BOOL WIN_IsWindowDrawable( HWND hwnd, BOOL icon )
3098 HWND *list;
3099 BOOL retval = TRUE;
3100 int i;
3101 LONG style = GetWindowLongW( hwnd, GWL_STYLE );
3103 if (!(style & WS_VISIBLE)) return FALSE;
3104 if ((style & WS_MINIMIZE) && icon && GetClassLongPtrW( hwnd, GCLP_HICON )) return FALSE;
3106 if (!(list = list_window_parents( hwnd ))) return TRUE;
3107 if (list[0])
3109 for (i = 0; list[i+1]; i++)
3110 if ((GetWindowLongW( list[i], GWL_STYLE ) & (WS_VISIBLE|WS_MINIMIZE)) != WS_VISIBLE)
3111 break;
3112 retval = !list[i+1] && (list[i] == GetDesktopWindow()); /* top message window isn't visible */
3114 HeapFree( GetProcessHeap(), 0, list );
3115 return retval;
3119 /*******************************************************************
3120 * GetTopWindow (USER32.@)
3122 HWND WINAPI GetTopWindow( HWND hwnd )
3124 if (!hwnd) hwnd = GetDesktopWindow();
3125 return GetWindow( hwnd, GW_CHILD );
3129 /*******************************************************************
3130 * GetWindow (USER32.@)
3132 HWND WINAPI GetWindow( HWND hwnd, UINT rel )
3134 HWND retval = 0;
3136 if (rel == GW_OWNER) /* this one may be available locally */
3138 WND *wndPtr = WIN_GetPtr( hwnd );
3139 if (!wndPtr)
3141 SetLastError( ERROR_INVALID_HANDLE );
3142 return 0;
3144 if (wndPtr == WND_DESKTOP) return 0;
3145 if (wndPtr != WND_OTHER_PROCESS)
3147 retval = wndPtr->owner;
3148 WIN_ReleasePtr( wndPtr );
3149 return retval;
3151 /* else fall through to server call */
3154 SERVER_START_REQ( get_window_tree )
3156 req->handle = wine_server_user_handle( hwnd );
3157 if (!wine_server_call_err( req ))
3159 switch(rel)
3161 case GW_HWNDFIRST:
3162 retval = wine_server_ptr_handle( reply->first_sibling );
3163 break;
3164 case GW_HWNDLAST:
3165 retval = wine_server_ptr_handle( reply->last_sibling );
3166 break;
3167 case GW_HWNDNEXT:
3168 retval = wine_server_ptr_handle( reply->next_sibling );
3169 break;
3170 case GW_HWNDPREV:
3171 retval = wine_server_ptr_handle( reply->prev_sibling );
3172 break;
3173 case GW_OWNER:
3174 retval = wine_server_ptr_handle( reply->owner );
3175 break;
3176 case GW_CHILD:
3177 retval = wine_server_ptr_handle( reply->first_child );
3178 break;
3182 SERVER_END_REQ;
3183 return retval;
3187 /*******************************************************************
3188 * ShowOwnedPopups (USER32.@)
3190 BOOL WINAPI ShowOwnedPopups( HWND owner, BOOL fShow )
3192 int count = 0;
3193 WND *pWnd;
3194 HWND *win_array = WIN_ListChildren( GetDesktopWindow() );
3196 if (!win_array) return TRUE;
3198 while (win_array[count]) count++;
3199 while (--count >= 0)
3201 if (GetWindow( win_array[count], GW_OWNER ) != owner) continue;
3202 if (!(pWnd = WIN_GetPtr( win_array[count] ))) continue;
3203 if (pWnd == WND_OTHER_PROCESS) continue;
3204 if (fShow)
3206 if (pWnd->flags & WIN_NEEDS_SHOW_OWNEDPOPUP)
3208 WIN_ReleasePtr( pWnd );
3209 /* In Windows, ShowOwnedPopups(TRUE) generates
3210 * WM_SHOWWINDOW messages with SW_PARENTOPENING,
3211 * regardless of the state of the owner
3213 SendMessageW(win_array[count], WM_SHOWWINDOW, SW_SHOWNORMAL, SW_PARENTOPENING);
3214 continue;
3217 else
3219 if (pWnd->dwStyle & WS_VISIBLE)
3221 WIN_ReleasePtr( pWnd );
3222 /* In Windows, ShowOwnedPopups(FALSE) generates
3223 * WM_SHOWWINDOW messages with SW_PARENTCLOSING,
3224 * regardless of the state of the owner
3226 SendMessageW(win_array[count], WM_SHOWWINDOW, SW_HIDE, SW_PARENTCLOSING);
3227 continue;
3230 WIN_ReleasePtr( pWnd );
3232 HeapFree( GetProcessHeap(), 0, win_array );
3233 return TRUE;
3237 /*******************************************************************
3238 * GetLastActivePopup (USER32.@)
3240 HWND WINAPI GetLastActivePopup( HWND hwnd )
3242 HWND retval = hwnd;
3244 SERVER_START_REQ( get_window_info )
3246 req->handle = wine_server_user_handle( hwnd );
3247 if (!wine_server_call_err( req )) retval = wine_server_ptr_handle( reply->last_active );
3249 SERVER_END_REQ;
3250 return retval;
3254 /*******************************************************************
3255 * WIN_ListChildren
3257 * Build an array of the children of a given window. The array must be
3258 * freed with HeapFree. Returns NULL when no windows are found.
3260 HWND *WIN_ListChildren( HWND hwnd )
3262 if (!hwnd)
3264 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
3265 return NULL;
3267 return list_window_children( 0, hwnd, NULL, 0 );
3271 /*******************************************************************
3272 * EnumWindows (USER32.@)
3274 BOOL WINAPI EnumWindows( WNDENUMPROC lpEnumFunc, LPARAM lParam )
3276 HWND *list;
3277 BOOL ret = TRUE;
3278 int i;
3280 USER_CheckNotLock();
3282 /* We have to build a list of all windows first, to avoid */
3283 /* unpleasant side-effects, for instance if the callback */
3284 /* function changes the Z-order of the windows. */
3286 if (!(list = WIN_ListChildren( GetDesktopWindow() ))) return TRUE;
3288 /* Now call the callback function for every window */
3290 for (i = 0; list[i]; i++)
3292 /* Make sure that the window still exists */
3293 if (!IsWindow( list[i] )) continue;
3294 if (!(ret = lpEnumFunc( list[i], lParam ))) break;
3296 HeapFree( GetProcessHeap(), 0, list );
3297 return ret;
3301 /**********************************************************************
3302 * EnumThreadWindows (USER32.@)
3304 BOOL WINAPI EnumThreadWindows( DWORD id, WNDENUMPROC func, LPARAM lParam )
3306 HWND *list;
3307 int i;
3308 BOOL ret = TRUE;
3310 USER_CheckNotLock();
3312 if (!(list = list_window_children( 0, GetDesktopWindow(), NULL, id ))) return TRUE;
3314 /* Now call the callback function for every window */
3316 for (i = 0; list[i]; i++)
3317 if (!(ret = func( list[i], lParam ))) break;
3318 HeapFree( GetProcessHeap(), 0, list );
3319 return ret;
3323 /***********************************************************************
3324 * EnumDesktopWindows (USER32.@)
3326 BOOL WINAPI EnumDesktopWindows( HDESK desktop, WNDENUMPROC func, LPARAM lparam )
3328 HWND *list;
3329 int i;
3331 USER_CheckNotLock();
3333 if (!(list = list_window_children( desktop, 0, NULL, 0 ))) return TRUE;
3335 for (i = 0; list[i]; i++)
3336 if (!func( list[i], lparam )) break;
3337 HeapFree( GetProcessHeap(), 0, list );
3338 return TRUE;
3342 /**********************************************************************
3343 * WIN_EnumChildWindows
3345 * Helper function for EnumChildWindows().
3347 static BOOL WIN_EnumChildWindows( HWND *list, WNDENUMPROC func, LPARAM lParam )
3349 HWND *childList;
3350 BOOL ret = FALSE;
3352 for ( ; *list; list++)
3354 /* Make sure that the window still exists */
3355 if (!IsWindow( *list )) continue;
3356 /* Build children list first */
3357 childList = WIN_ListChildren( *list );
3359 ret = func( *list, lParam );
3361 if (childList)
3363 if (ret) ret = WIN_EnumChildWindows( childList, func, lParam );
3364 HeapFree( GetProcessHeap(), 0, childList );
3366 if (!ret) return FALSE;
3368 return TRUE;
3372 /**********************************************************************
3373 * EnumChildWindows (USER32.@)
3375 BOOL WINAPI EnumChildWindows( HWND parent, WNDENUMPROC func, LPARAM lParam )
3377 HWND *list;
3378 BOOL ret;
3380 USER_CheckNotLock();
3382 if (!(list = WIN_ListChildren( parent ))) return FALSE;
3383 ret = WIN_EnumChildWindows( list, func, lParam );
3384 HeapFree( GetProcessHeap(), 0, list );
3385 return ret;
3389 /*******************************************************************
3390 * AnyPopup (USER32.@)
3392 BOOL WINAPI AnyPopup(void)
3394 int i;
3395 BOOL retvalue;
3396 HWND *list = WIN_ListChildren( GetDesktopWindow() );
3398 if (!list) return FALSE;
3399 for (i = 0; list[i]; i++)
3401 if (IsWindowVisible( list[i] ) && GetWindow( list[i], GW_OWNER )) break;
3403 retvalue = (list[i] != 0);
3404 HeapFree( GetProcessHeap(), 0, list );
3405 return retvalue;
3409 /*******************************************************************
3410 * FlashWindow (USER32.@)
3412 BOOL WINAPI FlashWindow( HWND hWnd, BOOL bInvert )
3414 WND *wndPtr;
3416 TRACE("%p\n", hWnd);
3418 if (IsIconic( hWnd ))
3420 RedrawWindow( hWnd, 0, 0, RDW_INVALIDATE | RDW_ERASE | RDW_UPDATENOW | RDW_FRAME );
3422 wndPtr = WIN_GetPtr(hWnd);
3423 if (!wndPtr || wndPtr == WND_OTHER_PROCESS || wndPtr == WND_DESKTOP) return FALSE;
3424 if (bInvert && !(wndPtr->flags & WIN_NCACTIVATED))
3426 wndPtr->flags |= WIN_NCACTIVATED;
3428 else
3430 wndPtr->flags &= ~WIN_NCACTIVATED;
3432 WIN_ReleasePtr( wndPtr );
3433 return TRUE;
3435 else
3437 WPARAM wparam;
3439 wndPtr = WIN_GetPtr(hWnd);
3440 if (!wndPtr || wndPtr == WND_OTHER_PROCESS || wndPtr == WND_DESKTOP) return FALSE;
3441 hWnd = wndPtr->obj.handle; /* make it a full handle */
3443 if (bInvert) wparam = !(wndPtr->flags & WIN_NCACTIVATED);
3444 else wparam = (hWnd == GetForegroundWindow());
3446 WIN_ReleasePtr( wndPtr );
3447 SendMessageW( hWnd, WM_NCACTIVATE, wparam, 0 );
3448 return wparam;
3452 /*******************************************************************
3453 * FlashWindowEx (USER32.@)
3455 BOOL WINAPI FlashWindowEx( PFLASHWINFO pfwi )
3457 FIXME("%p\n", pfwi);
3458 return TRUE;
3461 /*******************************************************************
3462 * GetWindowContextHelpId (USER32.@)
3464 DWORD WINAPI GetWindowContextHelpId( HWND hwnd )
3466 DWORD retval;
3467 WND *wnd = WIN_GetPtr( hwnd );
3468 if (!wnd || wnd == WND_DESKTOP) return 0;
3469 if (wnd == WND_OTHER_PROCESS)
3471 if (IsWindow( hwnd )) FIXME( "not supported on other process window %p\n", hwnd );
3472 return 0;
3474 retval = wnd->helpContext;
3475 WIN_ReleasePtr( wnd );
3476 return retval;
3480 /*******************************************************************
3481 * SetWindowContextHelpId (USER32.@)
3483 BOOL WINAPI SetWindowContextHelpId( HWND hwnd, DWORD id )
3485 WND *wnd = WIN_GetPtr( hwnd );
3486 if (!wnd || wnd == WND_DESKTOP) return FALSE;
3487 if (wnd == WND_OTHER_PROCESS)
3489 if (IsWindow( hwnd )) FIXME( "not supported on other process window %p\n", hwnd );
3490 return FALSE;
3492 wnd->helpContext = id;
3493 WIN_ReleasePtr( wnd );
3494 return TRUE;
3498 /*******************************************************************
3499 * DragDetect (USER32.@)
3501 BOOL WINAPI DragDetect( HWND hWnd, POINT pt )
3503 MSG msg;
3504 RECT rect;
3505 WORD wDragWidth = GetSystemMetrics(SM_CXDRAG);
3506 WORD wDragHeight= GetSystemMetrics(SM_CYDRAG);
3508 rect.left = pt.x - wDragWidth;
3509 rect.right = pt.x + wDragWidth;
3511 rect.top = pt.y - wDragHeight;
3512 rect.bottom = pt.y + wDragHeight;
3514 SetCapture(hWnd);
3516 while(1)
3518 while (PeekMessageW( &msg, 0, WM_MOUSEFIRST, WM_MOUSELAST, PM_REMOVE ))
3520 if( msg.message == WM_LBUTTONUP )
3522 ReleaseCapture();
3523 return FALSE;
3525 if( msg.message == WM_MOUSEMOVE )
3527 POINT tmp;
3528 tmp.x = (short)LOWORD(msg.lParam);
3529 tmp.y = (short)HIWORD(msg.lParam);
3530 if( !PtInRect( &rect, tmp ))
3532 ReleaseCapture();
3533 return TRUE;
3537 WaitMessage();
3539 return FALSE;
3542 /******************************************************************************
3543 * GetWindowModuleFileNameA (USER32.@)
3545 UINT WINAPI GetWindowModuleFileNameA( HWND hwnd, LPSTR module, UINT size )
3547 WND *win;
3548 HINSTANCE hinst;
3550 TRACE( "%p, %p, %u\n", hwnd, module, size );
3552 win = WIN_GetPtr( hwnd );
3553 if (!win || win == WND_OTHER_PROCESS || win == WND_DESKTOP)
3555 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
3556 return 0;
3558 hinst = win->hInstance;
3559 WIN_ReleasePtr( win );
3561 return GetModuleFileNameA( hinst, module, size );
3564 /******************************************************************************
3565 * GetWindowModuleFileNameW (USER32.@)
3567 UINT WINAPI GetWindowModuleFileNameW( HWND hwnd, LPWSTR module, UINT size )
3569 WND *win;
3570 HINSTANCE hinst;
3572 TRACE( "%p, %p, %u\n", hwnd, module, size );
3574 win = WIN_GetPtr( hwnd );
3575 if (!win || win == WND_OTHER_PROCESS || win == WND_DESKTOP)
3577 SetLastError( ERROR_INVALID_WINDOW_HANDLE );
3578 return 0;
3580 hinst = win->hInstance;
3581 WIN_ReleasePtr( win );
3583 return GetModuleFileNameW( hinst, module, size );
3586 /******************************************************************************
3587 * GetWindowInfo (USER32.@)
3589 * Note: tests show that Windows doesn't check cbSize of the structure.
3591 BOOL WINAPI DECLSPEC_HOTPATCH GetWindowInfo( HWND hwnd, PWINDOWINFO pwi)
3593 if (!pwi) return FALSE;
3594 if (!WIN_GetRectangles( hwnd, COORDS_SCREEN, &pwi->rcWindow, &pwi->rcClient )) return FALSE;
3596 pwi->dwStyle = GetWindowLongW(hwnd, GWL_STYLE);
3597 pwi->dwExStyle = GetWindowLongW(hwnd, GWL_EXSTYLE);
3598 pwi->dwWindowStatus = ((GetActiveWindow() == hwnd) ? WS_ACTIVECAPTION : 0);
3600 pwi->cxWindowBorders = pwi->rcClient.left - pwi->rcWindow.left;
3601 pwi->cyWindowBorders = pwi->rcWindow.bottom - pwi->rcClient.bottom;
3603 pwi->atomWindowType = GetClassLongW( hwnd, GCW_ATOM );
3604 pwi->wCreatorVersion = 0x0400;
3606 return TRUE;
3609 /******************************************************************************
3610 * SwitchDesktop (USER32.@)
3612 * NOTES: Sets the current input or interactive desktop.
3614 BOOL WINAPI SwitchDesktop( HDESK hDesktop)
3616 FIXME("(hwnd %p) stub!\n", hDesktop);
3617 return TRUE;
3621 /***********************************************************************
3622 * __wine_set_pixel_format
3624 BOOL CDECL __wine_set_pixel_format( HWND hwnd, int format )
3626 WND *win = WIN_GetPtr( hwnd );
3628 if (!win || win == WND_DESKTOP || win == WND_OTHER_PROCESS)
3630 WARN( "setting format %d on win %p not supported\n", format, hwnd );
3631 return FALSE;
3633 win->pixel_format = format;
3634 WIN_ReleasePtr( win );
3636 update_window_state( hwnd );
3637 return TRUE;
3641 /*****************************************************************************
3642 * SetLayeredWindowAttributes (USER32.@)
3644 BOOL WINAPI SetLayeredWindowAttributes( HWND hwnd, COLORREF key, BYTE alpha, DWORD flags )
3646 BOOL ret;
3648 TRACE("(%p,%08x,%d,%x): stub!\n", hwnd, key, alpha, flags);
3650 SERVER_START_REQ( set_window_layered_info )
3652 req->handle = wine_server_user_handle( hwnd );
3653 req->color_key = key;
3654 req->alpha = alpha;
3655 req->flags = flags;
3656 ret = !wine_server_call_err( req );
3658 SERVER_END_REQ;
3660 if (ret) USER_Driver->pSetLayeredWindowAttributes( hwnd, key, alpha, flags );
3662 return ret;
3666 /*****************************************************************************
3667 * GetLayeredWindowAttributes (USER32.@)
3669 BOOL WINAPI GetLayeredWindowAttributes( HWND hwnd, COLORREF *key, BYTE *alpha, DWORD *flags )
3671 BOOL ret;
3673 SERVER_START_REQ( get_window_layered_info )
3675 req->handle = wine_server_user_handle( hwnd );
3676 if ((ret = !wine_server_call_err( req )))
3678 if (key) *key = reply->color_key;
3679 if (alpha) *alpha = reply->alpha;
3680 if (flags) *flags = reply->flags;
3683 SERVER_END_REQ;
3685 return ret;
3689 /*****************************************************************************
3690 * UpdateLayeredWindowIndirect (USER32.@)
3692 BOOL WINAPI UpdateLayeredWindowIndirect( HWND hwnd, const UPDATELAYEREDWINDOWINFO *info )
3694 DWORD flags = SWP_NOSIZE | SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOREDRAW;
3695 RECT window_rect, client_rect;
3696 SIZE offset;
3698 if (!info ||
3699 info->cbSize != sizeof(*info) ||
3700 info->dwFlags & ~(ULW_COLORKEY | ULW_ALPHA | ULW_OPAQUE | ULW_EX_NORESIZE) ||
3701 !(GetWindowLongW( hwnd, GWL_EXSTYLE ) & WS_EX_LAYERED) ||
3702 GetLayeredWindowAttributes( hwnd, NULL, NULL, NULL ))
3704 SetLastError( ERROR_INVALID_PARAMETER );
3705 return FALSE;
3708 WIN_GetRectangles( hwnd, COORDS_PARENT, &window_rect, &client_rect );
3710 if (info->pptDst)
3712 offset.cx = info->pptDst->x - window_rect.left;
3713 offset.cy = info->pptDst->y - window_rect.top;
3714 OffsetRect( &client_rect, offset.cx, offset.cy );
3715 OffsetRect( &window_rect, offset.cx, offset.cy );
3716 flags &= ~SWP_NOMOVE;
3718 if (info->psize)
3720 offset.cx = info->psize->cx - (window_rect.right - window_rect.left);
3721 offset.cy = info->psize->cy - (window_rect.bottom - window_rect.top);
3722 if (info->psize->cx <= 0 || info->psize->cy <= 0)
3724 SetLastError( ERROR_INVALID_PARAMETER );
3725 return FALSE;
3727 if ((info->dwFlags & ULW_EX_NORESIZE) && (offset.cx || offset.cy))
3729 SetLastError( ERROR_INCORRECT_SIZE );
3730 return FALSE;
3732 client_rect.right += offset.cx;
3733 client_rect.bottom += offset.cy;
3734 window_rect.right += offset.cx;
3735 window_rect.bottom += offset.cy;
3736 flags &= ~SWP_NOSIZE;
3739 TRACE( "window %p win %s client %s\n", hwnd,
3740 wine_dbgstr_rect(&window_rect), wine_dbgstr_rect(&client_rect) );
3742 if (!USER_Driver->pUpdateLayeredWindow( hwnd, info, &window_rect )) return FALSE;
3744 set_window_pos( hwnd, 0, flags, &window_rect, &client_rect, NULL );
3745 return TRUE;
3749 /*****************************************************************************
3750 * UpdateLayeredWindow (USER32.@)
3752 BOOL WINAPI UpdateLayeredWindow( HWND hwnd, HDC hdcDst, POINT *pptDst, SIZE *psize,
3753 HDC hdcSrc, POINT *pptSrc, COLORREF crKey, BLENDFUNCTION *pblend,
3754 DWORD flags)
3756 UPDATELAYEREDWINDOWINFO info;
3758 if (flags & ULW_EX_NORESIZE) /* only valid for UpdateLayeredWindowIndirect */
3760 SetLastError( ERROR_INVALID_PARAMETER );
3761 return FALSE;
3763 info.cbSize = sizeof(info);
3764 info.hdcDst = hdcDst;
3765 info.pptDst = pptDst;
3766 info.psize = psize;
3767 info.hdcSrc = hdcSrc;
3768 info.pptSrc = pptSrc;
3769 info.crKey = crKey;
3770 info.pblend = pblend;
3771 info.dwFlags = flags;
3772 info.prcDirty = NULL;
3773 return UpdateLayeredWindowIndirect( hwnd, &info );
3777 /******************************************************************************
3778 * GetProcessDefaultLayout [USER32.@]
3780 * Gets the default layout for parentless windows.
3782 BOOL WINAPI GetProcessDefaultLayout( DWORD *layout )
3784 if (!layout)
3786 SetLastError( ERROR_NOACCESS );
3787 return FALSE;
3789 if (process_layout == ~0u)
3791 static const WCHAR translationW[] = { '\\','V','a','r','F','i','l','e','I','n','f','o',
3792 '\\','T','r','a','n','s','l','a','t','i','o','n', 0 };
3793 static const WCHAR filedescW[] = { '\\','S','t','r','i','n','g','F','i','l','e','I','n','f','o',
3794 '\\','%','0','4','x','%','0','4','x',
3795 '\\','F','i','l','e','D','e','s','c','r','i','p','t','i','o','n',0 };
3796 WCHAR *str, buffer[MAX_PATH];
3797 DWORD i, len, version_layout = 0;
3798 DWORD user_lang = GetUserDefaultLangID();
3799 DWORD *languages;
3800 void *data = NULL;
3802 GetModuleFileNameW( 0, buffer, MAX_PATH );
3803 if (!(len = GetFileVersionInfoSizeW( buffer, NULL ))) goto done;
3804 if (!(data = HeapAlloc( GetProcessHeap(), 0, len ))) goto done;
3805 if (!GetFileVersionInfoW( buffer, 0, len, data )) goto done;
3806 if (!VerQueryValueW( data, translationW, (void **)&languages, &len ) || !len) goto done;
3808 len /= sizeof(DWORD);
3809 for (i = 0; i < len; i++) if (LOWORD(languages[i]) == user_lang) break;
3810 if (i == len) /* try neutral language */
3811 for (i = 0; i < len; i++)
3812 if (LOWORD(languages[i]) == MAKELANGID( PRIMARYLANGID(user_lang), SUBLANG_NEUTRAL )) break;
3813 if (i == len) i = 0; /* default to the first one */
3815 sprintfW( buffer, filedescW, LOWORD(languages[i]), HIWORD(languages[i]) );
3816 if (!VerQueryValueW( data, buffer, (void **)&str, &len )) goto done;
3817 TRACE( "found description %s\n", debugstr_w( str ));
3818 if (str[0] == 0x200e && str[1] == 0x200e) version_layout = LAYOUT_RTL;
3820 done:
3821 HeapFree( GetProcessHeap(), 0, data );
3822 process_layout = version_layout;
3824 *layout = process_layout;
3825 return TRUE;
3829 /******************************************************************************
3830 * SetProcessDefaultLayout [USER32.@]
3832 * Sets the default layout for parentless windows.
3834 BOOL WINAPI SetProcessDefaultLayout( DWORD layout )
3836 process_layout = layout;
3837 return TRUE;
3841 /* 64bit versions */
3843 #ifdef GetWindowLongPtrW
3844 #undef GetWindowLongPtrW
3845 #endif
3847 #ifdef GetWindowLongPtrA
3848 #undef GetWindowLongPtrA
3849 #endif
3851 #ifdef SetWindowLongPtrW
3852 #undef SetWindowLongPtrW
3853 #endif
3855 #ifdef SetWindowLongPtrA
3856 #undef SetWindowLongPtrA
3857 #endif
3859 /*****************************************************************************
3860 * GetWindowLongPtrW (USER32.@)
3862 LONG_PTR WINAPI GetWindowLongPtrW( HWND hwnd, INT offset )
3864 return WIN_GetWindowLong( hwnd, offset, sizeof(LONG_PTR), TRUE );
3867 /*****************************************************************************
3868 * GetWindowLongPtrA (USER32.@)
3870 LONG_PTR WINAPI GetWindowLongPtrA( HWND hwnd, INT offset )
3872 return WIN_GetWindowLong( hwnd, offset, sizeof(LONG_PTR), FALSE );
3875 /*****************************************************************************
3876 * SetWindowLongPtrW (USER32.@)
3878 LONG_PTR WINAPI SetWindowLongPtrW( HWND hwnd, INT offset, LONG_PTR newval )
3880 return WIN_SetWindowLong( hwnd, offset, sizeof(LONG_PTR), newval, TRUE );
3883 /*****************************************************************************
3884 * SetWindowLongPtrA (USER32.@)
3886 LONG_PTR WINAPI SetWindowLongPtrA( HWND hwnd, INT offset, LONG_PTR newval )
3888 return WIN_SetWindowLong( hwnd, offset, sizeof(LONG_PTR), newval, FALSE );
3891 /*****************************************************************************
3892 * RegisterTouchWindow (USER32.@)
3894 BOOL WINAPI RegisterTouchWindow(HWND hwnd, ULONG flags)
3896 FIXME("(%p %08x): stub\n", hwnd, flags);
3897 SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
3898 return FALSE;