2006-10-01 22:30:10 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
Comprehensively rename to Mutter
Code:
All references in the code not related to themes, keybindings, or
GConf were changed from 'metacity' to 'mutter'. This includes, among other
things, strings, comments, the atoms used in the message protocol, and
the envvars used for debugging. The GConf schema file was reduced to
the 3 settings new to mutter.
The overall version was brought up to 2.27 to match current gnome.
Structure:
All files named '*metacity*' were renamed '*mutter*' with appropriate
changes in the automake system. Files removed are
doc/creating_themes, src/themes, doc/metacity-theme.dtd,
metacity.doap. These files will eventually end up in an external
gnome-wm-data module.
Installation location:
On the filesystem the mutter-plugindir was change from
$(libdir)/metacity/plugins/clutter to just $(libdir)/mutter/plugins.
The mutter-plugins.pc.in reflects these changes.
Note:
mutter.desktop.in and mutter-wm.desktop both continue to have
X-GNOME-WMSettingsModule=metacity set. This allows
gnome-control-center to continue using libmetacity.so for
configuration. This is fine since most the general keybindings and wm
settings are being read from /apps/metacity/* in gconf.
2009-06-10 10:29:20 +00:00
|
|
|
/* Mutter window icons */
|
2002-02-09 23:03:52 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2002 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
|
2014-01-12 01:42:06 +00:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2002-02-09 23:03:52 +00:00
|
|
|
*/
|
|
|
|
|
2002-10-27 03:03:32 +00:00
|
|
|
#include <config.h>
|
2002-02-09 23:03:52 +00:00
|
|
|
#include "iconcache.h"
|
|
|
|
#include "ui.h"
|
2011-03-06 00:29:12 +00:00
|
|
|
#include <meta/errors.h>
|
2002-02-09 23:03:52 +00:00
|
|
|
|
|
|
|
#include <X11/Xatom.h>
|
|
|
|
|
|
|
|
/* The icon-reading code is also in libwnck, please sync bugfixes */
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
find_largest_sizes (gulong *data,
|
2004-04-19 14:53:14 +00:00
|
|
|
gulong nitems,
|
2002-02-09 23:03:52 +00:00
|
|
|
int *width,
|
|
|
|
int *height)
|
|
|
|
{
|
|
|
|
*width = 0;
|
|
|
|
*height = 0;
|
|
|
|
|
|
|
|
while (nitems > 0)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
|
|
|
|
if (nitems < 3)
|
|
|
|
return FALSE; /* no space for w, h */
|
|
|
|
|
|
|
|
w = data[0];
|
|
|
|
h = data[1];
|
|
|
|
|
A load of fixes of issues reported by sparse. Closes bug #152849
2004-09-17 Kjartan Maraas <kmaraas@gnome.org>
* src/bell.c: (meta_bell_flash_screen):
* src/compositor.c:
* src/effects.c: (meta_effects_draw_box_animation):
* src/fixedtip.c: (meta_fixed_tip_show):
* src/frame.c: (find_argb_visual):
* src/frames.c: (unsigned_long_hash), (meta_frames_manage_window),
(meta_frames_apply_shapes):
* src/iconcache.c: (find_largest_sizes), (find_best_size):
* src/keybindings.c: (meta_spawn_command_line_async_on_screen):
* src/main.c: (main):
* src/menu.c: (meta_window_menu_new):
* src/prefs.c: (meta_prefs_get_visual_bell),
(meta_prefs_bell_is_audible), (meta_prefs_get_visual_bell_type),
(meta_prefs_get_action_double_click_titlebar),
(meta_prefs_get_auto_raise), (meta_prefs_get_auto_raise_delay),
(meta_prefs_get_reduced_resources):
* src/screen.c: (meta_create_offscreen_window):
* src/tabpopup.c: (meta_ui_tab_popup_get_selected):
* src/theme-parser.c: (meta_theme_load):
* src/theme.c: (meta_gtk_widget_get_font_desc):
* src/tools/metacity-mag.c: (mouse_press), (begin_area_grab):
* src/util.c: (meta_unsigned_long_hash): A load of fixes of issues
reported by sparse. Closes bug #152849
2004-09-16 23:18:22 +00:00
|
|
|
if (nitems < ((gulong)(w * h) + 2))
|
2002-02-09 23:03:52 +00:00
|
|
|
return FALSE; /* not enough data */
|
|
|
|
|
|
|
|
*width = MAX (w, *width);
|
|
|
|
*height = MAX (h, *height);
|
|
|
|
|
|
|
|
data += (w * h) + 2;
|
|
|
|
nitems -= (w * h) + 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
find_best_size (gulong *data,
|
2004-04-19 14:53:14 +00:00
|
|
|
gulong nitems,
|
2002-02-09 23:03:52 +00:00
|
|
|
int ideal_width,
|
|
|
|
int ideal_height,
|
|
|
|
int *width,
|
|
|
|
int *height,
|
|
|
|
gulong **start)
|
|
|
|
{
|
|
|
|
int best_w;
|
|
|
|
int best_h;
|
|
|
|
gulong *best_start;
|
|
|
|
int max_width, max_height;
|
|
|
|
|
|
|
|
*width = 0;
|
|
|
|
*height = 0;
|
|
|
|
*start = NULL;
|
|
|
|
|
|
|
|
if (!find_largest_sizes (data, nitems, &max_width, &max_height))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (ideal_width < 0)
|
|
|
|
ideal_width = max_width;
|
|
|
|
if (ideal_height < 0)
|
|
|
|
ideal_height = max_height;
|
|
|
|
|
|
|
|
best_w = 0;
|
|
|
|
best_h = 0;
|
|
|
|
best_start = NULL;
|
|
|
|
|
|
|
|
while (nitems > 0)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
gboolean replace;
|
|
|
|
|
|
|
|
replace = FALSE;
|
|
|
|
|
|
|
|
if (nitems < 3)
|
|
|
|
return FALSE; /* no space for w, h */
|
|
|
|
|
|
|
|
w = data[0];
|
|
|
|
h = data[1];
|
|
|
|
|
A load of fixes of issues reported by sparse. Closes bug #152849
2004-09-17 Kjartan Maraas <kmaraas@gnome.org>
* src/bell.c: (meta_bell_flash_screen):
* src/compositor.c:
* src/effects.c: (meta_effects_draw_box_animation):
* src/fixedtip.c: (meta_fixed_tip_show):
* src/frame.c: (find_argb_visual):
* src/frames.c: (unsigned_long_hash), (meta_frames_manage_window),
(meta_frames_apply_shapes):
* src/iconcache.c: (find_largest_sizes), (find_best_size):
* src/keybindings.c: (meta_spawn_command_line_async_on_screen):
* src/main.c: (main):
* src/menu.c: (meta_window_menu_new):
* src/prefs.c: (meta_prefs_get_visual_bell),
(meta_prefs_bell_is_audible), (meta_prefs_get_visual_bell_type),
(meta_prefs_get_action_double_click_titlebar),
(meta_prefs_get_auto_raise), (meta_prefs_get_auto_raise_delay),
(meta_prefs_get_reduced_resources):
* src/screen.c: (meta_create_offscreen_window):
* src/tabpopup.c: (meta_ui_tab_popup_get_selected):
* src/theme-parser.c: (meta_theme_load):
* src/theme.c: (meta_gtk_widget_get_font_desc):
* src/tools/metacity-mag.c: (mouse_press), (begin_area_grab):
* src/util.c: (meta_unsigned_long_hash): A load of fixes of issues
reported by sparse. Closes bug #152849
2004-09-16 23:18:22 +00:00
|
|
|
if (nitems < ((gulong)(w * h) + 2))
|
2002-02-09 23:03:52 +00:00
|
|
|
break; /* not enough data */
|
|
|
|
|
|
|
|
if (best_start == NULL)
|
|
|
|
{
|
|
|
|
replace = TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* work with averages */
|
|
|
|
const int ideal_size = (ideal_width + ideal_height) / 2;
|
|
|
|
int best_size = (best_w + best_h) / 2;
|
|
|
|
int this_size = (w + h) / 2;
|
|
|
|
|
|
|
|
/* larger than desired is always better than smaller */
|
|
|
|
if (best_size < ideal_size &&
|
|
|
|
this_size >= ideal_size)
|
|
|
|
replace = TRUE;
|
|
|
|
/* if we have too small, pick anything bigger */
|
|
|
|
else if (best_size < ideal_size &&
|
|
|
|
this_size > best_size)
|
|
|
|
replace = TRUE;
|
|
|
|
/* if we have too large, pick anything smaller
|
|
|
|
* but still >= the ideal
|
|
|
|
*/
|
|
|
|
else if (best_size > ideal_size &&
|
|
|
|
this_size >= ideal_size &&
|
|
|
|
this_size < best_size)
|
|
|
|
replace = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (replace)
|
|
|
|
{
|
|
|
|
best_start = data + 2;
|
|
|
|
best_w = w;
|
|
|
|
best_h = h;
|
|
|
|
}
|
|
|
|
|
|
|
|
data += (w * h) + 2;
|
|
|
|
nitems -= (w * h) + 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (best_start)
|
|
|
|
{
|
|
|
|
*start = best_start;
|
|
|
|
*width = best_w;
|
|
|
|
*height = best_h;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
argbdata_to_pixdata (gulong *argb_data, int len, guchar **pixdata)
|
|
|
|
{
|
|
|
|
guchar *p;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
*pixdata = g_new (guchar, len * 4);
|
|
|
|
p = *pixdata;
|
|
|
|
|
|
|
|
/* One could speed this up a lot. */
|
|
|
|
i = 0;
|
|
|
|
while (i < len)
|
|
|
|
{
|
|
|
|
guint argb;
|
|
|
|
guint rgba;
|
|
|
|
|
|
|
|
argb = argb_data[i];
|
|
|
|
rgba = (argb << 8) | (argb >> 24);
|
|
|
|
|
|
|
|
*p = rgba >> 24;
|
|
|
|
++p;
|
|
|
|
*p = (rgba >> 16) & 0xff;
|
|
|
|
++p;
|
|
|
|
*p = (rgba >> 8) & 0xff;
|
|
|
|
++p;
|
|
|
|
*p = rgba & 0xff;
|
|
|
|
++p;
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
read_rgb_icon (MetaDisplay *display,
|
|
|
|
Window xwindow,
|
|
|
|
int ideal_width,
|
|
|
|
int ideal_height,
|
|
|
|
int ideal_mini_width,
|
|
|
|
int ideal_mini_height,
|
|
|
|
int *width,
|
|
|
|
int *height,
|
|
|
|
guchar **pixdata,
|
|
|
|
int *mini_width,
|
|
|
|
int *mini_height,
|
|
|
|
guchar **mini_pixdata)
|
|
|
|
{
|
|
|
|
Atom type;
|
|
|
|
int format;
|
|
|
|
gulong nitems;
|
|
|
|
gulong bytes_after;
|
|
|
|
int result, err;
|
2008-12-01 22:54:57 +00:00
|
|
|
guchar *data;
|
2002-02-09 23:03:52 +00:00
|
|
|
gulong *best;
|
|
|
|
int w, h;
|
|
|
|
gulong *best_mini;
|
|
|
|
int mini_w, mini_h;
|
2008-12-02 08:15:00 +00:00
|
|
|
gulong *data_as_long;
|
2002-02-09 23:03:52 +00:00
|
|
|
|
2014-04-06 19:52:44 +00:00
|
|
|
meta_error_trap_push (display);
|
2002-02-09 23:03:52 +00:00
|
|
|
type = None;
|
|
|
|
data = NULL;
|
|
|
|
result = XGetWindowProperty (display->xdisplay,
|
|
|
|
xwindow,
|
2008-05-02 18:49:01 +00:00
|
|
|
display->atom__NET_WM_ICON,
|
2002-02-09 23:03:52 +00:00
|
|
|
0, G_MAXLONG,
|
|
|
|
False, XA_CARDINAL, &type, &format, &nitems,
|
2008-12-01 22:54:57 +00:00
|
|
|
&bytes_after, &data);
|
2010-10-25 18:44:30 +00:00
|
|
|
err = meta_error_trap_pop_with_return (display);
|
2002-02-09 23:03:52 +00:00
|
|
|
|
|
|
|
if (err != Success ||
|
|
|
|
result != Success)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (type != XA_CARDINAL)
|
|
|
|
{
|
|
|
|
XFree (data);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2008-12-02 08:15:00 +00:00
|
|
|
data_as_long = (gulong *)data;
|
|
|
|
|
2008-12-01 22:54:57 +00:00
|
|
|
if (!find_best_size (data_as_long, nitems,
|
2002-02-09 23:03:52 +00:00
|
|
|
ideal_width, ideal_height,
|
|
|
|
&w, &h, &best))
|
|
|
|
{
|
|
|
|
XFree (data);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2008-12-01 22:54:57 +00:00
|
|
|
if (!find_best_size (data_as_long, nitems,
|
2002-02-09 23:03:52 +00:00
|
|
|
ideal_mini_width, ideal_mini_height,
|
|
|
|
&mini_w, &mini_h, &best_mini))
|
|
|
|
{
|
|
|
|
XFree (data);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*width = w;
|
|
|
|
*height = h;
|
|
|
|
|
|
|
|
*mini_width = mini_w;
|
|
|
|
*mini_height = mini_h;
|
|
|
|
|
|
|
|
argbdata_to_pixdata (best, w * h, pixdata);
|
|
|
|
argbdata_to_pixdata (best_mini, mini_w * mini_h, mini_pixdata);
|
|
|
|
|
|
|
|
XFree (data);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_pixels (guchar *pixels, gpointer data)
|
|
|
|
{
|
|
|
|
g_free (pixels);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_pixmap_geometry (MetaDisplay *display,
|
|
|
|
Pixmap pixmap,
|
|
|
|
int *w,
|
|
|
|
int *h,
|
|
|
|
int *d)
|
|
|
|
{
|
|
|
|
Window root_ignored;
|
|
|
|
int x_ignored, y_ignored;
|
|
|
|
guint width, height;
|
|
|
|
guint border_width_ignored;
|
|
|
|
guint depth;
|
|
|
|
|
|
|
|
if (w)
|
|
|
|
*w = 1;
|
|
|
|
if (h)
|
|
|
|
*h = 1;
|
|
|
|
if (d)
|
|
|
|
*d = 1;
|
|
|
|
|
|
|
|
XGetGeometry (display->xdisplay,
|
|
|
|
pixmap, &root_ignored, &x_ignored, &y_ignored,
|
|
|
|
&width, &height, &border_width_ignored, &depth);
|
|
|
|
|
|
|
|
if (w)
|
|
|
|
*w = width;
|
|
|
|
if (h)
|
|
|
|
*h = height;
|
|
|
|
if (d)
|
|
|
|
*d = depth;
|
|
|
|
}
|
|
|
|
|
2011-03-21 22:45:20 +00:00
|
|
|
static void
|
|
|
|
apply_foreground_background (GdkPixbuf *pixbuf)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
int i, j;
|
|
|
|
guchar *pixels;
|
|
|
|
int stride;
|
|
|
|
|
|
|
|
w = gdk_pixbuf_get_width (pixbuf);
|
|
|
|
h = gdk_pixbuf_get_height (pixbuf);
|
|
|
|
pixels = gdk_pixbuf_get_pixels (pixbuf);
|
|
|
|
stride = gdk_pixbuf_get_rowstride (pixbuf);
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
while (i < h)
|
|
|
|
{
|
|
|
|
j = 0;
|
|
|
|
while (j < w)
|
|
|
|
{
|
|
|
|
guchar *p = pixels + i * stride + j * 4;
|
|
|
|
if (p[3] == 0)
|
|
|
|
p[0] = p[1] = p[2] = 0xff; /* white background */
|
|
|
|
else
|
|
|
|
p[0] = p[1] = p[2] = 0x00; /* black foreground */
|
|
|
|
|
|
|
|
p[3] = 0xff;
|
|
|
|
|
|
|
|
++j;
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
static GdkPixbuf*
|
|
|
|
apply_mask (GdkPixbuf *pixbuf,
|
|
|
|
GdkPixbuf *mask)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
int i, j;
|
|
|
|
GdkPixbuf *with_alpha;
|
|
|
|
guchar *src;
|
|
|
|
guchar *dest;
|
|
|
|
int src_stride;
|
|
|
|
int dest_stride;
|
|
|
|
|
|
|
|
w = MIN (gdk_pixbuf_get_width (mask), gdk_pixbuf_get_width (pixbuf));
|
|
|
|
h = MIN (gdk_pixbuf_get_height (mask), gdk_pixbuf_get_height (pixbuf));
|
|
|
|
|
|
|
|
with_alpha = gdk_pixbuf_add_alpha (pixbuf, FALSE, 0, 0, 0);
|
|
|
|
|
|
|
|
dest = gdk_pixbuf_get_pixels (with_alpha);
|
|
|
|
src = gdk_pixbuf_get_pixels (mask);
|
|
|
|
|
|
|
|
dest_stride = gdk_pixbuf_get_rowstride (with_alpha);
|
|
|
|
src_stride = gdk_pixbuf_get_rowstride (mask);
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
while (i < h)
|
|
|
|
{
|
|
|
|
j = 0;
|
|
|
|
while (j < w)
|
|
|
|
{
|
2011-02-28 08:41:08 +00:00
|
|
|
guchar *s = src + i * src_stride + j * 4;
|
2002-02-09 23:03:52 +00:00
|
|
|
guchar *d = dest + i * dest_stride + j * 4;
|
|
|
|
|
2011-02-28 08:41:08 +00:00
|
|
|
d[3] = s[3];
|
2002-02-09 23:03:52 +00:00
|
|
|
|
|
|
|
++j;
|
|
|
|
}
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return with_alpha;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
try_pixmap_and_mask (MetaDisplay *display,
|
|
|
|
Pixmap src_pixmap,
|
|
|
|
Pixmap src_mask,
|
|
|
|
GdkPixbuf **iconp,
|
|
|
|
int ideal_width,
|
|
|
|
int ideal_height,
|
|
|
|
GdkPixbuf **mini_iconp,
|
|
|
|
int ideal_mini_width,
|
|
|
|
int ideal_mini_height)
|
|
|
|
{
|
|
|
|
GdkPixbuf *unscaled = NULL;
|
|
|
|
GdkPixbuf *mask = NULL;
|
2011-03-21 22:45:20 +00:00
|
|
|
int w, h, d;
|
2002-02-09 23:03:52 +00:00
|
|
|
|
|
|
|
if (src_pixmap == None)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
meta_error_trap_push (display);
|
|
|
|
|
2011-03-21 22:45:20 +00:00
|
|
|
get_pixmap_geometry (display, src_pixmap, &w, &h, &d);
|
2002-02-09 23:03:52 +00:00
|
|
|
|
2010-10-02 01:40:15 +00:00
|
|
|
unscaled = meta_gdk_pixbuf_get_from_pixmap (src_pixmap,
|
|
|
|
0, 0,
|
2002-02-09 23:03:52 +00:00
|
|
|
w, h);
|
|
|
|
|
2011-03-21 22:45:20 +00:00
|
|
|
/* A depth 1 pixmap has 0 background, and 1 foreground, but
|
|
|
|
* cairo and meta_gdk_pixbuf_get_from_pixmap consider it
|
|
|
|
* to be 0 transparent, 1 opaque */
|
|
|
|
if (d == 1)
|
|
|
|
apply_foreground_background (unscaled);
|
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
if (unscaled && src_mask != None)
|
|
|
|
{
|
2011-03-21 22:49:55 +00:00
|
|
|
get_pixmap_geometry (display, src_mask, &w, &h, &d);
|
|
|
|
if (d == 1)
|
|
|
|
mask = meta_gdk_pixbuf_get_from_pixmap (src_mask,
|
|
|
|
0, 0,
|
|
|
|
w, h);
|
2002-02-09 23:03:52 +00:00
|
|
|
}
|
|
|
|
|
2010-10-25 18:44:30 +00:00
|
|
|
meta_error_trap_pop (display);
|
2002-02-09 23:03:52 +00:00
|
|
|
|
|
|
|
if (mask)
|
|
|
|
{
|
|
|
|
GdkPixbuf *masked;
|
|
|
|
|
|
|
|
masked = apply_mask (unscaled, mask);
|
|
|
|
g_object_unref (G_OBJECT (unscaled));
|
|
|
|
unscaled = masked;
|
|
|
|
|
|
|
|
g_object_unref (G_OBJECT (mask));
|
|
|
|
mask = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unscaled)
|
|
|
|
{
|
|
|
|
*iconp =
|
|
|
|
gdk_pixbuf_scale_simple (unscaled,
|
|
|
|
ideal_width > 0 ? ideal_width :
|
|
|
|
gdk_pixbuf_get_width (unscaled),
|
|
|
|
ideal_height > 0 ? ideal_height :
|
|
|
|
gdk_pixbuf_get_height (unscaled),
|
|
|
|
GDK_INTERP_BILINEAR);
|
|
|
|
*mini_iconp =
|
|
|
|
gdk_pixbuf_scale_simple (unscaled,
|
|
|
|
ideal_mini_width > 0 ? ideal_mini_width :
|
|
|
|
gdk_pixbuf_get_width (unscaled),
|
|
|
|
ideal_mini_height > 0 ? ideal_mini_height :
|
|
|
|
gdk_pixbuf_get_height (unscaled),
|
|
|
|
GDK_INTERP_BILINEAR);
|
|
|
|
|
|
|
|
g_object_unref (G_OBJECT (unscaled));
|
2014-05-02 13:34:02 +00:00
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
if (*iconp && *mini_iconp)
|
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (*iconp)
|
|
|
|
g_object_unref (G_OBJECT (*iconp));
|
|
|
|
if (*mini_iconp)
|
|
|
|
g_object_unref (G_OBJECT (*mini_iconp));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_kwm_win_icon (MetaDisplay *display,
|
|
|
|
Window xwindow,
|
|
|
|
Pixmap *pixmap,
|
|
|
|
Pixmap *mask)
|
|
|
|
{
|
|
|
|
Atom type;
|
|
|
|
int format;
|
|
|
|
gulong nitems;
|
|
|
|
gulong bytes_after;
|
2008-12-01 22:54:57 +00:00
|
|
|
guchar *data;
|
2008-12-02 08:15:00 +00:00
|
|
|
Pixmap *icons;
|
2002-02-09 23:03:52 +00:00
|
|
|
int err, result;
|
|
|
|
|
|
|
|
*pixmap = None;
|
|
|
|
*mask = None;
|
|
|
|
|
2014-04-06 19:52:44 +00:00
|
|
|
meta_error_trap_push (display);
|
2002-02-09 23:03:52 +00:00
|
|
|
icons = NULL;
|
|
|
|
result = XGetWindowProperty (display->xdisplay, xwindow,
|
2008-05-02 18:49:01 +00:00
|
|
|
display->atom__KWM_WIN_ICON,
|
2002-02-09 23:03:52 +00:00
|
|
|
0, G_MAXLONG,
|
|
|
|
False,
|
2008-05-02 18:49:01 +00:00
|
|
|
display->atom__KWM_WIN_ICON,
|
2002-02-09 23:03:52 +00:00
|
|
|
&type, &format, &nitems,
|
2008-12-01 22:54:57 +00:00
|
|
|
&bytes_after, &data);
|
2008-12-02 08:15:00 +00:00
|
|
|
icons = (Pixmap *)data;
|
2002-02-09 23:03:52 +00:00
|
|
|
|
2010-10-25 18:44:30 +00:00
|
|
|
err = meta_error_trap_pop_with_return (display);
|
2002-02-09 23:03:52 +00:00
|
|
|
if (err != Success ||
|
|
|
|
result != Success)
|
|
|
|
return;
|
|
|
|
|
2008-05-02 18:49:01 +00:00
|
|
|
if (type != display->atom__KWM_WIN_ICON)
|
2002-02-09 23:03:52 +00:00
|
|
|
{
|
|
|
|
XFree (icons);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*pixmap = icons[0];
|
|
|
|
*mask = icons[1];
|
|
|
|
|
|
|
|
XFree (icons);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_icon_cache_init (MetaIconCache *icon_cache)
|
|
|
|
{
|
|
|
|
g_return_if_fail (icon_cache != NULL);
|
|
|
|
|
|
|
|
icon_cache->origin = USING_NO_ICON;
|
|
|
|
icon_cache->prev_pixmap = None;
|
2003-11-24 19:09:25 +00:00
|
|
|
icon_cache->prev_mask = None;
|
2002-02-09 23:03:52 +00:00
|
|
|
icon_cache->wm_hints_dirty = TRUE;
|
|
|
|
icon_cache->kwm_win_icon_dirty = TRUE;
|
|
|
|
icon_cache->net_wm_icon_dirty = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_icon_cache_property_changed (MetaIconCache *icon_cache,
|
|
|
|
MetaDisplay *display,
|
|
|
|
Atom atom)
|
|
|
|
{
|
2008-05-02 18:49:01 +00:00
|
|
|
if (atom == display->atom__NET_WM_ICON)
|
2002-02-09 23:03:52 +00:00
|
|
|
icon_cache->net_wm_icon_dirty = TRUE;
|
2008-05-02 18:49:01 +00:00
|
|
|
else if (atom == display->atom__KWM_WIN_ICON)
|
2002-02-09 23:03:52 +00:00
|
|
|
icon_cache->kwm_win_icon_dirty = TRUE;
|
|
|
|
else if (atom == XA_WM_HINTS)
|
|
|
|
icon_cache->wm_hints_dirty = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_icon_cache_get_icon_invalidated (MetaIconCache *icon_cache)
|
|
|
|
{
|
|
|
|
if (icon_cache->origin <= USING_KWM_WIN_ICON &&
|
|
|
|
icon_cache->kwm_win_icon_dirty)
|
|
|
|
return TRUE;
|
|
|
|
else if (icon_cache->origin <= USING_WM_HINTS &&
|
|
|
|
icon_cache->wm_hints_dirty)
|
|
|
|
return TRUE;
|
|
|
|
else if (icon_cache->origin <= USING_NET_WM_ICON &&
|
|
|
|
icon_cache->net_wm_icon_dirty)
|
|
|
|
return TRUE;
|
2014-07-14 15:23:18 +00:00
|
|
|
else if (icon_cache->origin < USING_FALLBACK_ICON)
|
2002-02-09 23:03:52 +00:00
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GdkPixbuf*
|
|
|
|
scaled_from_pixdata (guchar *pixdata,
|
|
|
|
int w,
|
|
|
|
int h,
|
|
|
|
int new_w,
|
|
|
|
int new_h)
|
|
|
|
{
|
|
|
|
GdkPixbuf *src;
|
|
|
|
GdkPixbuf *dest;
|
2014-05-02 13:34:02 +00:00
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
src = gdk_pixbuf_new_from_data (pixdata,
|
|
|
|
GDK_COLORSPACE_RGB,
|
|
|
|
TRUE,
|
|
|
|
8,
|
|
|
|
w, h, w * 4,
|
2014-05-02 13:34:02 +00:00
|
|
|
free_pixels,
|
2002-02-09 23:03:52 +00:00
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (src == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
2002-04-21 20:17:44 +00:00
|
|
|
if (w != h)
|
|
|
|
{
|
|
|
|
GdkPixbuf *tmp;
|
|
|
|
int size;
|
|
|
|
|
|
|
|
size = MAX (w, h);
|
2014-05-02 13:34:02 +00:00
|
|
|
|
2002-04-21 20:17:44 +00:00
|
|
|
tmp = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8, size, size);
|
|
|
|
|
|
|
|
if (tmp)
|
|
|
|
{
|
|
|
|
gdk_pixbuf_fill (tmp, 0);
|
|
|
|
gdk_pixbuf_copy_area (src, 0, 0, w, h,
|
|
|
|
tmp,
|
|
|
|
(size - w) / 2, (size - h) / 2);
|
2014-05-02 13:34:02 +00:00
|
|
|
|
2002-04-21 20:17:44 +00:00
|
|
|
g_object_unref (src);
|
|
|
|
src = tmp;
|
|
|
|
}
|
|
|
|
}
|
2014-05-02 13:34:02 +00:00
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
if (w != new_w || h != new_h)
|
|
|
|
{
|
|
|
|
dest = gdk_pixbuf_scale_simple (src, new_w, new_h, GDK_INTERP_BILINEAR);
|
2014-05-02 13:34:02 +00:00
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
g_object_unref (G_OBJECT (src));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dest = src;
|
|
|
|
}
|
|
|
|
|
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_read_icons (MetaScreen *screen,
|
|
|
|
Window xwindow,
|
|
|
|
MetaIconCache *icon_cache,
|
|
|
|
Pixmap wm_hints_pixmap,
|
|
|
|
Pixmap wm_hints_mask,
|
|
|
|
GdkPixbuf **iconp,
|
|
|
|
int ideal_width,
|
|
|
|
int ideal_height,
|
|
|
|
GdkPixbuf **mini_iconp,
|
|
|
|
int ideal_mini_width,
|
|
|
|
int ideal_mini_height)
|
|
|
|
{
|
|
|
|
/* Return value is whether the icon changed */
|
|
|
|
|
|
|
|
g_return_val_if_fail (icon_cache != NULL, FALSE);
|
|
|
|
|
|
|
|
*iconp = NULL;
|
|
|
|
*mini_iconp = NULL;
|
|
|
|
|
|
|
|
if (!meta_icon_cache_get_icon_invalidated (icon_cache))
|
|
|
|
return FALSE; /* we have no new info to use */
|
|
|
|
|
|
|
|
/* Our algorithm here assumes that we can't have for example origin
|
|
|
|
* < USING_NET_WM_ICON and icon_cache->net_wm_icon_dirty == FALSE
|
|
|
|
* unless we have tried to read NET_WM_ICON.
|
|
|
|
*
|
|
|
|
* Put another way, if an icon origin is not dirty, then we have
|
|
|
|
* tried to read it at the current size. If it is dirty, then
|
|
|
|
* we haven't done that since the last change.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (icon_cache->origin <= USING_NET_WM_ICON &&
|
|
|
|
icon_cache->net_wm_icon_dirty)
|
|
|
|
{
|
2014-07-14 15:29:54 +00:00
|
|
|
guchar *pixdata;
|
|
|
|
int w, h;
|
|
|
|
guchar *mini_pixdata;
|
|
|
|
int mini_w, mini_h;
|
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
icon_cache->net_wm_icon_dirty = FALSE;
|
|
|
|
|
|
|
|
if (read_rgb_icon (screen->display, xwindow,
|
|
|
|
ideal_width, ideal_height,
|
|
|
|
ideal_mini_width, ideal_mini_height,
|
|
|
|
&w, &h, &pixdata,
|
|
|
|
&mini_w, &mini_h, &mini_pixdata))
|
|
|
|
{
|
|
|
|
*iconp = scaled_from_pixdata (pixdata, w, h,
|
|
|
|
ideal_width, ideal_height);
|
|
|
|
|
|
|
|
*mini_iconp = scaled_from_pixdata (mini_pixdata, mini_w, mini_h,
|
|
|
|
ideal_mini_width, ideal_mini_height);
|
|
|
|
|
|
|
|
if (*iconp && *mini_iconp)
|
|
|
|
{
|
2014-07-14 15:27:01 +00:00
|
|
|
icon_cache->origin = USING_NET_WM_ICON;
|
2002-02-09 23:03:52 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (*iconp)
|
|
|
|
g_object_unref (G_OBJECT (*iconp));
|
|
|
|
if (*mini_iconp)
|
|
|
|
g_object_unref (G_OBJECT (*mini_iconp));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (icon_cache->origin <= USING_WM_HINTS &&
|
|
|
|
icon_cache->wm_hints_dirty)
|
|
|
|
{
|
2014-07-14 15:29:54 +00:00
|
|
|
Pixmap pixmap;
|
|
|
|
Pixmap mask;
|
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
icon_cache->wm_hints_dirty = FALSE;
|
|
|
|
|
|
|
|
pixmap = wm_hints_pixmap;
|
|
|
|
mask = wm_hints_mask;
|
|
|
|
|
|
|
|
/* We won't update if pixmap is unchanged;
|
|
|
|
* avoids a get_from_drawable() on every geometry
|
|
|
|
* hints change
|
|
|
|
*/
|
|
|
|
if ((pixmap != icon_cache->prev_pixmap ||
|
|
|
|
mask != icon_cache->prev_mask) &&
|
|
|
|
pixmap != None)
|
|
|
|
{
|
|
|
|
if (try_pixmap_and_mask (screen->display,
|
|
|
|
pixmap, mask,
|
|
|
|
iconp, ideal_width, ideal_height,
|
|
|
|
mini_iconp, ideal_mini_width, ideal_mini_height))
|
|
|
|
{
|
|
|
|
icon_cache->prev_pixmap = pixmap;
|
|
|
|
icon_cache->prev_mask = mask;
|
2014-07-14 15:27:01 +00:00
|
|
|
icon_cache->origin = USING_WM_HINTS;
|
2002-02-09 23:03:52 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (icon_cache->origin <= USING_KWM_WIN_ICON &&
|
|
|
|
icon_cache->kwm_win_icon_dirty)
|
|
|
|
{
|
2014-07-14 15:29:54 +00:00
|
|
|
Pixmap pixmap;
|
|
|
|
Pixmap mask;
|
|
|
|
|
2002-02-09 23:03:52 +00:00
|
|
|
icon_cache->kwm_win_icon_dirty = FALSE;
|
|
|
|
|
|
|
|
get_kwm_win_icon (screen->display, xwindow, &pixmap, &mask);
|
|
|
|
|
|
|
|
if ((pixmap != icon_cache->prev_pixmap ||
|
|
|
|
mask != icon_cache->prev_mask) &&
|
|
|
|
pixmap != None)
|
|
|
|
{
|
|
|
|
if (try_pixmap_and_mask (screen->display, pixmap, mask,
|
|
|
|
iconp, ideal_width, ideal_height,
|
|
|
|
mini_iconp, ideal_mini_width, ideal_mini_height))
|
|
|
|
{
|
|
|
|
icon_cache->prev_pixmap = pixmap;
|
|
|
|
icon_cache->prev_mask = mask;
|
2014-07-14 15:27:01 +00:00
|
|
|
icon_cache->origin = USING_KWM_WIN_ICON;
|
2002-02-09 23:03:52 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-14 15:23:18 +00:00
|
|
|
if (icon_cache->origin < USING_FALLBACK_ICON)
|
2002-02-09 23:03:52 +00:00
|
|
|
{
|
2014-07-14 15:27:01 +00:00
|
|
|
icon_cache->origin = USING_FALLBACK_ICON;
|
2014-07-14 16:01:22 +00:00
|
|
|
*iconp = NULL;
|
|
|
|
*mini_iconp = NULL;
|
2002-02-09 23:03:52 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* found nothing new */
|
|
|
|
return FALSE;
|
|
|
|
}
|