2001-05-30 11:36:31 -04:00
|
|
|
/* Metacity X managed windows */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2001 Havoc Pennington
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "window.h"
|
|
|
|
#include "util.h"
|
2001-05-31 02:42:58 -04:00
|
|
|
#include "frame.h"
|
|
|
|
#include "errors.h"
|
2001-06-02 00:14:18 -04:00
|
|
|
#include <X11/Xatom.h>
|
|
|
|
|
|
|
|
static void constrain_size (MetaWindow *window,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int *new_width,
|
|
|
|
int *new_height);
|
|
|
|
static int update_size_hints (MetaWindow *window);
|
|
|
|
static int update_title (MetaWindow *window);
|
|
|
|
static int update_protocols (MetaWindow *window);
|
|
|
|
static gboolean process_configure_request (MetaWindow *window,
|
|
|
|
XConfigureRequestEvent *event);
|
|
|
|
static gboolean process_property_notify (MetaWindow *window,
|
|
|
|
XPropertyEvent *event);
|
|
|
|
|
|
|
|
|
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
|
|
|
|
MetaWindow*
|
|
|
|
meta_window_new (MetaDisplay *display, Window xwindow)
|
|
|
|
{
|
|
|
|
MetaWindow *window;
|
|
|
|
XWindowAttributes attrs;
|
|
|
|
GSList *tmp;
|
2001-06-02 00:14:18 -04:00
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_verbose ("Attempting to manage 0x%lx\n", xwindow);
|
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
/* round trip */
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_error_trap_push (display);
|
2001-06-02 00:14:18 -04:00
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
if (XGetWindowAttributes (display->xdisplay,
|
2001-05-31 02:42:58 -04:00
|
|
|
xwindow, &attrs) == Success &&
|
|
|
|
attrs.override_redirect)
|
|
|
|
{
|
|
|
|
/* Oops. Probably attempted to manage override redirect window
|
|
|
|
* in initial screen_manage_all_windows() call.
|
|
|
|
*/
|
|
|
|
meta_error_trap_pop (display);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
XAddToSaveSet (display->xdisplay, xwindow);
|
|
|
|
|
|
|
|
XSelectInput (display->xdisplay, xwindow,
|
|
|
|
StructureNotifyMask);
|
|
|
|
|
|
|
|
if (meta_error_trap_pop (display) != Success)
|
2001-05-30 11:36:31 -04:00
|
|
|
{
|
|
|
|
meta_verbose ("Window 0x%lx disappeared just as we tried to manage it\n",
|
|
|
|
xwindow);
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
window = g_new (MetaWindow, 1);
|
|
|
|
|
|
|
|
window->xwindow = xwindow;
|
2001-05-31 02:42:58 -04:00
|
|
|
|
|
|
|
/* this is in window->screen->display, but that's too annoying to
|
|
|
|
* type
|
|
|
|
*/
|
|
|
|
window->display = display;
|
2001-05-30 11:36:31 -04:00
|
|
|
|
|
|
|
window->screen = NULL;
|
|
|
|
tmp = display->screens;
|
|
|
|
while (tmp != NULL)
|
|
|
|
{
|
|
|
|
if (((MetaScreen *)tmp->data)->xscreen == attrs.screen)
|
|
|
|
{
|
|
|
|
window->screen = tmp->data;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = tmp->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_assert (window->screen);
|
2001-05-31 02:42:58 -04:00
|
|
|
|
|
|
|
window->rect.x = attrs.x;
|
|
|
|
window->rect.y = attrs.y;
|
|
|
|
window->rect.width = attrs.width;
|
|
|
|
window->rect.height = attrs.height;
|
|
|
|
window->depth = attrs.depth;
|
|
|
|
window->xvisual = attrs.visual;
|
2001-05-31 23:00:01 -04:00
|
|
|
|
2001-06-02 00:14:18 -04:00
|
|
|
window->title = NULL;
|
|
|
|
window->iconic = FALSE;
|
|
|
|
|
|
|
|
window->desc = g_strdup_printf ("0x%lx", window->xwindow);
|
2001-05-31 02:42:58 -04:00
|
|
|
|
|
|
|
meta_display_register_x_window (display, &window->xwindow, window);
|
2001-05-30 11:36:31 -04:00
|
|
|
|
2001-06-02 00:14:18 -04:00
|
|
|
update_size_hints (window);
|
|
|
|
update_title (window);
|
|
|
|
update_protocols (window);
|
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
window->frame = NULL;
|
|
|
|
meta_window_ensure_frame (window);
|
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_window_free (MetaWindow *window)
|
|
|
|
{
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_verbose ("Unmanaging 0x%lx\n", window->xwindow);
|
|
|
|
|
|
|
|
meta_display_unregister_x_window (window->display, window->xwindow);
|
2001-05-31 23:00:01 -04:00
|
|
|
|
|
|
|
g_free (window->title);
|
2001-05-31 02:42:58 -04:00
|
|
|
|
|
|
|
meta_window_destroy_frame (window);
|
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
g_free (window);
|
|
|
|
}
|
|
|
|
|
2001-06-02 00:14:18 -04:00
|
|
|
void
|
|
|
|
meta_window_show (MetaWindow *window)
|
|
|
|
{
|
|
|
|
if (window->frame)
|
|
|
|
XMapWindow (window->display->xdisplay, window->frame->xwindow);
|
|
|
|
XMapWindow (window->display->xdisplay, window->xwindow);
|
|
|
|
|
|
|
|
window->iconic = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_window_hide (MetaWindow *window)
|
|
|
|
{
|
|
|
|
if (window->frame)
|
|
|
|
XUnmapWindow (window->display->xdisplay, window->frame->xwindow);
|
|
|
|
XUnmapWindow (window->display->xdisplay, window->xwindow);
|
|
|
|
|
|
|
|
window->iconic = TRUE;
|
|
|
|
}
|
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
gboolean
|
|
|
|
meta_window_event (MetaWindow *window,
|
|
|
|
XEvent *event)
|
|
|
|
{
|
2001-05-31 02:42:58 -04:00
|
|
|
if (window->frame &&
|
|
|
|
event->xany.window == window->frame->xwindow)
|
|
|
|
return meta_frame_event (window->frame, event);
|
2001-05-30 11:36:31 -04:00
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
if (event->xany.window != window->xwindow)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case KeyPress:
|
|
|
|
break;
|
|
|
|
case KeyRelease:
|
|
|
|
break;
|
|
|
|
case ButtonPress:
|
|
|
|
break;
|
|
|
|
case ButtonRelease:
|
|
|
|
break;
|
|
|
|
case MotionNotify:
|
|
|
|
break;
|
|
|
|
case EnterNotify:
|
|
|
|
break;
|
|
|
|
case LeaveNotify:
|
|
|
|
break;
|
|
|
|
case FocusIn:
|
|
|
|
break;
|
|
|
|
case FocusOut:
|
|
|
|
break;
|
|
|
|
case KeymapNotify:
|
|
|
|
break;
|
|
|
|
case Expose:
|
|
|
|
break;
|
|
|
|
case GraphicsExpose:
|
|
|
|
break;
|
|
|
|
case NoExpose:
|
|
|
|
break;
|
|
|
|
case VisibilityNotify:
|
|
|
|
break;
|
|
|
|
case CreateNotify:
|
|
|
|
break;
|
|
|
|
case DestroyNotify:
|
|
|
|
meta_window_free (window);
|
|
|
|
return TRUE;
|
|
|
|
break;
|
|
|
|
case UnmapNotify:
|
2001-06-02 00:14:18 -04:00
|
|
|
/* Window withdrawn */
|
|
|
|
meta_window_free (window);
|
|
|
|
return TRUE;
|
2001-05-31 02:42:58 -04:00
|
|
|
break;
|
|
|
|
case MapNotify:
|
|
|
|
break;
|
|
|
|
case MapRequest:
|
|
|
|
break;
|
|
|
|
case ReparentNotify:
|
|
|
|
break;
|
|
|
|
case ConfigureNotify:
|
|
|
|
if (event->xconfigure.override_redirect)
|
|
|
|
{
|
2001-06-02 00:14:18 -04:00
|
|
|
/* Unmanage it, override_redirect was toggled on?
|
|
|
|
* Can this happen?
|
|
|
|
*/
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_window_free (window);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ConfigureRequest:
|
2001-06-02 00:14:18 -04:00
|
|
|
return process_configure_request (window, &event->xconfigurerequest);
|
2001-05-31 02:42:58 -04:00
|
|
|
break;
|
|
|
|
case GravityNotify:
|
|
|
|
break;
|
|
|
|
case ResizeRequest:
|
|
|
|
break;
|
|
|
|
case CirculateNotify:
|
|
|
|
break;
|
|
|
|
case CirculateRequest:
|
|
|
|
break;
|
|
|
|
case PropertyNotify:
|
2001-06-02 00:14:18 -04:00
|
|
|
return process_property_notify (window, &event->xproperty);
|
2001-05-31 02:42:58 -04:00
|
|
|
break;
|
|
|
|
case SelectionClear:
|
|
|
|
break;
|
|
|
|
case SelectionRequest:
|
|
|
|
break;
|
|
|
|
case SelectionNotify:
|
|
|
|
break;
|
|
|
|
case ColormapNotify:
|
|
|
|
break;
|
|
|
|
case ClientMessage:
|
|
|
|
break;
|
|
|
|
case MappingNotify:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Didn't use this event */
|
|
|
|
return FALSE;
|
2001-05-30 11:36:31 -04:00
|
|
|
}
|
2001-06-02 00:14:18 -04:00
|
|
|
|
|
|
|
static gboolean
|
|
|
|
process_property_notify (MetaWindow *window,
|
|
|
|
XPropertyEvent *event)
|
|
|
|
{
|
|
|
|
if (event->atom == XA_WM_NAME ||
|
|
|
|
event->atom == window->display->atom_net_wm_name)
|
|
|
|
{
|
|
|
|
update_title (window);
|
|
|
|
}
|
|
|
|
else if (event->atom == XA_WM_NORMAL_HINTS)
|
|
|
|
{
|
|
|
|
update_size_hints (window);
|
|
|
|
}
|
|
|
|
else if (event->atom == XA_WM_PROTOCOLS)
|
|
|
|
{
|
|
|
|
update_protocols (window);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
process_configure_request (MetaWindow *window,
|
|
|
|
XConfigureRequestEvent *event)
|
|
|
|
{
|
|
|
|
/* ICCCM 4.1.5 */
|
|
|
|
|
|
|
|
/* Note that x, y is the corner of the window border,
|
|
|
|
* and width, height is the size of the window inside
|
|
|
|
* its border, but that we always deny border requests
|
|
|
|
* because we don't believe in clients who use lame-ass
|
|
|
|
* X features like that.
|
|
|
|
*/
|
|
|
|
window->border_width = event->border_width;
|
|
|
|
window->size_hints.x = event->x;
|
|
|
|
window->size_hints.y = event->y;
|
|
|
|
window->size_hints.width = event->width;
|
|
|
|
window->size_hints.height = event->height;
|
|
|
|
|
|
|
|
/* FIXME */
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
update_size_hints (MetaWindow *window)
|
|
|
|
{
|
|
|
|
int x, y, w, h;
|
|
|
|
|
|
|
|
/* Save the last ConfigureRequest, which we put here.
|
|
|
|
* Values here set in the hints are supposed to
|
|
|
|
* be ignored.
|
|
|
|
*/
|
|
|
|
x = window->size_hints.x;
|
|
|
|
y = window->size_hints.y;
|
|
|
|
w = window->size_hints.width;
|
|
|
|
h = window->size_hints.height;
|
|
|
|
|
|
|
|
window->size_hints.flags = 0;
|
|
|
|
|
|
|
|
meta_error_trap_push (window->display);
|
|
|
|
XGetNormalHints (window->display->xdisplay,
|
|
|
|
window->xwindow,
|
|
|
|
&window->size_hints);
|
|
|
|
|
|
|
|
/* Put it back. */
|
|
|
|
window->size_hints.x = x;
|
|
|
|
window->size_hints.y = y;
|
|
|
|
window->size_hints.width = w;
|
|
|
|
window->size_hints.height = h;
|
|
|
|
|
|
|
|
if (window->size_hints.flags & PBaseSize)
|
|
|
|
;
|
|
|
|
else if (window->size_hints.flags & PMinSize)
|
|
|
|
{
|
|
|
|
window->size_hints.base_width = window->size_hints.min_width;
|
|
|
|
window->size_hints.base_height = window->size_hints.min_height;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
window->size_hints.base_width = 0;
|
|
|
|
window->size_hints.base_height = 0;
|
|
|
|
}
|
|
|
|
window->size_hints.flags |= PBaseSize;
|
|
|
|
|
|
|
|
if (window->size_hints.flags & PMinSize)
|
|
|
|
;
|
|
|
|
else if (window->size_hints.flags & PBaseSize)
|
|
|
|
{
|
|
|
|
window->size_hints.min_width = window->size_hints.base_width;
|
|
|
|
window->size_hints.min_height = window->size_hints.base_height;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
window->size_hints.min_width = 0;
|
|
|
|
window->size_hints.min_height = 0;
|
|
|
|
}
|
|
|
|
window->size_hints.flags |= PMinSize;
|
|
|
|
|
|
|
|
if (window->size_hints.flags & PMaxSize)
|
|
|
|
;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
window->size_hints.max_width = G_MAXINT;
|
|
|
|
window->size_hints.max_height = G_MAXINT;
|
|
|
|
window->size_hints.flags |= PMaxSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->size_hints.flags & PResizeInc)
|
|
|
|
;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
window->size_hints.width_inc = 1;
|
|
|
|
window->size_hints.height_inc = 1;
|
|
|
|
window->size_hints.flags |= PResizeInc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->size_hints.flags & PAspect)
|
|
|
|
{
|
|
|
|
/* don't divide by 0 */
|
|
|
|
if (window->size_hints.min_aspect.y < 1)
|
|
|
|
window->size_hints.min_aspect.y = 1;
|
|
|
|
if (window->size_hints.max_aspect.y < 1)
|
|
|
|
window->size_hints.max_aspect.y = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
window->size_hints.min_aspect.x = 1;
|
|
|
|
window->size_hints.min_aspect.y = G_MAXINT;
|
|
|
|
window->size_hints.max_aspect.x = G_MAXINT;
|
|
|
|
window->size_hints.max_aspect.y = 1;
|
|
|
|
window->size_hints.flags |= PAspect;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->size_hints.flags & PWinGravity)
|
|
|
|
;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
window->size_hints.win_gravity = NorthWestGravity;
|
|
|
|
window->size_hints.flags |= PWinGravity;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME constrain the window to these hints */
|
|
|
|
|
|
|
|
return meta_error_trap_pop (window->display);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
update_title (MetaWindow *window)
|
|
|
|
{
|
|
|
|
XTextProperty text;
|
|
|
|
|
|
|
|
meta_error_trap_push (window->display);
|
|
|
|
|
|
|
|
if (window->title)
|
|
|
|
{
|
|
|
|
g_free (window->title);
|
|
|
|
window->title = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME How does memory management for text.value work? */
|
|
|
|
XGetTextProperty (window->display->xdisplay,
|
|
|
|
window->xwindow,
|
|
|
|
&text,
|
|
|
|
window->display->atom_net_wm_name);
|
|
|
|
|
|
|
|
if (text.nitems > 0 &&
|
|
|
|
text.format == 8 &&
|
|
|
|
g_utf8_validate (text.value, text.nitems, NULL))
|
|
|
|
{
|
|
|
|
meta_verbose ("Using _NET_WM_NAME for new title of %s: '%s'\n",
|
|
|
|
window->desc, text.value);
|
|
|
|
|
|
|
|
window->title = g_strdup (text.value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->title == NULL &&
|
|
|
|
text.nitems > 0)
|
|
|
|
meta_warning ("_NET_WM_NAME property for %s contained invalid UTF-8\n",
|
|
|
|
window->desc);
|
|
|
|
|
|
|
|
if (window->title == NULL)
|
|
|
|
{
|
|
|
|
XGetTextProperty (window->display->xdisplay,
|
|
|
|
window->xwindow,
|
|
|
|
&text,
|
|
|
|
XA_WM_NAME);
|
|
|
|
|
|
|
|
if (text.nitems > 0)
|
|
|
|
{
|
|
|
|
/* FIXME This isn't particularly correct. Need to copy the
|
|
|
|
* GDK code...
|
|
|
|
*/
|
|
|
|
char *str;
|
|
|
|
GError *err;
|
|
|
|
|
|
|
|
err = NULL;
|
|
|
|
str = g_locale_to_utf8 (text.value,
|
|
|
|
(text.format / 8) * text.nitems,
|
|
|
|
NULL, NULL,
|
|
|
|
&err);
|
|
|
|
if (err != NULL)
|
|
|
|
{
|
|
|
|
meta_warning ("WM_NAME property for %s contained stuff we are too dumb to figure out: %s\n", window->desc, err->message);
|
|
|
|
g_error_free (err);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->title)
|
|
|
|
meta_verbose ("Using WM_NAME for new title of %s: '%s'\n",
|
|
|
|
window->desc, text.value);
|
|
|
|
|
|
|
|
window->title = str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->title == NULL)
|
|
|
|
window->title = g_strdup ("");
|
|
|
|
|
|
|
|
window->desc = g_strdup_printf ("0x%lx (%.10s)", window->xwindow, window->title);
|
|
|
|
|
|
|
|
return meta_error_trap_pop (window->display);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
update_protocols (MetaWindow *window)
|
|
|
|
{
|
|
|
|
Atom *protocols;
|
|
|
|
int n_protocols;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
meta_error_trap_push (window->display);
|
|
|
|
XGetWMProtocols (window->display->xdisplay,
|
|
|
|
window->xwindow,
|
|
|
|
&protocols,
|
|
|
|
&n_protocols);
|
|
|
|
|
|
|
|
window->take_focus = FALSE;
|
|
|
|
window->delete_window = FALSE;
|
|
|
|
i = 0;
|
|
|
|
while (i < n_protocols)
|
|
|
|
{
|
|
|
|
if (protocols[i] == _XA_WM_TAKE_FOCUS)
|
|
|
|
window->takes_focus = TRUE;
|
|
|
|
else if (protocols[i] == _XA_WM_DELETE_WINDOW)
|
|
|
|
window->delete_window = TRUE;
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (protocols)
|
|
|
|
XFree (protocols);
|
|
|
|
|
|
|
|
return meta_error_trap_pop (window->display);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
constrain_size (MetaWindow *window,
|
|
|
|
int width, int height,
|
|
|
|
int *new_width, int *new_height)
|
|
|
|
{
|
|
|
|
/* This is partially borrowed from GTK (LGPL), which in turn
|
|
|
|
* partially borrowed from fvwm,
|
|
|
|
*
|
|
|
|
* Copyright 1993, Robert Nation
|
|
|
|
* You may use this code for any purpose, as long as the original
|
|
|
|
* copyright remains in the source code and all documentation
|
|
|
|
*
|
|
|
|
* which in turn borrows parts of the algorithm from uwm
|
|
|
|
*/
|
|
|
|
int delta;
|
|
|
|
double min_aspect, max_aspect;
|
|
|
|
|
|
|
|
#define FLOOR(value, base) ( ((gint) ((value) / (base))) * (base) )
|
|
|
|
|
|
|
|
/* clamp width and height to min and max values
|
|
|
|
*/
|
|
|
|
width = CLAMP (width,
|
|
|
|
window->size_hints.min_width,
|
|
|
|
window->size_hints.max_width);
|
|
|
|
height = CLAMP (height,
|
|
|
|
window->size_hints.min_height,
|
|
|
|
window->size_hints.max_height);
|
|
|
|
|
|
|
|
/* shrink to base + N * inc
|
|
|
|
*/
|
|
|
|
width = window->size_hints.base_width +
|
|
|
|
FLOOR (width - window->size_hints.base_width, window->size_hints.width_inc);
|
|
|
|
height = window->size_hints.base_height +
|
|
|
|
FLOOR (height - window->size_hints.base_height, window->size_hints.height_inc);
|
|
|
|
|
|
|
|
/* constrain aspect ratio, according to:
|
|
|
|
*
|
|
|
|
* width
|
|
|
|
* min_aspect <= -------- <= max_aspect
|
|
|
|
* height
|
|
|
|
*/
|
|
|
|
|
|
|
|
min_aspect = window->size_hints.min_aspect.x / (double) window->size_hints.min_aspect.y;
|
|
|
|
max_aspect = window->size_hints.max_aspect.x / (double) window->size_hints.max_aspect.y;
|
|
|
|
|
|
|
|
if (min_aspect * height > width)
|
|
|
|
{
|
|
|
|
delta = FLOOR (height - width * min_aspect, window->size_hints.height_inc);
|
|
|
|
if (height - delta >= window->size_hints.min_height)
|
|
|
|
height -= delta;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delta = FLOOR (height * min_aspect - width, window->size_hints.width_inc);
|
|
|
|
if (width + delta <= window->size_hints.max_width)
|
|
|
|
width += delta;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (max_aspect * height < width)
|
|
|
|
{
|
|
|
|
delta = FLOOR (width - height * max_aspect, window->size_hints.width_inc);
|
|
|
|
if (width - delta >= window->size_hints.min_width)
|
|
|
|
width -= delta;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delta = FLOOR (width / max_aspect - height, window->size_hints.height_inc);
|
|
|
|
if (height + delta <= window->size_hints.max_height)
|
|
|
|
height += delta;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef FLOOR
|
|
|
|
|
|
|
|
*new_width = width;
|
|
|
|
*new_height = height;
|
|
|
|
}
|