mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2024-12-27 22:20:24 +00:00
34e8bd9dba
Patches by Sam Thursfield, from bug #623476.
293 lines
7.9 KiB
C
293 lines
7.9 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/* gdkgeometry-win32.c: emulation of 32 bit coordinates within the
|
|
* limits of Win32 GDI. The idea of big window emulation is more or less
|
|
* a copy of the X11 version, and the equvalent of guffaw scrolling
|
|
* is ScrollWindowEx(). While we determine the invalidated region
|
|
* ourself during scrolling, we do not pass SW_INVALIDATE to
|
|
* ScrollWindowEx() to avoid a unnecessary WM_PAINT.
|
|
*
|
|
* Bits are always scrolled correctly by ScrollWindowEx(), but
|
|
* some big children may hit the coordinate boundary (i.e.
|
|
* win32_x/win32_y < -16383) after scrolling. They needed to be moved
|
|
* back to the real position determined by gdk_window_compute_position().
|
|
* This is handled in gdk_window_postmove().
|
|
*
|
|
* The X11 version by Owen Taylor <otaylor@redhat.com>
|
|
* Copyright Red Hat, Inc. 2000
|
|
* Win32 hack by Tor Lillqvist <tml@iki.fi>
|
|
* and Hans Breuer <hans@breuer.org>
|
|
* Modified by Ivan, Wong Yat Cheung <email@ivanwong.info>
|
|
* so that big window emulation finally works.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "gdk.h" /* For gdk_rectangle_intersect */
|
|
#include "gdkinternals.h"
|
|
#include "gdkprivate-win32.h"
|
|
|
|
#define SIZE_LIMIT 32767
|
|
|
|
typedef struct _GdkWindowParentPos GdkWindowParentPos;
|
|
|
|
static void tmp_unset_bg (GdkWindow *window);
|
|
static void tmp_reset_bg (GdkWindow *window);
|
|
|
|
void
|
|
_gdk_window_move_resize_child (GdkWindow *window,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GdkWindowImplWin32 *impl;
|
|
GdkWindowObject *obj;
|
|
gboolean is_move;
|
|
gboolean is_resize;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
obj = GDK_WINDOW_OBJECT (window);
|
|
impl = GDK_WINDOW_IMPL_WIN32 (obj->impl);
|
|
|
|
is_move = (x - obj->x != 0) && (y - obj->y != 0);
|
|
is_resize = obj->width != width && obj->height != height;
|
|
|
|
GDK_NOTE (MISC, g_print ("_gdk_window_move_resize_child: %s@%+d%+d %dx%d@%+d%+d\n",
|
|
_gdk_win32_drawable_description (window),
|
|
obj->x, obj->y, width, height, x, y));
|
|
|
|
if (width > 65535 || height > 65535)
|
|
{
|
|
g_warning ("Native children wider or taller than 65535 pixels are not supported.");
|
|
|
|
if (width > 65535)
|
|
width = 65535;
|
|
if (height > 65535)
|
|
height = 65535;
|
|
}
|
|
|
|
obj->x = x;
|
|
obj->y = y;
|
|
obj->width = width;
|
|
obj->height = height;
|
|
|
|
_gdk_win32_window_tmp_unset_parent_bg (window);
|
|
_gdk_win32_window_tmp_unset_bg (window, TRUE);
|
|
|
|
GDK_NOTE (MISC, g_print ("... SetWindowPos(%p,NULL,%d,%d,%d,%d,"
|
|
"NOACTIVATE|NOZORDER%s%s)\n",
|
|
GDK_WINDOW_HWND (window),
|
|
obj->x + obj->parent->abs_x, obj->y + obj->parent->abs_y,
|
|
width, height,
|
|
(is_move ? "" : "|NOMOVE"),
|
|
(is_resize ? "" : "|NOSIZE")));
|
|
|
|
API_CALL (SetWindowPos, (GDK_WINDOW_HWND (window), NULL,
|
|
obj->x + obj->parent->abs_x, obj->y + obj->parent->abs_y,
|
|
width, height,
|
|
SWP_NOACTIVATE | SWP_NOZORDER |
|
|
(is_move ? 0 : SWP_NOMOVE) |
|
|
(is_resize ? 0 : SWP_NOSIZE)));
|
|
|
|
//_gdk_win32_window_tmp_reset_parent_bg (window);
|
|
_gdk_win32_window_tmp_reset_bg (window, TRUE);
|
|
}
|
|
|
|
void
|
|
_gdk_win32_window_tmp_unset_bg (GdkWindow *window,
|
|
gboolean recurse)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
private = (GdkWindowObject *)window;
|
|
|
|
if (private->input_only || private->destroyed ||
|
|
(private->window_type != GDK_WINDOW_ROOT &&
|
|
!GDK_WINDOW_IS_MAPPED (window)))
|
|
return;
|
|
|
|
if (_gdk_window_has_impl (window) &&
|
|
GDK_WINDOW_IS_WIN32 (window) &&
|
|
private->window_type != GDK_WINDOW_ROOT &&
|
|
private->window_type != GDK_WINDOW_FOREIGN)
|
|
tmp_unset_bg (window);
|
|
|
|
if (recurse)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = private->children; l != NULL; l = l->next)
|
|
_gdk_win32_window_tmp_unset_bg (l->data, TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
tmp_unset_bg (GdkWindow *window)
|
|
{
|
|
GdkWindowImplWin32 *impl;
|
|
GdkWindowObject *obj;
|
|
|
|
obj = (GdkWindowObject *) window;
|
|
impl = GDK_WINDOW_IMPL_WIN32 (obj->impl);
|
|
|
|
impl->no_bg = TRUE;
|
|
|
|
/*
|
|
* The X version sets background = None to avoid updateing for a moment.
|
|
* Not sure if this could really emulate it.
|
|
*/
|
|
if (obj->bg_pixmap != GDK_NO_BG)
|
|
{
|
|
///* handled in WM_ERASEBKGRND proceesing */;
|
|
|
|
//HDC hdc = GetDC (GDK_WINDOW_HWND (window));
|
|
//erase_background (window, hdc);
|
|
}
|
|
}
|
|
|
|
static void
|
|
tmp_reset_bg (GdkWindow *window)
|
|
{
|
|
GdkWindowObject *obj;
|
|
GdkWindowImplWin32 *impl;
|
|
|
|
obj = GDK_WINDOW_OBJECT (window);
|
|
impl = GDK_WINDOW_IMPL_WIN32 (obj->impl);
|
|
|
|
impl->no_bg = FALSE;
|
|
}
|
|
|
|
void
|
|
_gdk_win32_window_tmp_unset_parent_bg (GdkWindow *window)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject*)window;
|
|
|
|
if (GDK_WINDOW_TYPE (private->parent) == GDK_WINDOW_ROOT)
|
|
return;
|
|
|
|
window = _gdk_window_get_impl_window ((GdkWindow*)private->parent);
|
|
_gdk_win32_window_tmp_unset_bg (window, FALSE);
|
|
}
|
|
|
|
void
|
|
_gdk_win32_window_tmp_reset_bg (GdkWindow *window,
|
|
gboolean recurse)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject*)window;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (private->input_only || private->destroyed ||
|
|
(private->window_type != GDK_WINDOW_ROOT && !GDK_WINDOW_IS_MAPPED (window)))
|
|
return;
|
|
|
|
if (_gdk_window_has_impl (window) &&
|
|
GDK_WINDOW_IS_WIN32 (window) &&
|
|
private->window_type != GDK_WINDOW_ROOT &&
|
|
private->window_type != GDK_WINDOW_FOREIGN)
|
|
{
|
|
tmp_reset_bg (window);
|
|
}
|
|
|
|
if (recurse)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = private->children; l != NULL; l = l->next)
|
|
_gdk_win32_window_tmp_reset_bg (l->data, TRUE);
|
|
}
|
|
}
|
|
|
|
/*
|
|
void
|
|
_gdk_win32_window_tmp_reset_bg (GdkWindow *window)
|
|
{
|
|
GdkWindowImplWin32 *impl;
|
|
GdkWindowObject *obj;
|
|
|
|
obj = (GdkWindowObject *) window;
|
|
impl = GDK_WINDOW_IMPL_WIN32 (obj->impl);
|
|
|
|
impl->no_bg = FALSE;
|
|
}
|
|
*/
|
|
|
|
#if 0
|
|
static cairo_region_t *
|
|
gdk_window_clip_changed (GdkWindow *window,
|
|
GdkRectangle *old_clip,
|
|
GdkRectangle *new_clip)
|
|
{
|
|
GdkWindowImplWin32 *impl;
|
|
GdkWindowObject *obj;
|
|
cairo_region_t *old_clip_region;
|
|
cairo_region_t *new_clip_region;
|
|
|
|
if (((GdkWindowObject *)window)->input_only)
|
|
return NULL;
|
|
|
|
obj = (GdkWindowObject *) window;
|
|
impl = GDK_WINDOW_IMPL_WIN32 (obj->impl);
|
|
|
|
old_clip_region = cairo_region_create_rectangle (old_clip);
|
|
new_clip_region = cairo_region_create_rectangle (new_clip);
|
|
|
|
/* Trim invalid region of window to new clip rectangle
|
|
*/
|
|
if (obj->update_area)
|
|
cairo_region_intersect (obj->update_area, new_clip_region);
|
|
|
|
/* Invalidate newly exposed portion of window
|
|
*/
|
|
cairo_region_subtract (new_clip_region, old_clip_region);
|
|
if (!cairo_region_is_empty (new_clip_region))
|
|
gdk_window_tmp_unset_bg (window);
|
|
else
|
|
{
|
|
cairo_region_destroy (new_clip_region);
|
|
new_clip_region = NULL;
|
|
}
|
|
|
|
cairo_region_destroy (old_clip_region);
|
|
|
|
return new_clip_region;
|
|
}
|
|
#endif
|
|
|
|
#if 0
|
|
static void
|
|
gdk_window_post_scroll (GdkWindow *window,
|
|
cairo_region_t *new_clip_region)
|
|
{
|
|
GDK_NOTE (EVENTS,
|
|
g_print ("gdk_window_clip_changed: invalidating region: %s\n",
|
|
_gdk_win32_cairo_region_to_string (new_clip_region)));
|
|
|
|
gdk_window_invalidate_region (window, new_clip_region, FALSE);
|
|
g_print ("gdk_window_post_scroll\n");
|
|
cairo_region_destroy (new_clip_region);
|
|
}
|
|
|
|
#endif
|