mutter/src/frame.c

375 lines
10 KiB
C
Raw Normal View History

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 | \
FocusChangeMask)
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;
2001-06-10 23:24:20 -04:00
g_return_if_fail (window->display->server_grab_count > 0);
2001-06-02 21:33:27 -04:00
if (window->frame)
return;
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-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,
window->depth,
InputOutput,
window->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);
2001-06-04 02:17:52 -04:00
window->mapped = FALSE; /* the reparent will unmap the window,
* we don't want to take that as a withdraw
*/
2001-06-10 23:24:20 -04:00
window->unmaps_pending += 1;
2001-06-11 01:47:51 -04:00
window->rect.x = 0;
window->rect.y = 0;
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-05-31 02:42:58 -04:00
meta_error_trap_pop (window->display);
2001-06-17 23:24:25 -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);
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);
2001-06-04 02:17:52 -04:00
window->mapped = FALSE; /* Keep track of unmapping it, so we
* can identify a withdraw initiated
* by the client.
*/
2001-06-10 23:24:20 -04: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;
if (frame->window->has_resize_func)
flags |= META_FRAME_ALLOWS_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;
}
static void
set_background_none (MetaFrame *frame)
{
XSetWindowAttributes attrs;
attrs.background_pixmap = None;
XChangeWindowAttributes (frame->window->display->xdisplay,
frame->xwindow,
CWBackPixmap,
&attrs);
}
2001-06-02 21:33:27 -04:00
void
2001-06-09 23:17:15 -04:00
meta_frame_sync_to_window (MetaFrame *frame,
gboolean need_move,
gboolean need_resize)
2001-06-02 21:33:27 -04:00
{
2001-06-18 02:11:53 -04:00
meta_verbose ("Syncing frame geometry %d,%d %dx%d (SE: %d,%d)\n",
2001-06-09 17:58:30 -04:00
frame->rect.x, frame->rect.y,
frame->rect.width, frame->rect.height,
2001-06-10 23:24:20 -04:00
frame->rect.x + frame->rect.width,
2001-06-18 02:11:53 -04:00
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)
set_background_none (frame);
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)
meta_ui_reset_frame_bg (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
}
2001-05-31 02:42:58 -04:00
gboolean
meta_frame_event (MetaFrame *frame,
XEvent *event)
2001-06-02 00:14:18 -04:00
{
2001-05-31 02:42:58 -04:00
switch (event->type)
{
case KeyPress:
case KeyRelease:
2001-06-23 23:18:10 -04:00
meta_display_process_key_event (frame->window->display,
frame->window, event);
2001-05-31 02:42:58 -04:00
break;
case ButtonPress:
break;
case ButtonRelease:
break;
case MotionNotify:
break;
case EnterNotify:
2001-06-04 00:58:22 -04:00
/* We handle it here if a decorated window
* is involved, otherwise we handle it in display.c
*/
/* do this even if window->has_focus to avoid races */
meta_window_focus (frame->window,
event->xcrossing.time);
2001-05-31 02:42:58 -04:00
break;
case LeaveNotify:
break;
case FocusIn:
case FocusOut:
2001-06-23 01:49:35 -04:00
meta_window_notify_focus (frame->window,
event);
2001-05-31 02:42:58 -04:00
break;
case KeymapNotify:
break;
case Expose:
break;
case GraphicsExpose:
break;
case NoExpose:
break;
case VisibilityNotify:
break;
case CreateNotify:
break;
case DestroyNotify:
2001-06-02 21:33:27 -04:00
{
MetaDisplay *display;
meta_warning ("Unexpected destruction of frame 0x%lx, not sure if this should silently fail or be considered a bug\n", frame->xwindow);
display = frame->window->display;
meta_error_trap_push (display);
meta_window_destroy_frame (frame->window);
meta_error_trap_pop (display);
return TRUE;
}
2001-05-31 02:42:58 -04:00
break;
case UnmapNotify:
break;
case MapNotify:
break;
case MapRequest:
break;
case ReparentNotify:
break;
case ConfigureNotify:
break;
case ConfigureRequest:
break;
case GravityNotify:
break;
case ResizeRequest:
break;
case CirculateNotify:
break;
case CirculateRequest:
break;
case PropertyNotify:
break;
case SelectionClear:
break;
case SelectionRequest:
break;
case SelectionNotify:
break;
case ColormapNotify:
break;
case ClientMessage:
break;
case MappingNotify:
break;
default:
break;
}
return FALSE;
}