2001-06-17 15:53:45 -04:00
|
|
|
/* Metacity interface for talking to GTK+ UI module */
|
|
|
|
|
|
|
|
/*
|
2002-08-10 12:47:43 -04:00
|
|
|
* Copyright (C) 2002 Havoc Pennington
|
|
|
|
* stock icon code Copyright (C) 2002 Jorn Baayen <jorn@nl.linux.org>
|
2001-06-17 15:53:45 -04:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2001-10-13 00:15:25 -04:00
|
|
|
#include <config.h>
|
2004-01-10 12:16:07 -05:00
|
|
|
#include "prefs.h"
|
2001-06-17 15:53:45 -04:00
|
|
|
#include "ui.h"
|
2001-06-17 23:24:25 -04:00
|
|
|
#include "frames.h"
|
2001-06-18 02:11:53 -04:00
|
|
|
#include "util.h"
|
2001-06-23 01:49:35 -04:00
|
|
|
#include "menu.h"
|
2001-08-31 02:13:07 -04:00
|
|
|
#include "core.h"
|
2002-02-06 22:07:56 -05:00
|
|
|
#include "theme.h"
|
2001-06-17 23:24:25 -04:00
|
|
|
|
2002-07-13 23:16:41 -04:00
|
|
|
#include "eggaccelerators.h"
|
|
|
|
|
2001-09-17 00:42:37 -04:00
|
|
|
#include "inlinepixbufs.h"
|
|
|
|
|
2002-02-23 17:58:02 -05:00
|
|
|
#include <pango/pangox.h>
|
|
|
|
|
2002-04-28 00:52:26 -04:00
|
|
|
#include <string.h>
|
|
|
|
|
2002-08-10 12:47:43 -04:00
|
|
|
static void meta_stock_icons_init (void);
|
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
struct _MetaUI
|
|
|
|
{
|
|
|
|
Display *xdisplay;
|
|
|
|
Screen *xscreen;
|
|
|
|
MetaFrames *frames;
|
|
|
|
};
|
2001-06-17 15:53:45 -04:00
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_init (int *argc, char ***argv)
|
|
|
|
{
|
2001-06-18 02:11:53 -04:00
|
|
|
if (!gtk_init_check (argc, argv))
|
2001-09-10 23:54:54 -04:00
|
|
|
meta_fatal ("Unable to open X display %s\n", XDisplayName (NULL));
|
2001-09-17 23:40:03 -04:00
|
|
|
|
2002-08-10 12:47:43 -04:00
|
|
|
meta_stock_icons_init ();
|
2001-06-17 15:53:45 -04:00
|
|
|
}
|
2001-06-17 23:24:25 -04:00
|
|
|
|
2001-06-19 23:01:26 -04:00
|
|
|
Display*
|
|
|
|
meta_ui_get_display (const char *name)
|
|
|
|
{
|
|
|
|
if (name == NULL)
|
|
|
|
return gdk_display;
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct _EventFunc EventFunc;
|
|
|
|
|
|
|
|
struct _EventFunc
|
|
|
|
{
|
|
|
|
MetaEventFunc func;
|
|
|
|
gpointer data;
|
2003-11-24 13:09:47 -05:00
|
|
|
int last_event_serial;
|
2001-06-19 23:01:26 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
static GdkFilterReturn
|
|
|
|
filter_func (GdkXEvent *xevent,
|
|
|
|
GdkEvent *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
EventFunc *ef;
|
|
|
|
|
|
|
|
ef = data;
|
|
|
|
|
|
|
|
if ((* ef->func) (xevent, ef->data))
|
|
|
|
return GDK_FILTER_REMOVE;
|
|
|
|
else
|
2003-11-24 13:09:47 -05:00
|
|
|
{
|
|
|
|
ef->last_event_serial = ((XEvent*)xevent)->xany.serial;
|
|
|
|
return GDK_FILTER_CONTINUE;
|
|
|
|
}
|
2001-06-19 23:01:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static EventFunc *ef = NULL;
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_add_event_func (Display *xdisplay,
|
|
|
|
MetaEventFunc func,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
g_return_if_fail (ef == NULL);
|
|
|
|
|
|
|
|
ef = g_new (EventFunc, 1);
|
|
|
|
ef->func = func;
|
|
|
|
ef->data = data;
|
|
|
|
|
|
|
|
gdk_window_add_filter (NULL, filter_func, ef);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* removal is by data due to proxy function */
|
|
|
|
void
|
|
|
|
meta_ui_remove_event_func (Display *xdisplay,
|
|
|
|
MetaEventFunc func,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
g_return_if_fail (ef != NULL);
|
|
|
|
|
|
|
|
gdk_window_remove_filter (NULL, filter_func, ef);
|
|
|
|
|
|
|
|
g_free (ef);
|
|
|
|
ef = NULL;
|
|
|
|
}
|
|
|
|
|
2003-11-24 13:09:47 -05:00
|
|
|
int
|
|
|
|
meta_ui_get_last_event_serial (Display *xdisplay)
|
|
|
|
{
|
|
|
|
g_assert (ef != NULL);
|
|
|
|
|
|
|
|
/* This is technically broken since it's not per-display */
|
|
|
|
|
|
|
|
return ef->last_event_serial;
|
|
|
|
}
|
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
MetaUI*
|
|
|
|
meta_ui_new (Display *xdisplay,
|
|
|
|
Screen *screen)
|
|
|
|
{
|
|
|
|
MetaUI *ui;
|
|
|
|
|
|
|
|
ui = g_new (MetaUI, 1);
|
|
|
|
ui->xdisplay = xdisplay;
|
|
|
|
ui->xscreen = screen;
|
|
|
|
|
2002-05-08 12:24:28 -04:00
|
|
|
g_assert (xdisplay == gdk_display);
|
|
|
|
ui->frames = meta_frames_new (XScreenNumberOfScreen (screen));
|
2001-06-17 23:24:25 -04:00
|
|
|
gtk_widget_realize (GTK_WIDGET (ui->frames));
|
|
|
|
|
|
|
|
return ui;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_free (MetaUI *ui)
|
|
|
|
{
|
|
|
|
gtk_widget_destroy (GTK_WIDGET (ui->frames));
|
|
|
|
|
|
|
|
g_free (ui);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_get_frame_geometry (MetaUI *ui,
|
|
|
|
Window frame_xwindow,
|
|
|
|
int *top_height, int *bottom_height,
|
|
|
|
int *left_width, int *right_width)
|
|
|
|
{
|
|
|
|
meta_frames_get_geometry (ui->frames, frame_xwindow,
|
|
|
|
top_height, bottom_height,
|
|
|
|
left_width, right_width);
|
|
|
|
}
|
|
|
|
|
2003-08-16 12:32:10 -04:00
|
|
|
Window
|
|
|
|
meta_ui_create_frame_window (MetaUI *ui,
|
2003-11-15 23:19:18 -05:00
|
|
|
Display *xdisplay,
|
|
|
|
Visual *xvisual,
|
2003-08-16 12:32:10 -04:00
|
|
|
gint x,
|
|
|
|
gint y,
|
|
|
|
gint width,
|
|
|
|
gint height,
|
|
|
|
gint screen_no)
|
|
|
|
{
|
|
|
|
GdkDisplay *display = gdk_x11_lookup_xdisplay (xdisplay);
|
|
|
|
GdkScreen *screen = gdk_display_get_screen (display, screen_no);
|
|
|
|
GdkWindowAttr attrs;
|
|
|
|
gint attributes_mask;
|
|
|
|
GdkWindow *window;
|
2003-11-15 23:19:18 -05:00
|
|
|
GdkVisual *visual;
|
|
|
|
GdkColormap *cmap = gdk_screen_get_default_colormap (screen);
|
2003-08-16 12:32:10 -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.
|
|
|
|
*/
|
2003-11-15 23:19:18 -05:00
|
|
|
if (!xvisual)
|
|
|
|
visual = gdk_screen_get_system_visual (screen);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
visual = gdk_x11_screen_lookup_visual (screen,
|
|
|
|
XVisualIDFromVisual (xvisual));
|
|
|
|
cmap = gdk_colormap_new (visual, FALSE);
|
|
|
|
}
|
2003-08-16 12:32:10 -04:00
|
|
|
|
|
|
|
attrs.title = NULL;
|
|
|
|
|
|
|
|
/* frame.c is going to replace the event mask immediately, but
|
|
|
|
* we still have to set it here to let GDK know what it is.
|
|
|
|
*/
|
|
|
|
attrs.event_mask =
|
|
|
|
GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK |
|
|
|
|
GDK_POINTER_MOTION_MASK | GDK_POINTER_MOTION_HINT_MASK |
|
|
|
|
GDK_ENTER_NOTIFY_MASK | GDK_LEAVE_NOTIFY_MASK | GDK_FOCUS_CHANGE_MASK;
|
|
|
|
attrs.x = x;
|
|
|
|
attrs.y = y;
|
|
|
|
attrs.wclass = GDK_INPUT_OUTPUT;
|
2003-11-15 23:19:18 -05:00
|
|
|
attrs.visual = visual;
|
|
|
|
attrs.colormap = cmap;
|
2003-08-16 12:32:10 -04:00
|
|
|
attrs.window_type = GDK_WINDOW_CHILD;
|
|
|
|
attrs.cursor = NULL;
|
|
|
|
attrs.wmclass_name = NULL;
|
|
|
|
attrs.wmclass_class = NULL;
|
|
|
|
attrs.override_redirect = FALSE;
|
|
|
|
|
|
|
|
attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
|
|
|
|
|
|
|
|
window =
|
|
|
|
gdk_window_new (gdk_screen_get_root_window(screen),
|
|
|
|
&attrs, attributes_mask);
|
|
|
|
|
|
|
|
gdk_window_resize (window, width, height);
|
|
|
|
|
|
|
|
meta_frames_manage_window (ui->frames, GDK_WINDOW_XID (window), window);
|
|
|
|
|
|
|
|
return GDK_WINDOW_XID (window);
|
|
|
|
}
|
2001-06-17 23:24:25 -04:00
|
|
|
|
|
|
|
void
|
2003-08-16 12:32:10 -04:00
|
|
|
meta_ui_destroy_frame_window (MetaUI *ui,
|
|
|
|
Window xwindow)
|
2001-06-17 23:24:25 -04:00
|
|
|
{
|
2003-08-16 12:32:10 -04:00
|
|
|
meta_frames_unmanage_window (ui->frames, xwindow);
|
2001-06-17 23:24:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2003-08-16 12:32:10 -04:00
|
|
|
meta_ui_move_resize_frame (MetaUI *ui,
|
|
|
|
Window frame,
|
|
|
|
int x,
|
|
|
|
int y,
|
|
|
|
int width,
|
|
|
|
int height)
|
2001-06-17 23:24:25 -04:00
|
|
|
{
|
2003-08-16 12:32:10 -04:00
|
|
|
meta_frames_move_resize_frame (ui->frames, frame, x, y, width, height);
|
2001-06-17 23:24:25 -04:00
|
|
|
}
|
|
|
|
|
2001-06-19 23:01:26 -04:00
|
|
|
void
|
|
|
|
meta_ui_map_frame (MetaUI *ui,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
GdkWindow *window;
|
|
|
|
|
|
|
|
window = gdk_xid_table_lookup (xwindow);
|
|
|
|
|
|
|
|
if (window)
|
2001-06-29 11:33:21 -04:00
|
|
|
gdk_window_show_unraised (window);
|
2001-06-19 23:01:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_unmap_frame (MetaUI *ui,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
GdkWindow *window;
|
|
|
|
|
|
|
|
window = gdk_xid_table_lookup (xwindow);
|
|
|
|
|
2001-08-30 00:01:38 -04:00
|
|
|
if (window)
|
2001-06-19 23:01:26 -04:00
|
|
|
gdk_window_hide (window);
|
|
|
|
}
|
|
|
|
|
2002-01-27 00:27:11 -05:00
|
|
|
void
|
|
|
|
meta_ui_unflicker_frame_bg (MetaUI *ui,
|
|
|
|
Window xwindow,
|
|
|
|
int target_width,
|
|
|
|
int target_height)
|
|
|
|
{
|
|
|
|
meta_frames_unflicker_bg (ui->frames, xwindow,
|
|
|
|
target_width, target_height);
|
|
|
|
}
|
|
|
|
|
2002-05-09 18:34:00 -04:00
|
|
|
void
|
|
|
|
meta_ui_repaint_frame (MetaUI *ui,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
meta_frames_repaint_frame (ui->frames, xwindow);
|
|
|
|
}
|
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
void
|
|
|
|
meta_ui_reset_frame_bg (MetaUI *ui,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
meta_frames_reset_bg (ui->frames, xwindow);
|
|
|
|
}
|
|
|
|
|
2002-05-30 20:02:54 -04:00
|
|
|
void
|
2003-01-05 02:51:02 -05:00
|
|
|
meta_ui_apply_frame_shape (MetaUI *ui,
|
|
|
|
Window xwindow,
|
|
|
|
int new_window_width,
|
|
|
|
int new_window_height,
|
|
|
|
gboolean window_has_shape)
|
2002-05-30 20:02:54 -04:00
|
|
|
{
|
2003-01-05 02:51:02 -05:00
|
|
|
meta_frames_apply_shapes (ui->frames, xwindow,
|
|
|
|
new_window_width, new_window_height,
|
|
|
|
window_has_shape);
|
2002-05-30 20:02:54 -04:00
|
|
|
}
|
|
|
|
|
2001-06-17 23:24:25 -04:00
|
|
|
void
|
|
|
|
meta_ui_queue_frame_draw (MetaUI *ui,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
meta_frames_queue_draw (ui->frames, xwindow);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-06-19 23:01:26 -04:00
|
|
|
void
|
|
|
|
meta_ui_set_frame_title (MetaUI *ui,
|
|
|
|
Window xwindow,
|
|
|
|
const char *title)
|
|
|
|
{
|
|
|
|
meta_frames_set_title (ui->frames, xwindow, title);
|
|
|
|
}
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
MetaWindowMenu*
|
|
|
|
meta_ui_window_menu_new (MetaUI *ui,
|
|
|
|
Window client_xwindow,
|
|
|
|
MetaMenuOp ops,
|
|
|
|
MetaMenuOp insensitive,
|
|
|
|
int active_workspace,
|
|
|
|
int n_workspaces,
|
|
|
|
MetaWindowMenuFunc func,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
return meta_window_menu_new (ui->frames,
|
|
|
|
ops, insensitive,
|
|
|
|
client_xwindow,
|
|
|
|
active_workspace,
|
|
|
|
n_workspaces,
|
|
|
|
func, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_window_menu_popup (MetaWindowMenu *menu,
|
|
|
|
int root_x,
|
|
|
|
int root_y,
|
|
|
|
int button,
|
|
|
|
guint32 timestamp)
|
|
|
|
{
|
|
|
|
meta_window_menu_popup (menu, root_x, root_y, button, timestamp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_window_menu_free (MetaWindowMenu *menu)
|
|
|
|
{
|
|
|
|
meta_window_menu_free (menu);
|
|
|
|
}
|
2001-06-17 23:24:25 -04:00
|
|
|
|
2001-08-06 03:58:49 -04:00
|
|
|
struct _MetaImageWindow
|
|
|
|
{
|
|
|
|
GtkWidget *window;
|
2002-03-05 00:05:00 -05:00
|
|
|
GdkPixmap *pixmap;
|
2001-08-06 03:58:49 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
MetaImageWindow*
|
2002-10-18 18:46:37 -04:00
|
|
|
meta_image_window_new (Display *xdisplay,
|
|
|
|
int screen_number,
|
|
|
|
int max_width,
|
|
|
|
int max_height)
|
2001-08-06 03:58:49 -04:00
|
|
|
{
|
|
|
|
MetaImageWindow *iw;
|
2003-04-21 19:37:42 -04:00
|
|
|
GdkDisplay *gdisplay;
|
|
|
|
GdkScreen *gscreen;
|
|
|
|
|
2001-08-06 03:58:49 -04:00
|
|
|
iw = g_new (MetaImageWindow, 1);
|
|
|
|
iw->window = gtk_window_new (GTK_WINDOW_POPUP);
|
2002-10-20 13:17:16 -04:00
|
|
|
|
2003-04-21 19:37:42 -04:00
|
|
|
gdisplay = gdk_x11_lookup_xdisplay (xdisplay);
|
|
|
|
gscreen = gdk_display_get_screen (gdisplay, screen_number);
|
|
|
|
|
|
|
|
gtk_window_set_screen (GTK_WINDOW (iw->window), gscreen);
|
2002-10-20 13:17:16 -04:00
|
|
|
|
2002-03-05 00:05:00 -05:00
|
|
|
gtk_widget_realize (iw->window);
|
|
|
|
iw->pixmap = gdk_pixmap_new (iw->window->window,
|
|
|
|
max_width, max_height,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
gtk_widget_set_size_request (iw->window, 1, 1);
|
|
|
|
gtk_widget_set_double_buffered (iw->window, FALSE);
|
|
|
|
gtk_widget_set_app_paintable (iw->window, TRUE);
|
2001-08-06 03:58:49 -04:00
|
|
|
|
|
|
|
return iw;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_image_window_free (MetaImageWindow *iw)
|
|
|
|
{
|
|
|
|
gtk_widget_destroy (iw->window);
|
2002-03-05 00:05:00 -05:00
|
|
|
g_object_unref (G_OBJECT (iw->pixmap));
|
2001-08-06 03:58:49 -04:00
|
|
|
g_free (iw);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_image_window_set_showing (MetaImageWindow *iw,
|
|
|
|
gboolean showing)
|
|
|
|
{
|
|
|
|
if (showing)
|
|
|
|
gtk_widget_show_all (iw->window);
|
|
|
|
else
|
2001-08-30 00:01:38 -04:00
|
|
|
{
|
|
|
|
gtk_widget_hide (iw->window);
|
|
|
|
meta_core_increment_event_serial (gdk_display);
|
|
|
|
}
|
2001-08-06 03:58:49 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-03-05 00:05:00 -05:00
|
|
|
meta_image_window_set (MetaImageWindow *iw,
|
|
|
|
GdkPixbuf *pixbuf,
|
|
|
|
int x,
|
|
|
|
int y)
|
2001-08-06 03:58:49 -04:00
|
|
|
{
|
2002-03-05 00:05:00 -05:00
|
|
|
/* We use a back pixmap to avoid having to handle exposes, because
|
|
|
|
* it's really too slow for large clients being minimized, etc.
|
|
|
|
* and this way flicker is genuinely zero.
|
2001-08-18 21:58:00 -04:00
|
|
|
*/
|
|
|
|
|
2002-03-05 00:05:00 -05:00
|
|
|
gdk_pixbuf_render_to_drawable (pixbuf,
|
|
|
|
iw->pixmap,
|
|
|
|
iw->window->style->black_gc,
|
|
|
|
0, 0,
|
|
|
|
0, 0,
|
|
|
|
gdk_pixbuf_get_width (pixbuf),
|
|
|
|
gdk_pixbuf_get_height (pixbuf),
|
|
|
|
GDK_RGB_DITHER_NORMAL,
|
|
|
|
0, 0);
|
2001-08-18 21:58:00 -04:00
|
|
|
|
2002-03-05 00:05:00 -05:00
|
|
|
gdk_window_set_back_pixmap (iw->window->window,
|
|
|
|
iw->pixmap,
|
|
|
|
FALSE);
|
2001-08-18 21:58:00 -04:00
|
|
|
|
2002-03-05 00:05:00 -05:00
|
|
|
gdk_window_move_resize (iw->window->window,
|
|
|
|
x, y,
|
|
|
|
gdk_pixbuf_get_width (pixbuf),
|
|
|
|
gdk_pixbuf_get_height (pixbuf));
|
|
|
|
|
|
|
|
gdk_window_clear (iw->window->window);
|
2001-08-06 03:58:49 -04:00
|
|
|
}
|
|
|
|
|
2001-08-23 20:32:17 -04:00
|
|
|
static GdkColormap*
|
|
|
|
get_cmap (GdkPixmap *pixmap)
|
|
|
|
{
|
|
|
|
GdkColormap *cmap;
|
|
|
|
|
|
|
|
cmap = gdk_drawable_get_colormap (pixmap);
|
|
|
|
if (cmap)
|
|
|
|
g_object_ref (G_OBJECT (cmap));
|
|
|
|
|
|
|
|
if (cmap == NULL)
|
|
|
|
{
|
|
|
|
if (gdk_drawable_get_depth (pixmap) == 1)
|
|
|
|
{
|
2001-08-25 22:09:53 -04:00
|
|
|
meta_verbose ("Using NULL colormap for snapshotting bitmap\n");
|
|
|
|
cmap = NULL;
|
2001-08-23 20:32:17 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
meta_verbose ("Using system cmap to snapshot pixmap\n");
|
2002-11-14 23:10:54 -05:00
|
|
|
cmap = gdk_screen_get_system_colormap (gdk_drawable_get_screen (pixmap));
|
2003-04-21 19:37:42 -04:00
|
|
|
|
2001-08-23 20:32:17 -04:00
|
|
|
g_object_ref (G_OBJECT (cmap));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-04-13 16:26:45 -04:00
|
|
|
/* Be sure we aren't going to blow up due to visual mismatch */
|
|
|
|
if (cmap &&
|
|
|
|
(gdk_colormap_get_visual (cmap)->depth !=
|
|
|
|
gdk_drawable_get_depth (pixmap)))
|
|
|
|
{
|
|
|
|
cmap = NULL;
|
|
|
|
meta_verbose ("Switching back to NULL cmap because of depth mismatch\n");
|
|
|
|
}
|
|
|
|
|
2001-08-23 20:32:17 -04:00
|
|
|
return cmap;
|
|
|
|
}
|
|
|
|
|
2001-08-06 03:58:49 -04:00
|
|
|
GdkPixbuf*
|
|
|
|
meta_gdk_pixbuf_get_from_window (GdkPixbuf *dest,
|
|
|
|
Window xwindow,
|
|
|
|
int src_x,
|
|
|
|
int src_y,
|
|
|
|
int dest_x,
|
|
|
|
int dest_y,
|
|
|
|
int width,
|
|
|
|
int height)
|
|
|
|
{
|
|
|
|
GdkDrawable *drawable;
|
|
|
|
GdkPixbuf *retval;
|
2001-08-23 20:32:17 -04:00
|
|
|
GdkColormap *cmap;
|
|
|
|
|
2001-08-06 03:58:49 -04:00
|
|
|
retval = NULL;
|
|
|
|
|
|
|
|
drawable = gdk_xid_table_lookup (xwindow);
|
|
|
|
|
|
|
|
if (drawable)
|
|
|
|
g_object_ref (G_OBJECT (drawable));
|
|
|
|
else
|
|
|
|
drawable = gdk_window_foreign_new (xwindow);
|
2001-08-23 20:32:17 -04:00
|
|
|
|
|
|
|
cmap = get_cmap (drawable);
|
|
|
|
|
2001-08-06 03:58:49 -04:00
|
|
|
retval = gdk_pixbuf_get_from_drawable (dest,
|
|
|
|
drawable,
|
2001-08-23 20:32:17 -04:00
|
|
|
cmap,
|
2001-08-06 03:58:49 -04:00
|
|
|
src_x, src_y,
|
|
|
|
dest_x, dest_y,
|
|
|
|
width, height);
|
2001-08-23 20:32:17 -04:00
|
|
|
|
2001-08-25 22:09:53 -04:00
|
|
|
if (cmap)
|
|
|
|
g_object_unref (G_OBJECT (cmap));
|
2001-08-06 03:58:49 -04:00
|
|
|
g_object_unref (G_OBJECT (drawable));
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2001-08-19 21:42:44 -04:00
|
|
|
GdkPixbuf*
|
|
|
|
meta_gdk_pixbuf_get_from_pixmap (GdkPixbuf *dest,
|
|
|
|
Pixmap xpixmap,
|
|
|
|
int src_x,
|
|
|
|
int src_y,
|
|
|
|
int dest_x,
|
|
|
|
int dest_y,
|
|
|
|
int width,
|
|
|
|
int height)
|
|
|
|
{
|
|
|
|
GdkDrawable *drawable;
|
|
|
|
GdkPixbuf *retval;
|
2001-08-23 20:32:17 -04:00
|
|
|
GdkColormap *cmap;
|
|
|
|
|
2001-08-19 21:42:44 -04:00
|
|
|
retval = NULL;
|
2003-09-29 12:55:26 -04:00
|
|
|
cmap = NULL;
|
2001-08-19 21:42:44 -04:00
|
|
|
|
|
|
|
drawable = gdk_xid_table_lookup (xpixmap);
|
|
|
|
|
|
|
|
if (drawable)
|
|
|
|
g_object_ref (G_OBJECT (drawable));
|
|
|
|
else
|
|
|
|
drawable = gdk_pixmap_foreign_new (xpixmap);
|
2001-08-23 20:32:17 -04:00
|
|
|
|
2003-08-29 21:48:31 -04:00
|
|
|
if (drawable)
|
|
|
|
{
|
|
|
|
cmap = get_cmap (drawable);
|
2001-08-23 20:32:17 -04:00
|
|
|
|
2003-08-29 21:48:31 -04:00
|
|
|
retval = gdk_pixbuf_get_from_drawable (dest,
|
|
|
|
drawable,
|
|
|
|
cmap,
|
|
|
|
src_x, src_y,
|
|
|
|
dest_x, dest_y,
|
|
|
|
width, height);
|
|
|
|
}
|
2001-08-25 22:09:53 -04:00
|
|
|
if (cmap)
|
|
|
|
g_object_unref (G_OBJECT (cmap));
|
2003-08-29 21:48:31 -04:00
|
|
|
if (drawable)
|
|
|
|
g_object_unref (G_OBJECT (drawable));
|
2001-08-19 21:42:44 -04:00
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2001-08-18 21:19:54 -04:00
|
|
|
void
|
|
|
|
meta_ui_push_delay_exposes (MetaUI *ui)
|
|
|
|
{
|
|
|
|
meta_frames_push_delay_exposes (ui->frames);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_ui_pop_delay_exposes (MetaUI *ui)
|
|
|
|
{
|
|
|
|
meta_frames_pop_delay_exposes (ui->frames);
|
|
|
|
}
|
|
|
|
|
2001-10-13 00:15:25 -04:00
|
|
|
#ifdef HAVE_GDK_PIXBUF_NEW_FROM_STREAM
|
|
|
|
#define gdk_pixbuf_new_from_inline gdk_pixbuf_new_from_stream
|
|
|
|
#endif
|
|
|
|
|
2001-08-19 14:09:10 -04:00
|
|
|
GdkPixbuf*
|
|
|
|
meta_ui_get_default_window_icon (MetaUI *ui)
|
|
|
|
{
|
2001-09-17 00:42:37 -04:00
|
|
|
static GdkPixbuf *default_icon = NULL;
|
|
|
|
|
|
|
|
if (default_icon == NULL)
|
|
|
|
{
|
|
|
|
GdkPixbuf *base;
|
|
|
|
|
2001-10-11 20:20:02 -04:00
|
|
|
base = gdk_pixbuf_new_from_inline (-1, default_icon_data,
|
2001-09-17 00:42:37 -04:00
|
|
|
FALSE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_assert (base);
|
|
|
|
|
|
|
|
default_icon = gdk_pixbuf_scale_simple (base,
|
|
|
|
META_ICON_WIDTH,
|
|
|
|
META_ICON_HEIGHT,
|
|
|
|
GDK_INTERP_BILINEAR);
|
|
|
|
|
|
|
|
g_object_unref (G_OBJECT (base));
|
|
|
|
}
|
|
|
|
|
2001-09-17 01:50:02 -04:00
|
|
|
g_object_ref (G_OBJECT (default_icon));
|
|
|
|
|
2001-09-17 00:42:37 -04:00
|
|
|
return default_icon;
|
2001-09-15 20:30:45 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
GdkPixbuf*
|
|
|
|
meta_ui_get_default_mini_icon (MetaUI *ui)
|
|
|
|
{
|
2001-09-17 00:42:37 -04:00
|
|
|
static GdkPixbuf *default_icon = NULL;
|
|
|
|
|
|
|
|
if (default_icon == NULL)
|
|
|
|
{
|
|
|
|
GdkPixbuf *base;
|
|
|
|
|
2001-10-11 20:20:02 -04:00
|
|
|
base = gdk_pixbuf_new_from_inline (-1, default_icon_data,
|
2001-09-17 00:42:37 -04:00
|
|
|
FALSE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_assert (base);
|
|
|
|
|
|
|
|
default_icon = gdk_pixbuf_scale_simple (base,
|
|
|
|
META_MINI_ICON_WIDTH,
|
|
|
|
META_MINI_ICON_HEIGHT,
|
|
|
|
GDK_INTERP_BILINEAR);
|
|
|
|
|
|
|
|
g_object_unref (G_OBJECT (base));
|
|
|
|
}
|
|
|
|
|
2001-09-17 01:50:02 -04:00
|
|
|
g_object_ref (G_OBJECT (default_icon));
|
|
|
|
|
2001-09-17 00:42:37 -04:00
|
|
|
return default_icon;
|
2001-08-19 14:09:10 -04:00
|
|
|
}
|
2001-08-30 00:01:38 -04:00
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_ui_window_should_not_cause_focus (Display *xdisplay,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
GdkWindow *window;
|
|
|
|
|
|
|
|
window = gdk_xid_table_lookup (xwindow);
|
|
|
|
|
|
|
|
/* we shouldn't cause focus if we're an override redirect
|
|
|
|
* toplevel which is not foreign
|
|
|
|
*/
|
|
|
|
if (window && gdk_window_get_type (window) == GDK_WINDOW_TEMP)
|
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
2001-09-11 00:37:10 -04:00
|
|
|
|
|
|
|
char*
|
|
|
|
meta_text_property_to_utf8 (Display *xdisplay,
|
|
|
|
const XTextProperty *prop)
|
|
|
|
{
|
|
|
|
char **list;
|
|
|
|
int count;
|
|
|
|
char *retval;
|
|
|
|
|
|
|
|
list = NULL;
|
|
|
|
|
2001-12-06 19:54:32 -05:00
|
|
|
count = gdk_text_property_to_utf8_list (gdk_x11_xatom_to_atom (prop->encoding),
|
2001-09-11 00:37:10 -04:00
|
|
|
prop->format,
|
|
|
|
prop->value,
|
|
|
|
prop->nitems,
|
|
|
|
&list);
|
|
|
|
|
|
|
|
if (count == 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
retval = list[0];
|
|
|
|
list[0] = g_strdup (""); /* something to free */
|
|
|
|
|
|
|
|
g_strfreev (list);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
2002-02-06 22:07:56 -05:00
|
|
|
|
2004-01-10 12:16:07 -05:00
|
|
|
void
|
|
|
|
meta_ui_theme_get_frame_borders (MetaUI *ui,
|
|
|
|
MetaFrameType type,
|
|
|
|
MetaFrameFlags flags,
|
|
|
|
int *top_height,
|
|
|
|
int *bottom_height,
|
|
|
|
int *left_width,
|
|
|
|
int *right_width)
|
|
|
|
{
|
|
|
|
int text_height;
|
|
|
|
PangoContext *context;
|
|
|
|
const PangoFontDescription *font_desc;
|
|
|
|
GtkStyle *default_style;
|
|
|
|
|
|
|
|
if (meta_ui_have_a_theme ())
|
|
|
|
{
|
|
|
|
context = gtk_widget_get_pango_context (GTK_WIDGET (ui->frames));
|
|
|
|
font_desc = meta_prefs_get_titlebar_font ();
|
|
|
|
|
|
|
|
if (!font_desc)
|
|
|
|
{
|
|
|
|
default_style = gtk_widget_get_default_style ();
|
|
|
|
font_desc = default_style->font_desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
text_height = meta_pango_font_desc_get_text_height (font_desc, context);
|
|
|
|
|
|
|
|
meta_theme_get_frame_borders (meta_theme_get_current (),
|
|
|
|
type, text_height, flags,
|
|
|
|
top_height, bottom_height,
|
|
|
|
left_width, right_width);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*top_height = *bottom_height = *left_width = *right_width = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-02-06 22:07:56 -05:00
|
|
|
void
|
|
|
|
meta_ui_set_current_theme (const char *name,
|
|
|
|
gboolean force_reload)
|
|
|
|
{
|
|
|
|
meta_theme_set_current (name, force_reload);
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_ui_have_a_theme (void)
|
|
|
|
{
|
|
|
|
return meta_theme_get_current () != NULL;
|
|
|
|
}
|
2002-04-28 00:52:26 -04:00
|
|
|
|
|
|
|
gboolean
|
2002-07-13 23:16:41 -04:00
|
|
|
meta_ui_parse_accelerator (const char *accel,
|
|
|
|
unsigned int *keysym,
|
|
|
|
MetaVirtualModifier *mask)
|
2002-04-28 00:52:26 -04:00
|
|
|
{
|
2002-07-13 23:16:41 -04:00
|
|
|
EggVirtualModifierType gdk_mask = 0;
|
2002-04-28 00:52:26 -04:00
|
|
|
guint gdk_sym = 0;
|
|
|
|
|
|
|
|
*keysym = 0;
|
|
|
|
*mask = 0;
|
|
|
|
|
|
|
|
if (strcmp (accel, "disabled") == 0)
|
|
|
|
return TRUE;
|
|
|
|
|
2002-07-13 23:16:41 -04:00
|
|
|
if (!egg_accelerator_parse_virtual (accel, &gdk_sym, &gdk_mask))
|
|
|
|
return FALSE;
|
2002-04-28 00:52:26 -04:00
|
|
|
|
|
|
|
if (gdk_sym == None)
|
|
|
|
return FALSE;
|
|
|
|
|
2002-07-13 23:16:41 -04:00
|
|
|
if (gdk_mask & EGG_VIRTUAL_RELEASE_MASK) /* we don't allow this */
|
2002-04-28 00:52:26 -04:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
*keysym = gdk_sym;
|
|
|
|
|
2002-07-13 23:16:41 -04:00
|
|
|
if (gdk_mask & EGG_VIRTUAL_SHIFT_MASK)
|
|
|
|
*mask |= META_VIRTUAL_SHIFT_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_CONTROL_MASK)
|
2002-10-07 19:14:40 -04:00
|
|
|
*mask |= META_VIRTUAL_CONTROL_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_ALT_MASK)
|
|
|
|
*mask |= META_VIRTUAL_ALT_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD2_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD2_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD3_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD3_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD4_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD4_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD5_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD5_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_SUPER_MASK)
|
|
|
|
*mask |= META_VIRTUAL_SUPER_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_HYPER_MASK)
|
|
|
|
*mask |= META_VIRTUAL_HYPER_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_META_MASK)
|
|
|
|
*mask |= META_VIRTUAL_META_MASK;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_ui_parse_modifier (const char *accel,
|
|
|
|
MetaVirtualModifier *mask)
|
|
|
|
{
|
|
|
|
EggVirtualModifierType gdk_mask = 0;
|
|
|
|
guint gdk_sym = 0;
|
|
|
|
|
|
|
|
*mask = 0;
|
|
|
|
|
|
|
|
if (strcmp (accel, "disabled") == 0)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (!egg_accelerator_parse_virtual (accel, &gdk_sym, &gdk_mask))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (gdk_sym != None)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_RELEASE_MASK) /* we don't allow this */
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_SHIFT_MASK)
|
|
|
|
*mask |= META_VIRTUAL_SHIFT_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_CONTROL_MASK)
|
2002-07-13 23:16:41 -04:00
|
|
|
*mask |= META_VIRTUAL_CONTROL_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_ALT_MASK)
|
|
|
|
*mask |= META_VIRTUAL_ALT_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD2_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD2_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD3_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD3_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD4_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD4_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_MOD5_MASK)
|
|
|
|
*mask |= META_VIRTUAL_MOD5_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_SUPER_MASK)
|
|
|
|
*mask |= META_VIRTUAL_SUPER_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_HYPER_MASK)
|
|
|
|
*mask |= META_VIRTUAL_HYPER_MASK;
|
|
|
|
if (gdk_mask & EGG_VIRTUAL_META_MASK)
|
|
|
|
*mask |= META_VIRTUAL_META_MASK;
|
|
|
|
|
2002-04-28 00:52:26 -04:00
|
|
|
return TRUE;
|
|
|
|
}
|
2002-08-10 11:55:18 -04:00
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_ui_window_is_widget (MetaUI *ui,
|
|
|
|
Window xwindow)
|
|
|
|
{
|
|
|
|
GdkWindow *window;
|
|
|
|
|
|
|
|
window = gdk_xid_table_lookup (xwindow);
|
|
|
|
|
2003-08-16 12:32:10 -04:00
|
|
|
if (window)
|
2002-08-10 11:55:18 -04:00
|
|
|
{
|
|
|
|
void *user_data = NULL;
|
|
|
|
gdk_window_get_user_data (window, &user_data);
|
2003-08-16 12:32:10 -04:00
|
|
|
return user_data != NULL && user_data != ui->frames;
|
2002-08-10 11:55:18 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-08-10 12:47:43 -04:00
|
|
|
|
|
|
|
/* stock icon code Copyright (C) 2002 Jorn Baayen <jorn@nl.linux.org> */
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
char *stock_id;
|
|
|
|
const guint8 *icon_data;
|
|
|
|
} MetaStockIcon;
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_stock_icons_init (void)
|
|
|
|
{
|
|
|
|
GtkIconFactory *factory;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
MetaStockIcon items[] =
|
|
|
|
{
|
|
|
|
{ METACITY_STOCK_DELETE, stock_delete_data },
|
|
|
|
{ METACITY_STOCK_MINIMIZE, stock_minimize_data },
|
|
|
|
{ METACITY_STOCK_MAXIMIZE, stock_maximize_data }
|
|
|
|
};
|
|
|
|
|
|
|
|
factory = gtk_icon_factory_new ();
|
|
|
|
gtk_icon_factory_add_default (factory);
|
|
|
|
|
|
|
|
for (i = 0; i < (gint) G_N_ELEMENTS (items); i++)
|
|
|
|
{
|
|
|
|
GtkIconSet *icon_set;
|
|
|
|
GdkPixbuf *pixbuf;
|
|
|
|
|
|
|
|
pixbuf = gdk_pixbuf_new_from_inline (-1, items[i].icon_data,
|
|
|
|
FALSE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
icon_set = gtk_icon_set_new_from_pixbuf (pixbuf);
|
|
|
|
gtk_icon_factory_add (factory, items[i].stock_id, icon_set);
|
|
|
|
gtk_icon_set_unref (icon_set);
|
2003-08-16 12:32:10 -04:00
|
|
|
|
2002-08-10 12:47:43 -04:00
|
|
|
g_object_unref (G_OBJECT (pixbuf));
|
|
|
|
}
|
|
|
|
|
|
|
|
g_object_unref (G_OBJECT (factory));
|
|
|
|
}
|
2003-05-29 22:53:42 -04:00
|
|
|
|
|
|
|
int
|
2003-06-12 01:55:06 -04:00
|
|
|
meta_ui_get_double_click_timeout (MetaUI *ui)
|
2003-05-29 22:53:42 -04:00
|
|
|
{
|
|
|
|
GtkSettings *settings;
|
2003-06-12 01:55:06 -04:00
|
|
|
int timeout;
|
2003-05-29 22:53:42 -04:00
|
|
|
|
2003-06-12 01:55:06 -04:00
|
|
|
settings = gtk_widget_get_settings (GTK_WIDGET (ui->frames));
|
2003-05-29 22:53:42 -04:00
|
|
|
|
2003-06-12 01:55:06 -04:00
|
|
|
timeout = 250;
|
2003-05-29 22:53:42 -04:00
|
|
|
g_object_get (G_OBJECT (settings), "gtk-double-click-time", &timeout, NULL);
|
|
|
|
|
|
|
|
return timeout;
|
|
|
|
}
|
2003-06-12 01:55:06 -04:00
|
|
|
|
|
|
|
int
|
|
|
|
meta_ui_get_drag_threshold (MetaUI *ui)
|
|
|
|
{
|
|
|
|
GtkSettings *settings;
|
|
|
|
int threshold;
|
|
|
|
|
|
|
|
settings = gtk_widget_get_settings (GTK_WIDGET (ui->frames));
|
|
|
|
|
|
|
|
threshold = 8;
|
|
|
|
g_object_get (G_OBJECT (settings), "gtk-dnd-drag-threshold", &threshold, NULL);
|
|
|
|
|
|
|
|
return threshold;
|
|
|
|
}
|