2001-05-31 02:42:58 -04:00
|
|
|
/* Metacity X window decorations */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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 "frame.h"
|
|
|
|
#include "errors.h"
|
2001-06-23 23:18:10 -04:00
|
|
|
#include "keybindings.h"
|
2001-06-03 17:39:57 -04:00
|
|
|
|
2001-06-19 23:01:26 -04:00
|
|
|
#define EVENT_MASK (SubstructureRedirectMask | \
|
|
|
|
StructureNotifyMask | SubstructureNotifyMask | \
|
|
|
|
ExposureMask | \
|
|
|
|
ButtonPressMask | ButtonReleaseMask | \
|
|
|
|
PointerMotionMask | PointerMotionHintMask | \
|
2001-06-23 01:49:35 -04:00
|
|
|
EnterWindowMask | LeaveWindowMask | \
|
2002-05-09 18:34:00 -04:00
|
|
|
FocusChangeMask | \
|
|
|
|
ColormapChangeMask)
|
2001-06-19 23:01:26 -04:00
|
|
|
|
2001-06-02 21:33:27 -04:00
|
|
|
void
|
|
|
|
meta_window_ensure_frame (MetaWindow *window)
|
|
|
|
{
|
|
|
|
MetaFrame *frame;
|
|
|
|
XSetWindowAttributes attrs;
|
|
|
|
|
|
|
|
if (window->frame)
|
|
|
|
return;
|
2002-01-02 12:42:14 -05:00
|
|
|
|
|
|
|
/* See comment below for why this is required. */
|
|
|
|
meta_display_grab (window->display);
|
2001-06-02 21:33:27 -04:00
|
|
|
|
|
|
|
frame = g_new (MetaFrame, 1);
|
|
|
|
|
|
|
|
frame->window = window;
|
|
|
|
frame->xwindow = None;
|
|
|
|
|
2001-06-09 17:58:30 -04:00
|
|
|
frame->rect = window->rect;
|
|
|
|
frame->child_x = 0;
|
|
|
|
frame->child_y = 0;
|
|
|
|
frame->bottom_height = 0;
|
|
|
|
frame->right_width = 0;
|
|
|
|
|
2001-06-09 23:17:15 -04:00
|
|
|
frame->mapped = FALSE;
|
|
|
|
|
2001-06-19 23:01:26 -04:00
|
|
|
attrs.event_mask = EVENT_MASK;
|
2001-09-08 23:44:42 -04:00
|
|
|
|
|
|
|
meta_verbose ("Framing window %s: visual %s default, depth %d default depth %d\n",
|
|
|
|
window->desc,
|
|
|
|
XVisualIDFromVisual (window->xvisual) ==
|
|
|
|
XVisualIDFromVisual (window->screen->default_xvisual) ?
|
|
|
|
"is" : "is not",
|
|
|
|
window->depth, window->screen->default_depth);
|
2002-04-22 14:18:38 -04:00
|
|
|
meta_verbose ("Frame geometry %d,%d %dx%d\n",
|
|
|
|
frame->rect.x, frame->rect.y,
|
|
|
|
frame->rect.width, frame->rect.height);
|
|
|
|
|
2001-09-08 23:44:42 -04:00
|
|
|
/* Default depth/visual handles clients with weird visuals; they can
|
|
|
|
* always be children of the root depth/visual obviously, but
|
|
|
|
* e.g. DRI games can't be children of a parent that has the same
|
|
|
|
* visual as the client.
|
|
|
|
*/
|
2001-05-31 02:42:58 -04:00
|
|
|
|
|
|
|
frame->xwindow = XCreateWindow (window->display->xdisplay,
|
|
|
|
window->screen->xroot,
|
|
|
|
frame->rect.x,
|
|
|
|
frame->rect.y,
|
|
|
|
frame->rect.width,
|
|
|
|
frame->rect.height,
|
|
|
|
0,
|
2001-09-08 23:44:42 -04:00
|
|
|
window->screen->default_depth,
|
|
|
|
CopyFromParent,
|
|
|
|
window->screen->default_xvisual,
|
2001-06-09 17:58:30 -04:00
|
|
|
CWEventMask,
|
2001-05-31 02:42:58 -04:00
|
|
|
&attrs);
|
2001-06-02 00:14:18 -04:00
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
/* So our UI can find the window ID */
|
|
|
|
XFlush (window->display->xdisplay);
|
|
|
|
|
2001-06-09 17:58:30 -04:00
|
|
|
meta_verbose ("Frame for %s is 0x%lx\n", frame->window->desc, frame->xwindow);
|
2001-06-02 00:14:18 -04:00
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_display_register_x_window (window->display, &frame->xwindow, window);
|
|
|
|
|
|
|
|
/* Reparent the client window; it may be destroyed,
|
|
|
|
* thus the error trap. We'll get a destroy notify later
|
|
|
|
* and free everything. Comment in FVWM source code says
|
2001-06-10 23:24:20 -04:00
|
|
|
* we need a server grab or the child can get its MapNotify
|
2001-05-31 02:42:58 -04:00
|
|
|
* before we've finished reparenting and getting the decoration
|
2001-06-10 23:24:20 -04:00
|
|
|
* window onscreen, so ensure_frame must be called with
|
|
|
|
* a grab.
|
2001-05-31 02:42:58 -04:00
|
|
|
*/
|
|
|
|
meta_error_trap_push (window->display);
|
2002-01-10 22:31:14 -05:00
|
|
|
if (window->mapped)
|
|
|
|
{
|
|
|
|
window->mapped = FALSE; /* the reparent will unmap the window,
|
|
|
|
* we don't want to take that as a withdraw
|
|
|
|
*/
|
2002-02-07 22:34:26 -05:00
|
|
|
meta_topic (META_DEBUG_WINDOW_STATE,
|
|
|
|
"Incrementing unmaps_pending on %s for reparent\n", window->desc);
|
2002-01-10 22:31:14 -05:00
|
|
|
window->unmaps_pending += 1;
|
|
|
|
}
|
2001-09-08 23:44:42 -04:00
|
|
|
/* window was reparented to this position */
|
2001-06-11 01:47:51 -04:00
|
|
|
window->rect.x = 0;
|
|
|
|
window->rect.y = 0;
|
2001-09-08 23:44:42 -04:00
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
XReparentWindow (window->display->xdisplay,
|
|
|
|
window->xwindow,
|
|
|
|
frame->xwindow,
|
2001-06-11 01:47:51 -04:00
|
|
|
window->rect.x,
|
|
|
|
window->rect.y);
|
2001-10-13 00:00:08 -04:00
|
|
|
/* FIXME handle this error */
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_error_trap_pop (window->display);
|
2001-09-08 23:44:42 -04:00
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
/* stick frame to the window */
|
2001-06-04 02:17:52 -04:00
|
|
|
window->frame = frame;
|
2001-06-19 23:01:26 -04:00
|
|
|
|
|
|
|
meta_ui_add_frame (window->screen->ui, frame->xwindow);
|
|
|
|
|
|
|
|
if (window->title)
|
|
|
|
meta_ui_set_frame_title (window->screen->ui,
|
|
|
|
window->frame->xwindow,
|
|
|
|
window->title);
|
2001-06-23 01:49:35 -04:00
|
|
|
|
|
|
|
/* Move keybindings to frame instead of window */
|
|
|
|
meta_window_grab_keys (window);
|
2002-01-02 12:42:14 -05:00
|
|
|
|
|
|
|
meta_display_ungrab (window->display);
|
2001-05-31 02:42:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_window_destroy_frame (MetaWindow *window)
|
|
|
|
{
|
|
|
|
MetaFrame *frame;
|
|
|
|
|
|
|
|
if (window->frame == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
frame = window->frame;
|
2001-06-23 01:49:35 -04:00
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
meta_ui_remove_frame (window->screen->ui, frame->xwindow);
|
2001-05-31 02:42:58 -04:00
|
|
|
|
|
|
|
/* Unparent the client window; it may be destroyed,
|
|
|
|
* thus the error trap.
|
|
|
|
*/
|
|
|
|
meta_error_trap_push (window->display);
|
2002-01-10 22:31:14 -05:00
|
|
|
if (window->mapped)
|
|
|
|
{
|
|
|
|
window->mapped = FALSE; /* Keep track of unmapping it, so we
|
|
|
|
* can identify a withdraw initiated
|
|
|
|
* by the client.
|
|
|
|
*/
|
2002-02-07 22:34:26 -05:00
|
|
|
meta_topic (META_DEBUG_WINDOW_STATE,
|
|
|
|
"Incrementing unmaps_pending on %s for reparent back to root\n", window->desc);
|
2002-01-10 22:31:14 -05:00
|
|
|
window->unmaps_pending += 1;
|
|
|
|
}
|
2001-05-31 02:42:58 -04:00
|
|
|
XReparentWindow (window->display->xdisplay,
|
|
|
|
window->xwindow,
|
|
|
|
window->screen->xroot,
|
|
|
|
/* FIXME where to put it back depends on the gravity */
|
|
|
|
window->frame->rect.x,
|
|
|
|
window->frame->rect.y);
|
|
|
|
meta_error_trap_pop (window->display);
|
|
|
|
|
|
|
|
meta_display_unregister_x_window (window->display,
|
|
|
|
frame->xwindow);
|
|
|
|
|
|
|
|
window->frame = NULL;
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
/* Move keybindings to window instead of frame */
|
|
|
|
meta_window_grab_keys (window);
|
|
|
|
|
2001-05-31 02:42:58 -04:00
|
|
|
/* should we push an error trap? */
|
|
|
|
XDestroyWindow (window->display->xdisplay, frame->xwindow);
|
|
|
|
|
|
|
|
g_free (frame);
|
2001-06-04 02:17:52 -04:00
|
|
|
|
|
|
|
/* Put our state back where it should be */
|
2001-06-06 00:47:37 -04:00
|
|
|
meta_window_queue_calc_showing (window);
|
2001-05-31 02:42:58 -04:00
|
|
|
}
|
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
|
|
|
|
MetaFrameFlags
|
|
|
|
meta_frame_get_flags (MetaFrame *frame)
|
|
|
|
{
|
|
|
|
MetaFrameFlags flags;
|
|
|
|
|
2001-06-23 14:30:27 -04:00
|
|
|
flags = META_FRAME_ALLOWS_MENU;
|
2001-06-20 23:40:14 -04:00
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
if (frame->window->has_close_func)
|
|
|
|
flags |= META_FRAME_ALLOWS_DELETE;
|
|
|
|
|
2001-06-23 02:54:28 -04:00
|
|
|
if (frame->window->has_maximize_func)
|
|
|
|
flags |= META_FRAME_ALLOWS_MAXIMIZE;
|
2001-06-17 23:24:25 -04:00
|
|
|
|
2001-06-23 02:54:28 -04:00
|
|
|
if (frame->window->has_minimize_func)
|
|
|
|
flags |= META_FRAME_ALLOWS_MINIMIZE;
|
2001-06-17 23:24:25 -04:00
|
|
|
|
2001-06-23 02:54:28 -04:00
|
|
|
if (frame->window->has_shade_func)
|
|
|
|
flags |= META_FRAME_ALLOWS_SHADE;
|
2001-06-23 14:30:27 -04:00
|
|
|
|
|
|
|
if (frame->window->has_move_func)
|
|
|
|
flags |= META_FRAME_ALLOWS_MOVE;
|
|
|
|
|
2001-07-05 01:22:00 -04:00
|
|
|
if (frame->window->has_resize_func &&
|
2002-01-19 18:59:11 -05:00
|
|
|
!frame->window->maximized &&
|
|
|
|
!frame->window->shaded)
|
2001-07-02 21:45:43 -04:00
|
|
|
{
|
|
|
|
if (frame->window->size_hints.min_width <
|
|
|
|
frame->window->size_hints.max_width)
|
|
|
|
flags |= META_FRAME_ALLOWS_HORIZONTAL_RESIZE;
|
|
|
|
|
|
|
|
if (frame->window->size_hints.min_height <
|
|
|
|
frame->window->size_hints.max_height)
|
|
|
|
flags |= META_FRAME_ALLOWS_VERTICAL_RESIZE;
|
|
|
|
}
|
2001-06-17 23:24:25 -04:00
|
|
|
|
|
|
|
if (frame->window->has_focus)
|
|
|
|
flags |= META_FRAME_HAS_FOCUS;
|
|
|
|
|
|
|
|
if (frame->window->shaded)
|
|
|
|
flags |= META_FRAME_SHADED;
|
|
|
|
|
|
|
|
if (frame->window->on_all_workspaces)
|
|
|
|
flags |= META_FRAME_STUCK;
|
2001-06-18 02:11:53 -04:00
|
|
|
|
2001-06-23 02:54:28 -04:00
|
|
|
if (frame->window->maximized)
|
|
|
|
flags |= META_FRAME_MAXIMIZED;
|
|
|
|
|
2001-06-18 02:11:53 -04:00
|
|
|
return flags;
|
2001-06-17 23:24:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_frame_calc_geometry (MetaFrame *frame,
|
|
|
|
MetaFrameGeometry *geomp)
|
|
|
|
{
|
|
|
|
MetaFrameGeometry geom;
|
|
|
|
MetaWindow *window;
|
|
|
|
|
|
|
|
window = frame->window;
|
|
|
|
|
|
|
|
meta_ui_get_frame_geometry (window->screen->ui,
|
|
|
|
frame->xwindow,
|
|
|
|
&geom.top_height,
|
|
|
|
&geom.bottom_height,
|
|
|
|
&geom.left_width,
|
|
|
|
&geom.right_width);
|
|
|
|
|
|
|
|
*geomp = geom;
|
|
|
|
}
|
|
|
|
|
2001-06-02 21:33:27 -04:00
|
|
|
void
|
2001-06-09 23:17:15 -04:00
|
|
|
meta_frame_sync_to_window (MetaFrame *frame,
|
2002-01-19 18:59:11 -05:00
|
|
|
int resize_gravity,
|
2001-06-09 23:17:15 -04:00
|
|
|
gboolean need_move,
|
|
|
|
gboolean need_resize)
|
2001-06-02 21:33:27 -04:00
|
|
|
{
|
2001-09-10 22:57:05 -04:00
|
|
|
if (!(need_move || need_resize))
|
|
|
|
return;
|
|
|
|
|
2002-02-24 14:58:27 -05:00
|
|
|
meta_topic (META_DEBUG_GEOMETRY,
|
|
|
|
"Syncing frame geometry %d,%d %dx%d (SE: %d,%d)\n",
|
|
|
|
frame->rect.x, frame->rect.y,
|
|
|
|
frame->rect.width, frame->rect.height,
|
|
|
|
frame->rect.x + frame->rect.width,
|
|
|
|
frame->rect.y + frame->rect.height);
|
2001-06-09 23:17:15 -04:00
|
|
|
|
|
|
|
/* set bg to none to avoid flicker */
|
2001-06-17 23:24:25 -04:00
|
|
|
if (need_resize)
|
2002-01-27 00:27:11 -05:00
|
|
|
meta_ui_unflicker_frame_bg (frame->window->screen->ui,
|
|
|
|
frame->xwindow,
|
|
|
|
frame->rect.width,
|
|
|
|
frame->rect.height);
|
2001-06-09 23:17:15 -04:00
|
|
|
|
|
|
|
if (need_move && need_resize)
|
|
|
|
XMoveResizeWindow (frame->window->display->xdisplay,
|
|
|
|
frame->xwindow,
|
|
|
|
frame->rect.x,
|
|
|
|
frame->rect.y,
|
|
|
|
frame->rect.width,
|
|
|
|
frame->rect.height);
|
|
|
|
else if (need_move)
|
|
|
|
XMoveWindow (frame->window->display->xdisplay,
|
|
|
|
frame->xwindow,
|
|
|
|
frame->rect.x,
|
|
|
|
frame->rect.y);
|
|
|
|
else if (need_resize)
|
|
|
|
XResizeWindow (frame->window->display->xdisplay,
|
|
|
|
frame->xwindow,
|
|
|
|
frame->rect.width,
|
2001-06-17 23:24:25 -04:00
|
|
|
frame->rect.height);
|
2001-06-03 17:39:57 -04:00
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
if (need_resize)
|
2002-05-09 18:34:00 -04:00
|
|
|
{
|
|
|
|
meta_ui_reset_frame_bg (frame->window->screen->ui,
|
|
|
|
frame->xwindow);
|
|
|
|
|
|
|
|
/* If we're interactively resizing the frame, repaint
|
|
|
|
* it immediately so we don't start to lag.
|
|
|
|
*/
|
|
|
|
if (frame->window->display->grab_window ==
|
|
|
|
frame->window)
|
|
|
|
meta_ui_repaint_frame (frame->window->screen->ui,
|
|
|
|
frame->xwindow);
|
|
|
|
}
|
2001-06-03 17:39:57 -04:00
|
|
|
}
|
|
|
|
|
2001-06-02 21:33:27 -04:00
|
|
|
void
|
|
|
|
meta_frame_queue_draw (MetaFrame *frame)
|
|
|
|
{
|
2001-06-17 23:24:25 -04:00
|
|
|
meta_ui_queue_frame_draw (frame->window->screen->ui,
|
|
|
|
frame->xwindow);
|
2001-06-09 02:08:44 -04:00
|
|
|
}
|