1 /* GDK - The GIMP Drawing Kit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3 * Copyright (C) 1998-1999 Tor Lillqvist
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 * Boston, MA 02111-1307, USA.
22 * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
23 * file for a list of people on the GTK+ Team. See the ChangeLog
24 * files for a list of changes. These files are distributed with
25 * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
30 #include "gdk.h" /* gdk_rectangle_intersect */
31 #include "gdkevents.h"
32 #include "gdkpixmap.h"
33 #include "gdkwindow.h"
34 #include "gdkinternals.h"
35 #include "gdkprivate-win32.h"
36 #include "gdkinput-win32.h"
38 static gboolean gdk_window_gravity_works (void);
39 static void gdk_window_set_static_win_gravity (GdkWindow *window,
42 static GdkColormap* gdk_window_impl_win32_get_colormap (GdkDrawable *drawable);
43 static void gdk_window_impl_win32_set_colormap (GdkDrawable *drawable,
45 static void gdk_window_impl_win32_get_size (GdkDrawable *drawable,
48 static GdkRegion* gdk_window_impl_win32_get_visible_region (GdkDrawable *drawable);
49 static void gdk_window_impl_win32_init (GdkWindowImplWin32 *window);
50 static void gdk_window_impl_win32_class_init (GdkWindowImplWin32Class *klass);
51 static void gdk_window_impl_win32_finalize (GObject *object);
53 static gpointer parent_class = NULL;
56 gdk_window_impl_win32_get_type (void)
58 static GType object_type = 0;
62 static const GTypeInfo object_info =
64 sizeof (GdkWindowImplWin32Class),
66 (GBaseFinalizeFunc) NULL,
67 (GClassInitFunc) gdk_window_impl_win32_class_init,
68 NULL, /* class_finalize */
69 NULL, /* class_data */
70 sizeof (GdkWindowImplWin32),
72 (GInstanceInitFunc) gdk_window_impl_win32_init,
75 object_type = g_type_register_static (GDK_TYPE_DRAWABLE_IMPL_WIN32,
84 _gdk_window_impl_get_type (void)
86 return gdk_window_impl_win32_get_type ();
90 gdk_window_impl_win32_init (GdkWindowImplWin32 *impl)
98 impl->extension_events_selected = FALSE;
99 impl->input_locale = GetKeyboardLayout (0);
100 TranslateCharsetInfo ((DWORD FAR *) GetACP (), &impl->charset_info,
105 gdk_window_impl_win32_class_init (GdkWindowImplWin32Class *klass)
107 GObjectClass *object_class = G_OBJECT_CLASS (klass);
108 GdkDrawableClass *drawable_class = GDK_DRAWABLE_CLASS (klass);
110 parent_class = g_type_class_peek_parent (klass);
112 object_class->finalize = gdk_window_impl_win32_finalize;
114 drawable_class->set_colormap = gdk_window_impl_win32_set_colormap;
115 drawable_class->get_colormap = gdk_window_impl_win32_get_colormap;
116 drawable_class->get_size = gdk_window_impl_win32_get_size;
118 /* Visible and clip regions are the same */
119 drawable_class->get_clip_region = gdk_window_impl_win32_get_visible_region;
120 drawable_class->get_visible_region = gdk_window_impl_win32_get_visible_region;
124 gdk_window_impl_win32_finalize (GObject *object)
126 GdkWindowObject *wrapper;
127 GdkDrawableImplWin32 *draw_impl;
128 GdkWindowImplWin32 *window_impl;
130 g_return_if_fail (GDK_IS_WINDOW_IMPL_WIN32 (object));
132 draw_impl = GDK_DRAWABLE_IMPL_WIN32 (object);
133 window_impl = GDK_WINDOW_IMPL_WIN32 (object);
135 wrapper = (GdkWindowObject*) draw_impl->wrapper;
137 if (!GDK_WINDOW_DESTROYED (wrapper))
139 gdk_win32_handle_table_remove (draw_impl->handle);
142 if (window_impl->hcursor != NULL)
144 if (!DestroyCursor (window_impl->hcursor))
145 WIN32_GDI_FAILED("DestroyCursor");
146 window_impl->hcursor = NULL;
149 G_OBJECT_CLASS (parent_class)->finalize (object);
153 gdk_window_impl_win32_get_colormap (GdkDrawable *drawable)
155 GdkDrawableImplWin32 *drawable_impl;
156 GdkWindowImplWin32 *window_impl;
158 g_return_val_if_fail (GDK_IS_WINDOW_IMPL_WIN32 (drawable), NULL);
160 drawable_impl = GDK_DRAWABLE_IMPL_WIN32 (drawable);
161 window_impl = GDK_WINDOW_IMPL_WIN32 (drawable);
163 if (!((GdkWindowObject *) drawable_impl->wrapper)->input_only &&
164 drawable_impl->colormap == NULL)
166 drawable_impl->colormap = gdk_colormap_get_system ();
167 gdk_colormap_ref (drawable_impl->colormap);
170 return drawable_impl->colormap;
174 gdk_window_impl_win32_set_colormap (GdkDrawable *drawable,
177 GdkWindowImplWin32 *impl;
178 GdkDrawableImplWin32 *draw_impl;
180 g_return_if_fail (GDK_IS_WINDOW_IMPL_WIN32 (drawable));
181 g_return_if_fail (gdk_colormap_get_visual (cmap) != gdk_drawable_get_visual (drawable));
183 impl = GDK_WINDOW_IMPL_WIN32 (drawable);
184 draw_impl = GDK_DRAWABLE_IMPL_WIN32 (drawable);
186 GDK_DRAWABLE_GET_CLASS (draw_impl)->set_colormap (drawable, cmap);
189 g_print("gdk_window_impl_win32_set_colormap: XXX\n");
193 gdk_window_impl_win32_get_size (GdkDrawable *drawable,
197 g_return_if_fail (GDK_IS_WINDOW_IMPL_WIN32 (drawable));
200 *width = GDK_WINDOW_IMPL_WIN32 (drawable)->width;
202 *height = GDK_WINDOW_IMPL_WIN32 (drawable)->height;
206 gdk_window_impl_win32_get_visible_region (GdkDrawable *drawable)
208 GdkWindowImplWin32 *impl = GDK_WINDOW_IMPL_WIN32 (drawable);
209 GdkRectangle result_rect;
213 result_rect.width = impl->width;
214 result_rect.height = impl->height;
216 gdk_rectangle_intersect (&result_rect, &impl->position_info.clip_rect, &result_rect);
218 return gdk_region_rectangle (&result_rect);
222 _gdk_windowing_window_init (void)
224 GdkWindowObject *private;
225 GdkWindowImplWin32 *impl;
226 GdkDrawableImplWin32 *draw_impl;
231 g_assert (gdk_parent_root == NULL);
233 SystemParametersInfo(SPI_GETWORKAREA, 0, &rect, 0);
234 width = rect.right - rect.left;
235 height = rect.bottom - rect.top;
237 gdk_parent_root = g_object_new (GDK_TYPE_WINDOW, NULL);
238 private = (GdkWindowObject *)gdk_parent_root;
239 impl = GDK_WINDOW_IMPL_WIN32 (private->impl);
240 draw_impl = GDK_DRAWABLE_IMPL_WIN32 (private->impl);
242 draw_impl->handle = gdk_root_window;
243 draw_impl->wrapper = GDK_DRAWABLE (private);
245 private->window_type = GDK_WINDOW_ROOT;
246 private->depth = gdk_visual_get_system ()->depth;
248 impl->height = height;
250 gdk_win32_handle_table_insert (&gdk_root_window, gdk_parent_root);
253 /* The Win API function AdjustWindowRect may return negative values
254 * resulting in obscured title bars. This helper function is coreccting it.
257 SafeAdjustWindowRectEx (RECT* lpRect,
262 if (!AdjustWindowRectEx(lpRect, dwStyle, bMenu, dwExStyle))
264 WIN32_API_FAILED ("AdjustWindowRectEx");
267 if (lpRect->left < 0)
269 lpRect->right -= lpRect->left;
274 lpRect->bottom -= lpRect->top;
281 * is a wrapper function for RegisterWindowClassEx.
282 * It creates at least one unique class for every
283 * GdkWindowType. If support for single window-specific icons
284 * is ever needed (e.g Dialog specific), every such window should
288 RegisterGdkClass (GdkWindowType wtype)
290 static ATOM klassTOPLEVEL = 0;
291 static ATOM klassDIALOG = 0;
292 static ATOM klassCHILD = 0;
293 static ATOM klassTEMP = 0;
294 static HICON hAppIcon = NULL;
295 static WNDCLASSEX wcl;
298 wcl.cbSize = sizeof(WNDCLASSEX);
299 wcl.style = 0; /* DON'T set CS_<H,V>REDRAW. It causes total redraw
300 * on WM_SIZE and WM_MOVE. Flicker, Performance!
302 wcl.lpfnWndProc = gdk_window_procedure;
305 wcl.hInstance = gdk_app_hmodule;
307 /* initialize once! */
310 gchar sLoc [_MAX_PATH+1];
312 if (0 != GetModuleFileName (gdk_app_hmodule, sLoc, _MAX_PATH))
314 hAppIcon = ExtractIcon (gdk_app_hmodule, sLoc, 0);
317 char *gdklibname = g_strdup_printf ("gdk-%s.dll", GDK_VERSION);
319 hAppIcon = ExtractIcon (gdk_app_hmodule, gdklibname, 0);
324 hAppIcon = LoadIcon (NULL, IDI_APPLICATION);
328 wcl.lpszMenuName = NULL;
331 /* initialize once per class */
332 #define ONCE_PER_CLASS() \
333 wcl.hIcon = CopyIcon (hAppIcon); \
334 wcl.hIconSm = CopyIcon (hAppIcon); \
335 wcl.hbrBackground = CreateSolidBrush (RGB (0,0,0)); \
336 wcl.hCursor = LoadCursor (NULL, IDC_ARROW);
340 case GDK_WINDOW_TOPLEVEL:
341 if (0 == klassTOPLEVEL)
343 wcl.lpszClassName = "gdkWindowToplevel";
346 klassTOPLEVEL = RegisterClassEx (&wcl);
348 klass = klassTOPLEVEL;
351 case GDK_WINDOW_CHILD:
354 wcl.lpszClassName = "gdkWindowChild";
356 wcl.style |= CS_PARENTDC; /* MSDN: ... enhances system performance. */
358 klassCHILD = RegisterClassEx (&wcl);
363 case GDK_WINDOW_DIALOG:
364 if (0 == klassDIALOG)
366 wcl.lpszClassName = "gdkWindowDialog";
367 wcl.style |= CS_SAVEBITS;
369 klassDIALOG = RegisterClassEx (&wcl);
374 case GDK_WINDOW_TEMP:
377 wcl.lpszClassName = "gdkWindowTemp";
378 wcl.style |= CS_SAVEBITS;
380 klassTEMP = RegisterClassEx (&wcl);
386 g_assert_not_reached ();
392 WIN32_API_FAILED ("RegisterClassEx");
393 g_error ("That is a fatal error");
399 gdk_window_new (GdkWindow *parent,
400 GdkWindowAttr *attributes,
401 gint attributes_mask)
404 GdkWindowObject *private;
405 GdkWindowObject *parent_private;
406 GdkWindowImplWin32 *impl;
407 GdkDrawableImplWin32 *draw_impl;
413 DWORD dwStyle, dwExStyle;
422 g_return_val_if_fail (attributes != NULL, NULL);
425 parent = gdk_parent_root;
427 g_return_val_if_fail (GDK_IS_WINDOW (parent), NULL);
430 g_print ("gdk_window_new: %s\n",
431 (attributes->window_type == GDK_WINDOW_TOPLEVEL ? "TOPLEVEL" :
432 (attributes->window_type == GDK_WINDOW_CHILD ? "CHILD" :
433 (attributes->window_type == GDK_WINDOW_DIALOG ? "DIALOG" :
434 (attributes->window_type == GDK_WINDOW_TEMP ? "TEMP" :
437 parent_private = (GdkWindowObject*) parent;
438 if (GDK_WINDOW_DESTROYED (parent))
441 hparent = GDK_WINDOW_HWND (parent);
443 window = g_object_new (GDK_TYPE_WINDOW, NULL);
444 private = (GdkWindowObject *)window;
445 impl = GDK_WINDOW_IMPL_WIN32 (private->impl);
446 draw_impl = GDK_DRAWABLE_IMPL_WIN32 (private->impl);
447 draw_impl->wrapper = GDK_DRAWABLE (window);
449 private->parent = (GdkWindowObject *)parent;
451 if (attributes_mask & GDK_WA_X)
456 if (attributes_mask & GDK_WA_Y)
458 else if (attributes_mask & GDK_WA_X)
459 y = 100; /* ??? We must put it somewhere... */
461 y = 0; /* x is CW_USEDEFAULT, y doesn't matter then */
465 impl->width = (attributes->width > 1) ? (attributes->width) : (1);
466 impl->height = (attributes->height > 1) ? (attributes->height) : (1);
467 impl->extension_events_selected = FALSE;
468 private->window_type = attributes->window_type;
470 _gdk_window_init_position (GDK_WINDOW (private));
471 if (impl->position_info.big)
472 private->guffaw_gravity = TRUE;
474 if (attributes_mask & GDK_WA_VISUAL)
475 visual = attributes->visual;
477 visual = gdk_visual_get_system ();
478 xvisual = ((GdkVisualPrivate*) visual)->xvisual;
480 if (attributes_mask & GDK_WA_TITLE)
481 title = attributes->title;
483 title = g_get_prgname ();
484 if (!title || !*title)
485 title = "GDK client window";
487 impl->event_mask = GDK_STRUCTURE_MASK | attributes->event_mask;
489 if (parent_private && parent_private->guffaw_gravity)
494 if (attributes->wclass == GDK_INPUT_OUTPUT)
498 private->input_only = FALSE;
499 private->depth = visual->depth;
501 if (attributes_mask & GDK_WA_COLORMAP)
503 draw_impl->colormap = attributes->colormap;
504 gdk_colormap_ref (attributes->colormap);
508 if ((((GdkVisualPrivate*)gdk_visual_get_system ())->xvisual) == xvisual)
510 draw_impl->colormap = gdk_colormap_get_system ();
511 gdk_colormap_ref (draw_impl->colormap);
512 GDK_NOTE (MISC, g_print ("...using system colormap %p\n",
513 draw_impl->colormap));
517 draw_impl->colormap = gdk_colormap_new (visual, FALSE);
518 GDK_NOTE (MISC, g_print ("...using new colormap %p\n",
519 draw_impl->colormap));
525 dwExStyle = WS_EX_TRANSPARENT;
527 private->input_only = TRUE;
528 draw_impl->colormap = NULL;
529 GDK_NOTE (MISC, g_print ("...GDK_INPUT_ONLY, NULL colormap\n"));
533 parent_private->children = g_list_prepend (parent_private->children, window);
535 switch (private->window_type)
537 case GDK_WINDOW_TOPLEVEL:
538 dwStyle = WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN;
539 hparent = gdk_root_window;
542 case GDK_WINDOW_CHILD:
543 dwStyle = WS_CHILDWINDOW | WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
546 case GDK_WINDOW_DIALOG:
547 dwStyle = WS_OVERLAPPED | WS_MINIMIZEBOX | WS_SYSMENU | WS_CAPTION | WS_THICKFRAME | WS_CLIPCHILDREN;
549 dwExStyle |= WS_EX_TOPMOST; /* //HB: want this? */
551 hparent = gdk_root_window;
554 case GDK_WINDOW_TEMP:
555 dwStyle = WS_POPUP | WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
556 dwExStyle |= WS_EX_TOOLWINDOW;
559 case GDK_WINDOW_ROOT:
560 g_error ("cannot make windows of type GDK_WINDOW_ROOT");
564 klass = RegisterGdkClass (private->window_type);
566 if (private->window_type != GDK_WINDOW_CHILD)
568 if (x == CW_USEDEFAULT)
579 rect.right = rect.left + impl->width;
580 rect.bottom = rect.top + impl->height;
582 SafeAdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
584 if (x != CW_USEDEFAULT)
589 width = rect.right - rect.left;
590 height = rect.bottom - rect.top;
595 height = impl->height;
598 mbtitle = g_locale_from_utf8 (title, -1, NULL, NULL, NULL);
600 #ifdef WITHOUT_WM_CREATE
601 draw_impl->handle = CreateWindowEx (dwExStyle,
602 MAKEINTRESOURCE(klass),
614 CreateWindowEx (dwExStyle,
615 MAKEINTRESOURCE(klass),
624 if (GDK_WINDOW_HWND (window) != hwndNew)
626 g_warning("gdk_window_new: gdk_event_translate::WM_CREATE (%#x, %#x) HWND mismatch.",
627 (guint) GDK_WINDOW_HWND (window),
630 /* HB: IHMO due to a race condition the handle was increased by
631 * one, which causes much trouble. Because I can't find the
632 * real bug, try to workaround it ...
633 * To reproduce: compile with MSVC 5, DEBUG=1
636 gdk_win32_handle_table_remove (GDK_WINDOW_HWND (window));
637 GDK_WINDOW_HWND (window) = hwndNew;
638 gdk_win32_handle_table_insert (&GDK_WINDOW_HWND (window), window);
640 /* the old behaviour, but with warning */
641 GDK_WINDOW_HWND (window) = hwndNew;
646 gdk_drawable_ref (window);
647 gdk_win32_handle_table_insert (&GDK_WINDOW_HWND (window), window);
651 g_print ("... \"%s\" %dx%d@+%d+%d %#x = %#x\n"
652 "... locale %#x codepage %d\n",
654 width, height, (x == CW_USEDEFAULT ? -9999 : x), y,
656 (guint) GDK_WINDOW_HWND (window),
657 (guint) impl->input_locale,
658 (guint) impl->charset_info.ciACP));
662 if (draw_impl->handle == NULL)
664 WIN32_API_FAILED ("CreateWindowEx");
665 g_object_unref ((GObject *) window);
669 #ifdef WITHOUT_WM_CREATE
670 gdk_drawable_ref (window);
671 gdk_win32_handle_table_insert (&GDK_WINDOW_HWND (window), window);
674 gdk_window_set_cursor (window, ((attributes_mask & GDK_WA_CURSOR) ?
675 (attributes->cursor) :
682 gdk_window_foreign_new (GdkNativeWindow anid)
685 GdkWindowObject *private;
686 GdkWindowObject *parent_private;
687 GdkWindowImplWin32 *impl;
688 GdkDrawableImplWin32 *draw_impl;
694 window = g_object_new (GDK_TYPE_WINDOW, NULL);
695 private = (GdkWindowObject *)window;
696 impl = GDK_WINDOW_IMPL_WIN32 (private->impl);
697 draw_impl = GDK_DRAWABLE_IMPL_WIN32 (private->impl);
698 draw_impl->wrapper = GDK_DRAWABLE (window);
699 parent = GetParent ((HWND)anid);
701 private->parent = gdk_win32_handle_table_lookup ((GdkNativeWindow) parent);
703 parent_private = (GdkWindowObject *)private->parent;
706 parent_private->children = g_list_prepend (parent_private->children, window);
708 draw_impl->handle = (HWND) anid;
709 GetClientRect ((HWND) anid, &rect);
711 point.y = rect.right;
712 ClientToScreen ((HWND) anid, &point);
713 if (parent != gdk_root_window)
714 ScreenToClient (parent, &point);
715 private->x = point.x;
716 private->y = point.y;
717 impl->width = rect.right - rect.left;
718 impl->height = rect.bottom - rect.top;
719 private->window_type = GDK_WINDOW_FOREIGN;
720 private->destroyed = FALSE;
721 if (IsWindowVisible ((HWND) anid))
722 private->state &= (~GDK_WINDOW_STATE_WITHDRAWN);
724 private->state |= GDK_WINDOW_STATE_WITHDRAWN;
725 private->depth = gdk_visual_get_system ()->depth;
727 gdk_drawable_ref (window);
728 gdk_win32_handle_table_insert (&GDK_WINDOW_HWND (window), window);
734 _gdk_windowing_window_destroy (GdkWindow *window,
736 gboolean foreign_destroy)
738 GdkWindowObject *private = (GdkWindowObject *)window;
740 g_return_if_fail (GDK_IS_WINDOW (window));
742 GDK_NOTE (MISC, g_print ("_gdk_windowing_window_destroy %#x\n",
743 (guint) GDK_WINDOW_HWND (window)));
745 if (private->extension_events != 0)
746 gdk_input_window_destroy (window);
748 if (private->window_type == GDK_WINDOW_FOREIGN)
750 if (!foreign_destroy && (private->parent != NULL))
752 /* It's somebody else's window, but in our hierarchy,
753 * so reparent it to the root window, and then call
754 * DestroyWindow() on it.
756 gdk_window_hide (window);
757 gdk_window_reparent (window, NULL, 0, 0);
759 /* Is this too drastic? Many (most?) applications
760 * quit if any window receives WM_QUIT I think.
761 * OTOH, I don't think foreign windows are much
762 * used, so the question is maybe academic.
764 PostMessage (GDK_WINDOW_HWND (window), WM_QUIT, 0, 0);
767 else if (!recursing && !foreign_destroy)
768 DestroyWindow (GDK_WINDOW_HWND (window));
771 /* This function is called when the window really gone.
774 gdk_window_destroy_notify (GdkWindow *window)
776 g_return_if_fail (window != NULL);
777 g_return_if_fail (GDK_IS_WINDOW (window));
780 g_print ("gdk_window_destroy_notify: %#x %s\n",
781 (guint) GDK_WINDOW_HWND (window),
782 (GDK_WINDOW_DESTROYED (window) ? "(destroyed)" : "")));
784 if (!GDK_WINDOW_DESTROYED (window))
786 if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_FOREIGN)
787 g_warning ("window %#x unexpectedly destroyed",
788 (guint) GDK_WINDOW_HWND (window));
790 _gdk_window_destroy (window, TRUE);
793 gdk_win32_handle_table_remove (GDK_WINDOW_HWND (window));
794 gdk_drawable_unref (window);
798 gdk_window_show (GdkWindow *window)
800 GdkWindowObject *private;
802 g_return_if_fail (GDK_IS_WINDOW (window));
804 private = (GdkWindowObject*) window;
805 if (!private->destroyed)
807 GDK_NOTE (MISC, g_print ("gdk_window_show: %#x\n",
808 (guint) GDK_WINDOW_HWND (window)));
810 private->state &= (~GDK_WINDOW_STATE_WITHDRAWN);
811 if (GDK_WINDOW_TYPE (window) == GDK_WINDOW_TEMP)
813 ShowWindow (GDK_WINDOW_HWND (window), SW_SHOWNOACTIVATE);
814 SetWindowPos (GDK_WINDOW_HWND (window), HWND_TOPMOST, 0, 0, 0, 0,
815 SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE);
817 /* Don't put on toolbar */
818 ShowWindow (GDK_WINDOW_HWND (window), SW_HIDE);
823 if (GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE) & WS_EX_TRANSPARENT)
825 SetWindowPos(GDK_WINDOW_HWND (window), HWND_TOP, 0, 0, 0, 0,
826 SWP_SHOWWINDOW | SWP_NOREDRAW | SWP_NOMOVE | SWP_NOSIZE);
830 GdkWindow *parent = private->parent;
832 ShowWindow (GDK_WINDOW_HWND (window), SW_SHOWNORMAL);
833 ShowWindow (GDK_WINDOW_HWND (window), SW_RESTORE);
834 if (parent == gdk_parent_root)
835 SetForegroundWindow (GDK_WINDOW_HWND (window));
836 BringWindowToTop (GDK_WINDOW_HWND (window));
838 ShowOwnedPopups (GDK_WINDOW_HWND (window), TRUE);
846 gdk_window_hide (GdkWindow *window)
848 GdkWindowObject *private;
850 g_return_if_fail (window != NULL);
852 private = (GdkWindowObject*) window;
853 if (!private->destroyed)
855 GDK_NOTE (MISC, g_print ("gdk_window_hide: %#x\n",
856 (guint) GDK_WINDOW_HWND (window)));
858 private->state |= GDK_WINDOW_STATE_WITHDRAWN;
859 if (GDK_WINDOW_TYPE (window) == GDK_WINDOW_TOPLEVEL)
860 ShowOwnedPopups (GDK_WINDOW_HWND (window), FALSE);
862 if (GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE) & WS_EX_TRANSPARENT)
864 SetWindowPos(GDK_WINDOW_HWND (window), HWND_BOTTOM, 0, 0, 0, 0,
865 SWP_HIDEWINDOW | SWP_NOREDRAW | SWP_NOZORDER | SWP_NOMOVE | SWP_NOSIZE);
869 ShowWindow (GDK_WINDOW_HWND (window), SW_HIDE);
875 gdk_window_withdraw (GdkWindow *window)
877 GdkWindowObject *private;
879 g_return_if_fail (window != NULL);
881 private = (GdkWindowObject*) window;
882 if (!private->destroyed)
884 GDK_NOTE (MISC, g_print ("gdk_window_withdraw: %#x\n",
885 (guint) GDK_WINDOW_HWND (window)));
887 gdk_window_hide (window); /* ??? */
892 gdk_window_move (GdkWindow *window,
896 GdkWindowObject *private = (GdkWindowObject *)window;
897 GdkWindowImplWin32 *impl;
899 g_return_if_fail (window != NULL);
900 g_return_if_fail (GDK_IS_WINDOW (window));
902 impl = GDK_WINDOW_IMPL_WIN32 (private->impl);
904 gdk_window_move_resize (window, x, y, impl->width, impl->height);
908 gdk_window_resize (GdkWindow *window,
912 GdkWindowObject *private = (GdkWindowObject*) window;
913 GdkWindowImplWin32 *impl;
916 g_return_if_fail (window != NULL);
917 g_return_if_fail (GDK_IS_WINDOW (window));
924 impl = GDK_WINDOW_IMPL_WIN32 (private->impl);
926 if (!private->destroyed)
928 GDK_NOTE (MISC, g_print ("gdk_window_resize: %#x %dx%d\n",
929 (guint) GDK_WINDOW_HWND (window),
932 if (GDK_WINDOW_TYPE (private) != GDK_WINDOW_CHILD)
941 ClientToScreen (GDK_WINDOW_HWND (window), &pt);
944 rect.right = pt.x + width;
945 rect.bottom = pt.y + height;
947 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
948 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
949 if (!AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle))
950 WIN32_API_FAILED ("AdjustWindowRectEx");
954 width = rect.right - rect.left;
955 height = rect.bottom - rect.top;
962 impl->height = height;
965 private->resize_count += 1;
967 GDK_NOTE (MISC, g_print ("...MoveWindow(%#x,%dx%d@+%d+%d)\n",
968 (guint) GDK_WINDOW_HWND (window),
969 width, height, x, y));
970 if (!MoveWindow (GDK_WINDOW_HWND (window), x, y, width, height, TRUE))
971 WIN32_API_FAILED ("MoveWindow");
976 gdk_window_move_resize (GdkWindow *window,
982 GdkWindowObject *private = (GdkWindowObject*) window;
983 GdkWindowImplWin32 *impl;
985 g_return_if_fail (window != NULL);
986 g_return_if_fail (GDK_IS_WINDOW (window));
993 impl = GDK_WINDOW_IMPL_WIN32 (private->impl);
995 if (!private->destroyed)
1001 GDK_NOTE (MISC, g_print ("gdk_window_move_resize: %#x %dx%d@+%d+%d\n",
1002 (guint) GDK_WINDOW_HWND (window),
1003 width, height, x, y));
1005 if (GDK_WINDOW_TYPE (private) == GDK_WINDOW_CHILD)
1006 _gdk_window_move_resize_child (window, x, y, width, height);
1011 rect.right = x + width;
1012 rect.bottom = y + height;
1014 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1015 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1016 if (!AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle))
1017 WIN32_API_FAILED ("AdjustWindowRectEx");
1019 GDK_NOTE (MISC, g_print ("...MoveWindow(%#x,%ldx%ld@+%ld+%ld)\n",
1020 (guint) GDK_WINDOW_HWND (window),
1021 rect.right - rect.left, rect.bottom - rect.top,
1022 rect.left, rect.top));
1023 if (!MoveWindow (GDK_WINDOW_HWND (window),
1024 rect.left, rect.top,
1025 rect.right - rect.left, rect.bottom - rect.top,
1027 WIN32_API_FAILED ("MoveWindow");
1033 gdk_window_reparent (GdkWindow *window,
1034 GdkWindow *new_parent,
1038 GdkWindowObject *window_private;
1039 GdkWindowObject *parent_private;
1040 GdkWindowObject *old_parent_private;
1041 GdkWindowImplWin32 *impl;
1043 g_return_if_fail (window != NULL);
1046 new_parent = gdk_parent_root;
1048 window_private = (GdkWindowObject*) window;
1049 old_parent_private = (GdkWindowObject *) window_private->parent;
1050 parent_private = (GdkWindowObject*) new_parent;
1051 impl = GDK_WINDOW_IMPL_WIN32 (window_private->impl);
1053 if (!GDK_WINDOW_DESTROYED (window) && !GDK_WINDOW_DESTROYED (new_parent))
1055 GDK_NOTE (MISC, g_print ("gdk_window_reparent: %#x %#x\n",
1056 (guint) GDK_WINDOW_HWND (window),
1057 (guint) GDK_WINDOW_HWND (new_parent)));
1058 if (!SetParent (GDK_WINDOW_HWND (window),
1059 GDK_WINDOW_HWND (new_parent)))
1060 WIN32_API_FAILED ("SetParent");
1062 if (!MoveWindow (GDK_WINDOW_HWND (window),
1063 x, y, impl->width, impl->height, TRUE))
1064 WIN32_API_FAILED ("MoveWindow");
1067 window_private->parent = (GdkWindowObject *)new_parent;
1069 if (old_parent_private)
1070 old_parent_private->children =
1071 g_list_remove (old_parent_private->children, window);
1073 if ((old_parent_private &&
1074 (!old_parent_private->guffaw_gravity != !parent_private->guffaw_gravity)) ||
1075 (!old_parent_private && parent_private->guffaw_gravity))
1076 gdk_window_set_static_win_gravity (window, parent_private->guffaw_gravity);
1078 parent_private->children = g_list_prepend (parent_private->children, window);
1082 _gdk_windowing_window_clear_area (GdkWindow *window,
1088 GdkWindowImplWin32 *impl;
1090 g_return_if_fail (window != NULL);
1091 g_return_if_fail (GDK_IS_WINDOW (window));
1093 impl = GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl);
1095 if (!GDK_WINDOW_DESTROYED (window))
1100 width = impl->width - x;
1102 height = impl->height - y;
1103 GDK_NOTE (MISC, g_print ("_gdk_windowing_window_clear_area: "
1104 "%#x %dx%d@+%d+%d\n",
1105 (guint) GDK_WINDOW_HWND (window),
1106 width, height, x, y));
1107 hdc = GetDC (GDK_WINDOW_HWND (window));
1108 IntersectClipRect (hdc, x, y, x + width + 1, y + height + 1);
1109 SendMessage (GDK_WINDOW_HWND (window), WM_ERASEBKGND, (WPARAM) hdc, 0);
1110 ReleaseDC (GDK_WINDOW_HWND (window), hdc);
1115 _gdk_windowing_window_clear_area_e (GdkWindow *window,
1121 g_return_if_fail (window != NULL);
1122 g_return_if_fail (GDK_IS_WINDOW (window));
1124 if (!GDK_WINDOW_DESTROYED (window))
1128 GDK_NOTE (MISC, g_print ("_gdk_windowing_window_clear_area_e: "
1129 "%#x %dx%d@+%d+%d\n",
1130 (guint) GDK_WINDOW_HWND (window),
1131 width, height, x, y));
1134 rect.right = x + width + 1;
1136 rect.bottom = y + height + 1;
1137 if (!InvalidateRect (GDK_WINDOW_HWND (window), &rect, TRUE))
1138 WIN32_GDI_FAILED ("InvalidateRect");
1139 UpdateWindow (GDK_WINDOW_HWND (window));
1144 gdk_window_raise (GdkWindow *window)
1146 g_return_if_fail (window != NULL);
1147 g_return_if_fail (GDK_IS_WINDOW (window));
1149 if (!GDK_WINDOW_DESTROYED (window))
1151 GDK_NOTE (MISC, g_print ("gdk_window_raise: %#x\n",
1152 (guint) GDK_WINDOW_HWND (window)));
1154 if (!BringWindowToTop (GDK_WINDOW_HWND (window)))
1155 WIN32_API_FAILED ("BringWindowToTop");
1160 gdk_window_lower (GdkWindow *window)
1162 g_return_if_fail (window != NULL);
1163 g_return_if_fail (GDK_IS_WINDOW (window));
1165 if (!GDK_WINDOW_DESTROYED (window))
1167 GDK_NOTE (MISC, g_print ("gdk_window_lower: %#x\n",
1168 (guint) GDK_WINDOW_HWND (window)));
1170 if (!SetWindowPos (GDK_WINDOW_HWND (window), HWND_BOTTOM, 0, 0, 0, 0,
1171 SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE))
1172 WIN32_API_FAILED ("SetWindowPos");
1177 gdk_window_set_hints (GdkWindow *window,
1186 GdkWindowImplWin32 *impl;
1187 WINDOWPLACEMENT size_hints;
1193 g_return_if_fail (window != NULL);
1194 g_return_if_fail (GDK_IS_WINDOW (window));
1196 if (GDK_WINDOW_DESTROYED (window))
1199 impl = GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl);
1201 GDK_NOTE (MISC, g_print ("gdk_window_set_hints: %#x %dx%d..%dx%d @+%d+%d\n",
1202 (guint) GDK_WINDOW_HWND (window),
1203 min_width, min_height, max_width, max_height,
1206 impl->hint_flags = flags;
1207 size_hints.length = sizeof (size_hints);
1211 if (flags & GDK_HINT_POS)
1213 if (!GetWindowPlacement (GDK_WINDOW_HWND (window), &size_hints))
1214 WIN32_API_FAILED ("GetWindowPlacement");
1217 GDK_NOTE (MISC, g_print ("...rcNormalPosition:"
1218 " (%ld,%ld)--(%ld,%ld)\n",
1219 size_hints.rcNormalPosition.left,
1220 size_hints.rcNormalPosition.top,
1221 size_hints.rcNormalPosition.right,
1222 size_hints.rcNormalPosition.bottom));
1223 /* What are the corresponding window coordinates for client
1224 * area coordinates x, y
1228 rect.right = rect.left + 200; /* dummy */
1229 rect.bottom = rect.top + 200;
1230 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1231 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1232 AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
1233 size_hints.flags = 0;
1234 size_hints.showCmd = SW_SHOWNA;
1236 /* Set the normal position hint to that location, with unchanged
1239 diff = size_hints.rcNormalPosition.left - rect.left;
1240 size_hints.rcNormalPosition.left = rect.left;
1241 size_hints.rcNormalPosition.right -= diff;
1242 diff = size_hints.rcNormalPosition.top - rect.top;
1243 size_hints.rcNormalPosition.top = rect.top;
1244 size_hints.rcNormalPosition.bottom -= diff;
1245 GDK_NOTE (MISC, g_print ("...setting: (%ld,%ld)--(%ld,%ld)\n",
1246 size_hints.rcNormalPosition.left,
1247 size_hints.rcNormalPosition.top,
1248 size_hints.rcNormalPosition.right,
1249 size_hints.rcNormalPosition.bottom));
1250 if (!SetWindowPlacement (GDK_WINDOW_HWND (window), &size_hints))
1251 WIN32_API_FAILED ("SetWindowPlacement");
1252 impl->hint_x = rect.left;
1253 impl->hint_y = rect.top;
1257 if (flags & GDK_HINT_MIN_SIZE)
1261 rect.right = min_width;
1262 rect.bottom = min_height;
1263 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1264 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1265 AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
1266 impl->hint_min_width = rect.right - rect.left;
1267 impl->hint_min_height = rect.bottom - rect.top;
1269 /* Also chek if he current size of the window is in bounds. */
1270 GetClientRect (GDK_WINDOW_HWND (window), &rect);
1271 if (rect.right < min_width && rect.bottom < min_height)
1272 gdk_window_resize (window, min_width, min_height);
1273 else if (rect.right < min_width)
1274 gdk_window_resize (window, min_width, rect.bottom);
1275 else if (rect.bottom < min_height)
1276 gdk_window_resize (window, rect.right, min_height);
1279 if (flags & GDK_HINT_MAX_SIZE)
1283 rect.right = max_width;
1284 rect.bottom = max_height;
1285 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1286 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1287 AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
1288 impl->hint_max_width = rect.right - rect.left;
1289 impl->hint_max_height = rect.bottom - rect.top;
1290 /* Again, check if the window is too large currently. */
1291 GetClientRect (GDK_WINDOW_HWND (window), &rect);
1292 if (rect.right > max_width && rect.bottom > max_height)
1293 gdk_window_resize (window, max_width, max_height);
1294 else if (rect.right > max_width)
1295 gdk_window_resize (window, max_width, rect.bottom);
1296 else if (rect.bottom > max_height)
1297 gdk_window_resize (window, rect.right, max_height);
1303 gdk_window_set_geometry_hints (GdkWindow *window,
1304 GdkGeometry *geometry,
1305 GdkWindowHints geom_mask)
1307 GdkWindowImplWin32 *impl;
1308 WINDOWPLACEMENT size_hints;
1313 g_return_if_fail (window != NULL);
1314 g_return_if_fail (GDK_IS_WINDOW (window));
1316 if (GDK_WINDOW_DESTROYED (window))
1319 impl = GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl);
1320 size_hints.length = sizeof (size_hints);
1322 impl->hint_flags = geom_mask;
1324 if (geom_mask & GDK_HINT_POS)
1327 if (geom_mask & GDK_HINT_MIN_SIZE)
1331 rect.right = geometry->min_width;
1332 rect.bottom = geometry->min_height;
1333 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1334 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1335 AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
1336 impl->hint_min_width = rect.right - rect.left;
1337 impl->hint_min_height = rect.bottom - rect.top;
1339 /* Also check if he current size of the window is in bounds */
1340 GetClientRect (GDK_WINDOW_HWND (window), &rect);
1341 if (rect.right < geometry->min_width
1342 && rect.bottom < geometry->min_height)
1343 gdk_window_resize (window, geometry->min_width, geometry->min_height);
1344 else if (rect.right < geometry->min_width)
1345 gdk_window_resize (window, geometry->min_width, rect.bottom);
1346 else if (rect.bottom < geometry->min_height)
1347 gdk_window_resize (window, rect.right, geometry->min_height);
1350 if (geom_mask & GDK_HINT_MAX_SIZE)
1354 rect.right = geometry->max_width;
1355 rect.bottom = geometry->max_height;
1356 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1357 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1358 /* HB: dont' know why AdjustWindowRectEx is called here, ... */
1359 SafeAdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
1360 impl->hint_max_width = rect.right - rect.left;
1361 impl->hint_max_height = rect.bottom - rect.top;
1362 /* ... but negative sizes are always wrong */
1363 if (impl->hint_max_width < 0) impl->hint_max_width = G_MAXSHORT;
1364 if (impl->hint_max_height < 0) impl->hint_max_height = G_MAXSHORT;
1366 /* Again, check if the window is too large currently. */
1367 GetClientRect (GDK_WINDOW_HWND (window), &rect);
1368 if (rect.right > geometry->max_width
1369 && rect.bottom > geometry->max_height)
1370 gdk_window_resize (window, geometry->max_width, geometry->max_height);
1371 else if (rect.right > geometry->max_width)
1372 gdk_window_resize (window, geometry->max_width, rect.bottom);
1373 else if (rect.bottom > geometry->max_height)
1374 gdk_window_resize (window, rect.right, geometry->max_height);
1377 /* I don't know what to do when called with zero base_width and height. */
1378 if (geom_mask & GDK_HINT_BASE_SIZE
1379 && geometry->base_width > 0
1380 && geometry->base_height > 0)
1382 if (!GetWindowPlacement (GDK_WINDOW_HWND (window), &size_hints))
1383 WIN32_API_FAILED ("GetWindowPlacement");
1386 GDK_NOTE (MISC, g_print ("gdk_window_set_geometry_hints:"
1387 " rcNormalPosition: (%ld,%ld)--(%ld,%ld)\n",
1388 size_hints.rcNormalPosition.left,
1389 size_hints.rcNormalPosition.top,
1390 size_hints.rcNormalPosition.right,
1391 size_hints.rcNormalPosition.bottom));
1392 size_hints.rcNormalPosition.right =
1393 size_hints.rcNormalPosition.left + geometry->base_width;
1394 size_hints.rcNormalPosition.bottom =
1395 size_hints.rcNormalPosition.top + geometry->base_height;
1396 GDK_NOTE (MISC, g_print ("...setting: rcNormal: (%ld,%ld)--(%ld,%ld)\n",
1397 size_hints.rcNormalPosition.left,
1398 size_hints.rcNormalPosition.top,
1399 size_hints.rcNormalPosition.right,
1400 size_hints.rcNormalPosition.bottom));
1401 if (!SetWindowPlacement (GDK_WINDOW_HWND (window), &size_hints))
1402 WIN32_API_FAILED ("SetWindowPlacement");
1406 if (geom_mask & GDK_HINT_RESIZE_INC)
1411 if (geom_mask & GDK_HINT_ASPECT)
1418 gdk_window_set_title (GdkWindow *window,
1423 g_return_if_fail (window != NULL);
1424 g_return_if_fail (GDK_IS_WINDOW (window));
1425 g_return_if_fail (title != NULL);
1427 /* Empty window titles not allowed, so set it to just a period. */
1431 GDK_NOTE (MISC, g_print ("gdk_window_set_title: %#x %s\n",
1432 (guint) GDK_WINDOW_HWND (window), title));
1434 if (!GDK_WINDOW_DESTROYED (window))
1436 /* As the title is in UTF-8 we must translate it
1437 * to the system codepage.
1439 mbtitle = g_locale_from_utf8 (title, -1, NULL, NULL, NULL);
1440 if (!SetWindowText (GDK_WINDOW_HWND (window), mbtitle))
1441 WIN32_API_FAILED ("SetWindowText");
1448 gdk_window_set_role (GdkWindow *window,
1451 g_return_if_fail (window != NULL);
1452 g_return_if_fail (GDK_IS_WINDOW (window));
1454 GDK_NOTE (MISC, g_print ("gdk_window_set_role: %#x %s\n",
1455 (guint) GDK_WINDOW_HWND (window),
1456 (role ? role : "NULL")));
1461 gdk_window_set_transient_for (GdkWindow *window,
1464 g_return_if_fail (window != NULL);
1465 g_return_if_fail (GDK_IS_WINDOW (window));
1467 GDK_NOTE (MISC, g_print ("gdk_window_set_transient_for: %#x %#x\n",
1468 (guint) GDK_WINDOW_HWND (window),
1469 (guint) GDK_WINDOW_HWND (parent)));
1474 gdk_window_set_background (GdkWindow *window,
1477 GdkWindowObject *private = (GdkWindowObject *)window;
1479 g_return_if_fail (window != NULL);
1480 g_return_if_fail (GDK_IS_WINDOW (window));
1482 GDK_NOTE (MISC, g_print ("gdk_window_set_background: %#x %s\n",
1483 (guint) GDK_WINDOW_HWND (window),
1484 gdk_win32_color_to_string (color)));
1486 private->bg_color = *color;
1488 if (private->bg_pixmap &&
1489 private->bg_pixmap != GDK_PARENT_RELATIVE_BG &&
1490 private->bg_pixmap != GDK_NO_BG)
1492 gdk_drawable_unref (private->bg_pixmap);
1493 private->bg_pixmap = NULL;
1498 gdk_window_set_back_pixmap (GdkWindow *window,
1500 gint parent_relative)
1502 GdkWindowObject *private = (GdkWindowObject *)window;
1504 g_return_if_fail (window != NULL);
1505 g_return_if_fail (GDK_IS_WINDOW (window));
1506 g_return_if_fail (pixmap == NULL || !parent_relative);
1508 if (private->bg_pixmap &&
1509 private->bg_pixmap != GDK_PARENT_RELATIVE_BG &&
1510 private->bg_pixmap != GDK_NO_BG)
1511 gdk_drawable_unref (private->bg_pixmap);
1513 if (parent_relative)
1515 private->bg_pixmap = GDK_PARENT_RELATIVE_BG;
1521 gdk_drawable_ref (pixmap);
1522 private->bg_pixmap = pixmap;
1526 private->bg_pixmap = GDK_NO_BG;
1532 gdk_window_set_cursor (GdkWindow *window,
1535 GdkWindowImplWin32 *impl;
1536 GdkCursorPrivate *cursor_private;
1538 HCURSOR hprevcursor;
1541 g_return_if_fail (window != NULL);
1542 g_return_if_fail (GDK_IS_WINDOW (window));
1544 impl = GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl);
1545 cursor_private = (GdkCursorPrivate*) cursor;
1547 if (!GDK_WINDOW_DESTROYED (window))
1552 hcursor = cursor_private->hcursor;
1554 GDK_NOTE (MISC, g_print ("gdk_window_set_cursor: %#x %#x\n",
1555 (guint) GDK_WINDOW_HWND (window),
1557 hprevcursor = impl->hcursor;
1558 if (hcursor == NULL)
1559 impl->hcursor = NULL;
1562 /* We must copy the cursor as it is OK to destroy the GdkCursor
1563 * while still in use for some window. See for instance
1564 * gimp_change_win_cursor() which calls
1565 * gdk_window_set_cursor (win, cursor), and immediately
1566 * afterwards gdk_cursor_destroy (cursor).
1568 impl->hcursor = CopyCursor (hcursor);
1569 GDK_NOTE (MISC, g_print ("...CopyCursor (%#x) = %#x\n",
1570 (guint) hcursor, (guint) impl->hcursor));
1572 if (hprevcursor != NULL && GetCursor () == hprevcursor)
1573 SetCursor (impl->hcursor);
1575 if (hprevcursor != NULL)
1577 GDK_NOTE (MISC, g_print ("...DestroyCursor (%#x)\n",
1578 (guint) hprevcursor));
1580 if (!DestroyCursor (hprevcursor))
1581 WIN32_API_FAILED ("DestroyCursor");
1588 gdk_window_get_geometry (GdkWindow *window,
1595 g_return_if_fail (window == NULL || GDK_IS_WINDOW (window));
1598 window = gdk_parent_root;
1600 if (!GDK_WINDOW_DESTROYED (window))
1604 if (!GetClientRect (GDK_WINDOW_HWND (window), &rect))
1605 WIN32_API_FAILED ("GetClientRect");
1612 *width = rect.right - rect.left;
1614 *height = rect.bottom - rect.top;
1616 *depth = gdk_drawable_get_visual (window)->depth;
1621 gdk_window_get_origin (GdkWindow *window,
1629 g_return_val_if_fail (window != NULL, 0);
1631 if (!GDK_WINDOW_DESTROYED (window))
1637 ClientToScreen (GDK_WINDOW_HWND (window), &pt);
1650 GDK_NOTE (MISC, g_print ("gdk_window_get_origin: %#x: +%d+%d\n",
1651 (guint) GDK_WINDOW_HWND (window),
1657 gdk_window_get_deskrelative_origin (GdkWindow *window,
1661 return gdk_window_get_origin (window, x, y);
1665 gdk_window_get_root_origin (GdkWindow *window,
1669 GdkWindowObject *rover;
1672 g_return_if_fail (window != NULL);
1673 g_return_if_fail (GDK_IS_WINDOW (window));
1675 rover = (GdkWindowObject*) window;
1681 if (GDK_WINDOW_DESTROYED (window))
1684 while (rover->parent && ((GdkWindowObject*) rover->parent)->parent)
1685 rover = (GdkWindowObject *) rover->parent;
1686 if (rover->destroyed)
1691 ClientToScreen (GDK_WINDOW_HWND (rover), &pt);
1697 GDK_NOTE (MISC, g_print ("gdk_window_get_root_origin: %#x: (%#x) +%ld+%ld\n",
1698 (guint) GDK_WINDOW_HWND (window),
1699 (guint) GDK_WINDOW_HWND (rover),
1704 gdk_window_get_pointer (GdkWindow *window,
1707 GdkModifierType *mask)
1709 GdkWindow *return_val;
1710 POINT pointc, point;
1713 g_return_val_if_fail (window == NULL || GDK_IS_WINDOW (window), NULL);
1716 window = gdk_parent_root;
1719 GetCursorPos (&pointc);
1721 ScreenToClient (GDK_WINDOW_HWND (window), &point);
1728 hwnd = WindowFromPoint (point);
1730 ScreenToClient (hwnd, &point);
1733 hwndc = ChildWindowFromPoint (hwnd, point);
1734 ClientToScreen (hwnd, &point);
1735 ScreenToClient (hwndc, &point);
1736 } while (hwndc != hwnd && (hwnd = hwndc, 1)); /* Ouch! */
1738 return_val = gdk_win32_handle_table_lookup ((GdkNativeWindow) hwnd);
1744 GetKeyboardState (kbd);
1746 if (kbd[VK_SHIFT] & 0x80)
1747 *mask |= GDK_SHIFT_MASK;
1748 if (kbd[VK_CAPITAL] & 0x80)
1749 *mask |= GDK_LOCK_MASK;
1750 if (kbd[VK_CONTROL] & 0x80)
1751 *mask |= GDK_CONTROL_MASK;
1752 if (kbd[VK_MENU] & 0x80)
1753 *mask |= GDK_MOD1_MASK;
1754 if (kbd[VK_LBUTTON] & 0x80)
1755 *mask |= GDK_BUTTON1_MASK;
1756 if (kbd[VK_MBUTTON] & 0x80)
1757 *mask |= GDK_BUTTON2_MASK;
1758 if (kbd[VK_RBUTTON] & 0x80)
1759 *mask |= GDK_BUTTON3_MASK;
1766 gdk_window_at_pointer (gint *win_x,
1770 POINT point, pointc;
1774 GetCursorPos (&pointc);
1776 hwnd = WindowFromPoint (point);
1780 window = gdk_parent_root;
1788 ScreenToClient (hwnd, &point);
1791 hwndc = ChildWindowFromPoint (hwnd, point);
1792 ClientToScreen (hwnd, &point);
1793 ScreenToClient (hwndc, &point);
1794 } while (hwndc != hwnd && (hwnd = hwndc, 1));
1796 window = gdk_win32_handle_table_lookup ((GdkNativeWindow) hwnd);
1798 if (window && (win_x || win_y))
1800 GetClientRect (hwnd, &rect);
1802 *win_x = point.x - rect.left;
1804 *win_y = point.y - rect.top;
1807 GDK_NOTE (MISC, g_print ("gdk_window_at_pointer: +%ld+%ld %#x%s\n",
1810 (window == NULL ? " NULL" : "")));
1816 gdk_window_get_events (GdkWindow *window)
1818 g_return_val_if_fail (window != NULL, 0);
1819 g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
1821 if (GDK_WINDOW_DESTROYED (window))
1824 return GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->event_mask;
1828 gdk_window_set_events (GdkWindow *window,
1829 GdkEventMask event_mask)
1831 g_return_if_fail (window != NULL);
1832 g_return_if_fail (GDK_IS_WINDOW (window));
1834 if (GDK_WINDOW_DESTROYED (window))
1837 GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->event_mask = event_mask;
1841 gdk_window_shape_combine_mask (GdkWindow *window,
1845 g_return_if_fail (window != NULL);
1846 g_return_if_fail (GDK_IS_WINDOW (window));
1850 GDK_NOTE (MISC, g_print ("gdk_window_shape_combine_mask: %#x none\n",
1851 (guint) GDK_WINDOW_HWND (window)));
1852 SetWindowRgn (GDK_WINDOW_HWND (window), NULL, TRUE);
1861 /* Convert mask bitmap to region */
1862 hrgn = BitmapToRegion (GDK_WINDOW_HWND (mask));
1864 GDK_NOTE (MISC, g_print ("gdk_window_shape_combine_mask: %#x %#x\n",
1865 (guint) GDK_WINDOW_HWND (window),
1866 (guint) GDK_WINDOW_HWND (mask)));
1868 /* SetWindowRgn wants window (not client) coordinates */
1869 dwStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1870 dwExStyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1871 GetClientRect (GDK_WINDOW_HWND (window), &rect);
1872 AdjustWindowRectEx (&rect, dwStyle, FALSE, dwExStyle);
1873 OffsetRgn (hrgn, -rect.left, -rect.top);
1875 OffsetRgn (hrgn, x, y);
1877 /* If this is a top-level window, add the title bar to the region */
1878 if (GDK_WINDOW_TYPE (window) == GDK_WINDOW_TOPLEVEL)
1880 CombineRgn (hrgn, hrgn,
1881 CreateRectRgn (0, 0, rect.right - rect.left, -rect.top),
1885 SetWindowRgn (GDK_WINDOW_HWND (window), hrgn, TRUE);
1890 gdk_window_set_override_redirect (GdkWindow *window,
1891 gboolean override_redirect)
1893 g_return_if_fail (window != NULL);
1894 g_return_if_fail (GDK_IS_WINDOW (window));
1896 g_warning ("gdk_window_set_override_redirect not implemented");
1900 gdk_window_set_icon (GdkWindow *window,
1901 GdkWindow *icon_window,
1905 g_return_if_fail (window != NULL);
1906 g_return_if_fail (GDK_IS_WINDOW (window));
1908 if (GDK_WINDOW_DESTROYED (window))
1911 /* Nothing to do, really. As we share window classes between windows
1912 * we can't have window-specific icons, sorry. Don't print any warning
1918 gdk_window_set_icon_name (GdkWindow *window,
1921 g_return_if_fail (window != NULL);
1922 g_return_if_fail (GDK_IS_WINDOW (window));
1924 if (GDK_WINDOW_DESTROYED (window))
1927 if (!SetWindowText (GDK_WINDOW_HWND (window), name))
1928 WIN32_API_FAILED ("SetWindowText");
1932 gdk_window_set_group (GdkWindow *window,
1935 g_return_if_fail (window != NULL);
1936 g_return_if_fail (GDK_IS_WINDOW (window));
1937 g_return_if_fail (leader != NULL);
1938 g_return_if_fail (GDK_IS_WINDOW (leader));
1940 if (GDK_WINDOW_DESTROYED (window) || GDK_WINDOW_DESTROYED (leader))
1943 g_warning ("gdk_window_set_group not implemented");
1947 gdk_window_set_decorations (GdkWindow *window,
1948 GdkWMDecoration decorations)
1950 LONG style, exstyle;
1952 g_return_if_fail (window != NULL);
1953 g_return_if_fail (GDK_IS_WINDOW (window));
1955 style = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1956 exstyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1958 style &= (WS_OVERLAPPED|WS_POPUP|WS_CHILD|WS_MINIMIZE|WS_VISIBLE|WS_DISABLED
1959 |WS_CLIPSIBLINGS|WS_CLIPCHILDREN|WS_MAXIMIZE);
1961 exstyle &= (WS_EX_TOPMOST|WS_EX_TRANSPARENT);
1963 if (decorations & GDK_DECOR_ALL)
1964 style |= (WS_CAPTION|WS_SYSMENU|WS_THICKFRAME|WS_MINIMIZEBOX|WS_MAXIMIZEBOX);
1965 if (decorations & GDK_DECOR_BORDER)
1966 style |= (WS_BORDER);
1967 if (decorations & GDK_DECOR_RESIZEH)
1968 style |= (WS_THICKFRAME);
1969 if (decorations & GDK_DECOR_TITLE)
1970 style |= (WS_CAPTION);
1971 if (decorations & GDK_DECOR_MENU)
1972 style |= (WS_SYSMENU);
1973 if (decorations & GDK_DECOR_MINIMIZE)
1974 style |= (WS_MINIMIZEBOX);
1975 if (decorations & GDK_DECOR_MAXIMIZE)
1976 style |= (WS_MAXIMIZEBOX);
1978 SetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE, style);
1982 gdk_window_set_functions (GdkWindow *window,
1983 GdkWMFunction functions)
1985 LONG style, exstyle;
1987 g_return_if_fail (window != NULL);
1988 g_return_if_fail (GDK_IS_WINDOW (window));
1990 style = GetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE);
1991 exstyle = GetWindowLong (GDK_WINDOW_HWND (window), GWL_EXSTYLE);
1993 style &= (WS_OVERLAPPED|WS_POPUP|WS_CHILD|WS_MINIMIZE|WS_VISIBLE|WS_DISABLED
1994 |WS_CLIPSIBLINGS|WS_CLIPCHILDREN|WS_MAXIMIZE|WS_CAPTION|WS_BORDER
1997 exstyle &= (WS_EX_TOPMOST|WS_EX_TRANSPARENT);
1999 if (functions & GDK_FUNC_ALL)
2000 style |= (WS_THICKFRAME|WS_MINIMIZEBOX|WS_MAXIMIZEBOX);
2001 if (functions & GDK_FUNC_RESIZE)
2002 style |= (WS_THICKFRAME);
2003 if (functions & GDK_FUNC_MOVE)
2004 style |= (WS_THICKFRAME);
2005 if (functions & GDK_FUNC_MINIMIZE)
2006 style |= (WS_MINIMIZEBOX);
2007 if (functions & GDK_FUNC_MAXIMIZE)
2008 style |= (WS_MAXIMIZEBOX);
2010 SetWindowLong (GDK_WINDOW_HWND (window), GWL_STYLE, style);
2014 * propagate the shapes from all child windows of a GDK window to the parent
2015 * window. Shamelessly ripped from Enlightenment's code
2021 QueryTree (HWND hwnd,
2031 child = GetWindow (hwnd, GW_CHILD);
2033 child = GetWindow (child, GW_HWNDNEXT);
2036 } while (child != NULL);
2040 *children = g_new (HWND, n);
2041 for (i = 0; i < n; i++)
2044 child = GetWindow (hwnd, GW_CHILD);
2046 child = GetWindow (child, GW_HWNDNEXT);
2047 *children[i] = child;
2053 gdk_propagate_shapes (HANDLE win,
2057 HRGN region, childRegion;
2061 SetRectEmpty (&emptyRect);
2062 region = CreateRectRgnIndirect (&emptyRect);
2064 GetWindowRgn (win, region);
2066 QueryTree (win, &list, &num);
2069 WINDOWPLACEMENT placement;
2071 placement.length = sizeof (WINDOWPLACEMENT);
2072 /* go through all child windows and combine regions */
2073 for (i = 0; i < num; i++)
2075 GetWindowPlacement (list[i], &placement);
2076 if (placement.showCmd == SW_SHOWNORMAL)
2078 childRegion = CreateRectRgnIndirect (&emptyRect);
2079 GetWindowRgn (list[i], childRegion);
2080 CombineRgn (region, region, childRegion, RGN_OR);
2081 DeleteObject (childRegion);
2084 SetWindowRgn (win, region, TRUE);
2087 DeleteObject (region);
2091 gdk_window_set_child_shapes (GdkWindow *window)
2093 g_return_if_fail (window != NULL);
2094 g_return_if_fail (GDK_IS_WINDOW (window));
2096 if (GDK_WINDOW_DESTROYED (window))
2099 gdk_propagate_shapes (GDK_WINDOW_HWND (window), FALSE);
2103 gdk_window_merge_child_shapes (GdkWindow *window)
2105 g_return_if_fail (window != NULL);
2106 g_return_if_fail (GDK_IS_WINDOW (window));
2108 if (GDK_WINDOW_DESTROYED (window))
2111 gdk_propagate_shapes (GDK_WINDOW_HWND (window), TRUE);
2114 /* Support for windows that can be guffaw-scrolled
2115 * (See http://www.gtk.org/~otaylor/whitepapers/guffaw-scrolling.txt)
2119 gdk_window_gravity_works (void)
2121 enum { UNKNOWN, NO, YES };
2122 static gint gravity_works = UNKNOWN;
2124 if (gravity_works == UNKNOWN)
2131 attr.window_type = GDK_WINDOW_TEMP;
2132 attr.wclass = GDK_INPUT_OUTPUT;
2137 attr.event_mask = 0;
2139 parent = gdk_window_new (NULL, &attr, GDK_WA_X | GDK_WA_Y);
2141 attr.window_type = GDK_WINDOW_CHILD;
2142 child = gdk_window_new (parent, &attr, GDK_WA_X | GDK_WA_Y);
2144 gdk_window_set_static_win_gravity (child, TRUE);
2146 gdk_window_resize (parent, 100, 110);
2147 gdk_window_move (parent, 0, -10);
2148 gdk_window_move_resize (parent, 0, 0, 100, 100);
2150 gdk_window_resize (parent, 100, 110);
2151 gdk_window_move (parent, 0, -10);
2152 gdk_window_move_resize (parent, 0, 0, 100, 100);
2154 gdk_window_get_geometry (child, NULL, &y, NULL, NULL, NULL);
2156 gdk_window_destroy (parent);
2157 gdk_window_destroy (child);
2159 gravity_works = ((y == -20) ? YES : NO);
2162 return (gravity_works == YES);
2166 gdk_window_set_static_bit_gravity (GdkWindow *window, gboolean on)
2168 g_return_if_fail (window != NULL);
2170 GDK_NOTE (MISC, g_print ("gdk_window_set_static_bit_gravity: Not implemented\n"));
2174 gdk_window_set_static_win_gravity (GdkWindow *window, gboolean on)
2176 g_return_if_fail (window != NULL);
2179 g_print ("gdk_window_set_static_win_gravity: Not implemented\n"));
2182 /*************************************************************
2183 * gdk_window_set_static_gravities:
2184 * Set the bit gravity of the given window to static,
2185 * and flag it so all children get static subwindow
2188 * window: window for which to set static gravity
2189 * use_static: Whether to turn static gravity on or off.
2191 * Does the XServer support static gravity?
2192 *************************************************************/
2195 gdk_window_set_static_gravities (GdkWindow *window,
2196 gboolean use_static)
2198 GdkWindowObject *private = (GdkWindowObject *)window;
2201 g_return_val_if_fail (window != NULL, FALSE);
2202 g_return_val_if_fail (GDK_IS_WINDOW (window), FALSE);
2204 if (!use_static == !private->guffaw_gravity)
2207 if (use_static && !gdk_window_gravity_works ())
2210 private->guffaw_gravity = use_static;
2212 if (!GDK_WINDOW_DESTROYED (window))
2214 gdk_window_set_static_bit_gravity (window, use_static);
2216 tmp_list = private->children;
2219 gdk_window_set_static_win_gravity (window, use_static);
2221 tmp_list = tmp_list->next;
2229 * Setting window states
2232 gdk_window_iconify (GdkWindow *window)
2234 g_return_if_fail (window != NULL);
2235 g_return_if_fail (GDK_IS_WINDOW (window));
2237 if (GDK_WINDOW_DESTROYED (window))
2240 if (GDK_WINDOW_IS_MAPPED (window))
2242 ShowWindow (GDK_WINDOW_HWND (window), SW_MINIMIZE);
2246 /* Flip our client side flag, the real work happens on map. */
2247 gdk_synthesize_window_state (window,
2249 GDK_WINDOW_STATE_ICONIFIED);
2254 gdk_window_deiconify (GdkWindow *window)
2256 g_return_if_fail (window != NULL);
2257 g_return_if_fail (GDK_IS_WINDOW (window));
2259 if (GDK_WINDOW_DESTROYED (window))
2262 if (GDK_WINDOW_IS_MAPPED (window))
2264 gdk_window_show (window);
2268 /* Flip our client side flag, the real work happens on map. */
2269 gdk_synthesize_window_state (window,
2270 GDK_WINDOW_STATE_ICONIFIED,
2276 gdk_window_stick (GdkWindow *window)
2278 g_return_if_fail (GDK_IS_WINDOW (window));
2280 if (GDK_WINDOW_DESTROYED (window))
2283 if (GDK_WINDOW_IS_MAPPED (window))
2285 /* "stick" means stick to all desktops _and_ do not scroll with the
2286 * viewport. i.e. glue to the monitor glass in all cases.
2288 g_warning ("gdk_window_stick (0x%X) ???", GDK_WINDOW_HWND (window));
2292 /* Flip our client side flag, the real work happens on map. */
2293 gdk_synthesize_window_state (window,
2295 GDK_WINDOW_STATE_STICKY);
2300 gdk_window_unstick (GdkWindow *window)
2302 g_return_if_fail (GDK_IS_WINDOW (window));
2304 if (GDK_WINDOW_DESTROYED (window))
2307 if (GDK_WINDOW_IS_MAPPED (window))
2309 g_warning ("gdk_window_unstick (0x%X) ???", GDK_WINDOW_HWND (window));
2313 /* Flip our client side flag, the real work happens on map. */
2314 gdk_synthesize_window_state (window,
2315 GDK_WINDOW_STATE_STICKY,
2322 gdk_window_maximize (GdkWindow *window)
2324 g_return_if_fail (GDK_IS_WINDOW (window));
2326 if (GDK_WINDOW_DESTROYED (window))
2329 if (GDK_WINDOW_IS_MAPPED (window))
2330 ShowWindow (GDK_WINDOW_HWND (window), SW_MAXIMIZE);
2332 gdk_synthesize_window_state (window,
2334 GDK_WINDOW_STATE_MAXIMIZED);
2338 gdk_window_unmaximize (GdkWindow *window)
2340 g_return_if_fail (GDK_IS_WINDOW (window));
2342 if (GDK_WINDOW_DESTROYED (window))
2345 if (GDK_WINDOW_IS_MAPPED (window))
2346 ShowWindow (GDK_WINDOW_HWND (window), SW_RESTORE);
2348 gdk_synthesize_window_state (window,
2349 GDK_WINDOW_STATE_MAXIMIZED,
2354 gdk_window_focus (GdkWindow *window,
2357 g_return_if_fail (GDK_IS_WINDOW (window));
2359 if (GDK_WINDOW_DESTROYED (window))
2362 ShowWindow (GDK_WINDOW_HWND (window), SW_SHOWNORMAL);
2366 gdk_window_set_modal_hint (GdkWindow *window,
2369 GdkWindowObject *private;
2371 g_return_if_fail (window != NULL);
2372 g_return_if_fail (GDK_IS_WINDOW (window));
2374 if (GDK_WINDOW_DESTROYED (window))
2377 private = (GdkWindowObject*) window;
2379 private->modal_hint = modal;
2381 if (GDK_WINDOW_IS_MAPPED (window))
2382 if (!SetWindowPos (GDK_WINDOW_HWND (window), HWND_TOPMOST,
2383 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE))
2384 WIN32_API_FAILED ("SetWindowPos");
2388 gdk_window_set_type_hint (GdkWindow *window,
2389 GdkWindowTypeHint hint)
2393 g_return_if_fail (window != NULL);
2394 g_return_if_fail (GDK_IS_WINDOW (window));
2396 if (GDK_WINDOW_DESTROYED (window))
2401 case GDK_WINDOW_TYPE_HINT_DIALOG:
2402 atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_DIALOG", FALSE);
2404 case GDK_WINDOW_TYPE_HINT_MENU:
2405 atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_MENU", FALSE);
2407 case GDK_WINDOW_TYPE_HINT_TOOLBAR:
2408 atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_TOOLBAR", FALSE);
2411 g_warning ("Unknown hint %d passed to gdk_window_set_type_hint", hint);
2413 case GDK_WINDOW_TYPE_HINT_NORMAL:
2414 atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_NORMAL", FALSE);
2421 g_print ("gdk_window_set_type_hint (0x%0X)\n",
2422 GDK_WINDOW_HWND (window)));
2426 gdk_window_shape_combine_region (GdkWindow *window,
2427 GdkRegion *shape_region,
2431 gint xoffset, yoffset;
2433 g_return_if_fail (GDK_IS_WINDOW (window));
2435 if (GDK_WINDOW_DESTROYED (window))
2438 /* XXX: even on X implemented conditional ... */