mutter/src/compositor/compositor-clutter.c

2289 lines
60 KiB
C
Raw Normal View History

#define _GNU_SOURCE
#define _XOPEN_SOURCE 500 /* for usleep() */
#include <config.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include <unistd.h>
#include <gdk/gdk.h>
#include "display.h"
#include "screen.h"
#include "frame.h"
#include "errors.h"
#include "window.h"
#include "compositor-private.h"
#include "compositor-clutter.h"
#include "xprops.h"
#include <X11/Xatom.h>
#include <X11/extensions/shape.h>
#include <X11/extensions/Xcomposite.h>
#include <X11/extensions/Xdamage.h>
#include <X11/extensions/Xfixes.h>
#include <X11/extensions/Xrender.h>
#include <clutter/clutter.h>
#include <clutter/clutter-group.h>
#include <clutter/x11/clutter-x11.h>
#include <clutter/glx/clutter-glx.h>
#include <cogl/cogl.h>
#define SHADOW_RADIUS 8
#define SHADOW_OPACITY 0.9
2008-08-19 09:33:20 -04:00
#define SHADOW_OFFSET_X (SHADOW_RADIUS)
#define SHADOW_OFFSET_Y (SHADOW_RADIUS)
#define MAX_TILE_SZ 8 /* Must be <= shaddow radius */
#define TILE_WIDTH (3*MAX_TILE_SZ)
#define TILE_HEIGHT (3*MAX_TILE_SZ)
#define DESTROY_TIMEOUT 300
#define MINIMIZE_TIMEOUT 600
2008-08-20 04:33:39 -04:00
static ClutterActor* tidy_texture_frame_new (ClutterTexture *texture,
gint left,
gint top,
gint right,
gint bottom);
2008-08-20 04:33:39 -04:00
static unsigned char* shadow_gaussian_make_tile (void);
#ifdef HAVE_COMPOSITE_EXTENSIONS
static inline gboolean
2008-08-20 04:33:39 -04:00
composite_at_least_version (MetaDisplay *display, int maj, int min)
{
static int major = -1;
static int minor = -1;
if (major == -1)
meta_display_get_compositor_version (display, &major, &minor);
return (major > maj || (major == maj && minor >= min));
}
#endif
typedef enum _MetaCompWindowType
{
/*
* Types shared with MetaWindow
*/
META_COMP_WINDOW_NORMAL = META_WINDOW_NORMAL,
META_COMP_WINDOW_DESKTOP = META_WINDOW_DESKTOP,
META_COMP_WINDOW_DOCK = META_WINDOW_DOCK,
META_COMP_WINDOW_MENU = META_WINDOW_MENU,
/*
* Extended types that WM does not care about, but we do.
*/
META_COMP_WINDOW_TOOLTIP = 0xf000,
META_COMP_WINDOW_DROP_DOWN_MENU,
META_COMP_WINDOW_DND,
} MetaCompWindowType;
typedef struct _MetaCompositorClutter
{
2008-08-20 04:33:39 -04:00
MetaCompositor compositor;
MetaDisplay *display;
2008-08-20 04:33:39 -04:00
Atom atom_x_root_pixmap;
Atom atom_x_set_root;
Atom atom_net_wm_window_opacity;
2008-08-20 04:33:39 -04:00
gboolean show_redraw : 1;
gboolean debug : 1;
} MetaCompositorClutter;
typedef struct _MetaCompScreen
{
2008-08-20 04:33:39 -04:00
MetaScreen *screen;
2008-08-20 04:33:39 -04:00
ClutterActor *stage;
GList *windows;
GHashTable *windows_by_xid;
MetaWindow *focus_window;
Window output;
GSList *dock_windows;
ClutterEffectTemplate *destroy_effect;
ClutterEffectTemplate *minimize_effect;
2008-08-20 04:33:39 -04:00
ClutterActor *shadow_src;
} MetaCompScreen;
/*
* MetaCompWindow object (ClutterGroup sub-class)
*/
#define META_TYPE_COMP_WINDOW (meta_comp_window_get_type ())
#define META_COMP_WINDOW(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), META_TYPE_COMP_WINDOW, MetaCompWindow))
#define META_COMP_WINDOW_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), META_TYPE_COMP_WINDOW, MetaCompWindowClass))
#define IS_META_COMP_WINDOW(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), META_COMP_WINDOW_TYPE))
#define META_IS_COMP_WINDOW_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), META_TYPE_COMP_WINDOW))
#define META_COMP_WINDOW_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), META_TYPE_COMP_WINDOW, MetaCompWindowClass))
typedef struct _MetaCompWindow MetaCompWindow;
typedef struct _MetaCompWindowClass MetaCompWindowClass;
typedef struct _MetaCompWindowPrivate MetaCompWindowPrivate;
struct _MetaCompWindowClass
{
2008-08-20 04:33:39 -04:00
ClutterGroupClass parent_class;
};
struct _MetaCompWindow
{
2008-08-20 04:33:39 -04:00
ClutterGroup parent;
MetaCompWindowPrivate *priv;
};
struct _MetaCompWindowPrivate
{
MetaWindow *window;
Window xwindow;
MetaScreen *screen;
XWindowAttributes attrs;
ClutterActor *actor;
ClutterActor *shadow;
Pixmap back_pixmap;
MetaCompWindowType type;
Damage damage;
guint8 opacity;
gboolean needs_shadow : 1;
gboolean shaped : 1;
gboolean destroy_pending : 1;
gboolean argb32 : 1;
gboolean minimize_in_progress : 1;
2008-08-20 03:26:52 -04:00
gboolean disposed : 1;
};
enum
{
2008-08-20 04:33:39 -04:00
PROP_MCW_META_WINDOW = 1,
PROP_MCW_META_SCREEN,
PROP_MCW_X_WINDOW,
};
static void meta_comp_window_class_init (MetaCompWindowClass *klass);
static void meta_comp_window_init (MetaCompWindow *self);
static void meta_comp_window_dispose (GObject *object);
static void meta_comp_window_finalize (GObject *object);
2008-08-20 03:26:52 -04:00
static void meta_comp_window_constructed (GObject *object);
static void meta_comp_window_set_property (GObject *object,
2008-08-20 03:26:52 -04:00
guint prop_id,
const GValue *value,
GParamSpec *pspec);
static void meta_comp_window_get_property (GObject *object,
2008-08-20 03:26:52 -04:00
guint prop_id,
GValue *value,
GParamSpec *pspec);
static void meta_comp_window_get_window_type (MetaCompWindow *self);
2008-08-20 03:26:52 -04:00
static void meta_comp_window_detach (MetaCompWindow *self);
GType meta_comp_window_get_type (void);
G_DEFINE_TYPE (MetaCompWindow, meta_comp_window, CLUTTER_TYPE_GROUP);
static void
meta_comp_window_class_init (MetaCompWindowClass *klass)
{
2008-08-20 04:33:39 -04:00
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GParamSpec *pspec;
g_type_class_add_private (klass, sizeof (MetaCompWindowPrivate));
object_class->dispose = meta_comp_window_dispose;
object_class->finalize = meta_comp_window_finalize;
object_class->set_property = meta_comp_window_set_property;
object_class->get_property = meta_comp_window_get_property;
object_class->constructed = meta_comp_window_constructed;
pspec = g_param_spec_pointer ("meta-window",
"MetaWindow",
"MetaWindow",
G_PARAM_READWRITE | G_PARAM_CONSTRUCT);
g_object_class_install_property (object_class,
2008-08-20 04:33:39 -04:00
PROP_MCW_META_WINDOW,
pspec);
pspec = g_param_spec_pointer ("meta-screen",
"MetaScreen",
"MetaScreen",
G_PARAM_READWRITE | G_PARAM_CONSTRUCT);
g_object_class_install_property (object_class,
2008-08-20 04:33:39 -04:00
PROP_MCW_META_SCREEN,
pspec);
pspec = g_param_spec_ulong ("x-window",
"Window",
"Window",
0,
G_MAXULONG,
0,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT);
g_object_class_install_property (object_class,
2008-08-20 04:33:39 -04:00
PROP_MCW_X_WINDOW,
pspec);
}
static void
meta_comp_window_init (MetaCompWindow *self)
{
MetaCompWindowPrivate *priv;
priv = self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self,
META_TYPE_COMP_WINDOW,
MetaCompWindowPrivate);
priv->opacity = 0xff;
}
static gboolean is_shaped (MetaDisplay *display, Window xwindow);
static gboolean meta_comp_window_has_shadow (MetaCompWindow *self);
static void
2008-08-20 03:26:52 -04:00
meta_comp_window_constructed (GObject *object)
{
2008-08-20 04:33:39 -04:00
MetaCompWindow *self = META_COMP_WINDOW (object);
MetaCompWindowPrivate *priv = self->priv;
2008-08-20 04:33:39 -04:00
MetaScreen *screen = priv->screen;
MetaDisplay *display = meta_screen_get_display (screen);
Window xwindow = priv->xwindow;
Display *xdisplay = meta_display_get_xdisplay (display);
gulong event_mask;
XRenderPictFormat *format;
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
if (!XGetWindowAttributes (xdisplay, xwindow, &priv->attrs))
{
g_warning ("Could not get attributes for xwindow");
/* FIXME */
return;
}
meta_comp_window_get_window_type (self);
2008-08-20 04:33:39 -04:00
/*
* If Metacity has decided not to manage this window then the input events
* won't have been set on the window
*/
event_mask = priv->attrs.your_event_mask | PropertyChangeMask;
XSelectInput (xdisplay, xwindow, event_mask);
priv->shaped = is_shaped (display, xwindow);
if (priv->attrs.class == InputOnly)
priv->damage = None;
else
priv->damage = XDamageCreate (xdisplay, xwindow, XDamageReportNonEmpty);
format = XRenderFindVisualFormat (xdisplay, priv->attrs.visual);
2008-08-20 04:33:39 -04:00
if (format && format->type == PictTypeDirect && format->direct.alphaMask)
priv->argb32 = TRUE;
if (meta_comp_window_has_shadow (self))
{
priv->shadow =
tidy_texture_frame_new (CLUTTER_TEXTURE (info->shadow_src),
MAX_TILE_SZ,
MAX_TILE_SZ,
MAX_TILE_SZ,
MAX_TILE_SZ);
clutter_actor_set_position (priv->shadow,
SHADOW_OFFSET_X , SHADOW_OFFSET_Y);
clutter_container_add_actor (CLUTTER_CONTAINER (self), priv->shadow);
}
priv->actor = clutter_glx_texture_pixmap_new ();
clutter_container_add_actor (CLUTTER_CONTAINER (self), priv->actor);
}
static void
meta_comp_window_dispose (GObject *object)
{
2008-08-20 03:26:52 -04:00
MetaCompWindow *self = META_COMP_WINDOW (object);
MetaCompWindowPrivate *priv = self->priv;
MetaScreen *screen;
MetaDisplay *display;
Display *xdisplay;
MetaCompScreen *info;
if (priv->disposed)
return;
priv->disposed = TRUE;
screen = priv->screen;
display = meta_screen_get_display (screen);
xdisplay = meta_display_get_xdisplay (display);
info = meta_screen_get_compositor_data (screen);
meta_comp_window_detach (self);
if (priv->damage != None)
{
meta_error_trap_push (display);
XDamageDestroy (xdisplay, priv->damage);
meta_error_trap_pop (display, FALSE);
priv->damage = None;
}
/*
* Check we are not in the dock list -- FIXME (do this in a cleaner way)
*/
if (priv->type == META_COMP_WINDOW_DOCK)
info->dock_windows = g_slist_remove (info->dock_windows, self);
info->windows = g_list_remove (info->windows, (gconstpointer) self);
g_hash_table_remove (info->windows_by_xid, (gpointer) priv->xwindow);
G_OBJECT_CLASS (meta_comp_window_parent_class)->dispose (object);
}
static void
meta_comp_window_finalize (GObject *object)
{
G_OBJECT_CLASS (meta_comp_window_parent_class)->finalize (object);
}
static void
meta_comp_window_set_property (GObject *object,
2008-08-20 04:33:39 -04:00
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
MetaCompWindowPrivate *priv = META_COMP_WINDOW (object)->priv;
switch (prop_id)
{
2008-08-20 04:33:39 -04:00
case PROP_MCW_META_WINDOW:
priv->window = g_value_get_pointer (value);
break;
2008-08-20 04:33:39 -04:00
case PROP_MCW_META_SCREEN:
priv->screen = g_value_get_pointer (value);
break;
2008-08-20 04:33:39 -04:00
case PROP_MCW_X_WINDOW:
priv->xwindow = g_value_get_ulong (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
meta_comp_window_get_property (GObject *object,
2008-08-20 04:33:39 -04:00
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
MetaCompWindowPrivate *priv = META_COMP_WINDOW (object)->priv;
switch (prop_id)
{
2008-08-20 04:33:39 -04:00
case PROP_MCW_META_WINDOW:
g_value_set_pointer (value, priv->window);
break;
2008-08-20 04:33:39 -04:00
case PROP_MCW_META_SCREEN:
g_value_set_pointer (value, priv->screen);
break;
2008-08-20 04:33:39 -04:00
case PROP_MCW_X_WINDOW:
g_value_set_ulong (value, priv->xwindow);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static MetaCompWindow*
2008-08-20 04:33:39 -04:00
find_window_for_screen (MetaScreen *screen, Window xwindow)
{
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
if (info == NULL)
return NULL;
return g_hash_table_lookup (info->windows_by_xid, (gpointer) xwindow);
}
static MetaCompWindow *
2008-08-20 04:33:39 -04:00
find_window_in_display (MetaDisplay *display, Window xwindow)
{
GSList *index;
for (index = meta_display_get_screens (display);
index;
index = index->next)
{
MetaCompWindow *cw = find_window_for_screen (index->data, xwindow);
if (cw != NULL)
return cw;
}
return NULL;
}
static MetaCompWindow *
2008-08-20 04:33:39 -04:00
find_window_for_child_window_in_display (MetaDisplay *display, Window xwindow)
{
2008-08-20 04:33:39 -04:00
Window ignored1, *ignored2, parent;
guint ignored_children;
XQueryTree (meta_display_get_xdisplay (display), xwindow, &ignored1,
&parent, &ignored2, &ignored_children);
if (parent != None)
return find_window_in_display (display, parent);
return NULL;
}
static void
meta_comp_window_get_window_type (MetaCompWindow *self)
{
2008-08-20 04:33:39 -04:00
MetaCompWindowPrivate *priv = self->priv;
MetaScreen *screen = priv->screen;
MetaDisplay *display = meta_screen_get_display (screen);
Window xwindow = priv->xwindow;
gint n_atoms;
Atom *atoms;
gint i;
/*
* If the window is managed by the WM, get the type from the WM,
* otherwise do it the hard way.
*/
2008-08-20 04:33:39 -04:00
if (priv->window && meta_window_get_type_atom (priv->window) != None)
{
priv->type = (MetaCompWindowType) meta_window_get_type (priv->window);
return;
}
n_atoms = 0;
atoms = NULL;
/*
* Assume normal
*/
priv->type = META_COMP_WINDOW_NORMAL;
meta_prop_get_atom_list (display, xwindow,
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE),
&atoms, &n_atoms);
for (i = 0; i < n_atoms; i++)
{
if (atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_DND))
{
priv->type = META_COMP_WINDOW_DND;
break;
}
else if (atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_DESKTOP))
{
priv->type = META_COMP_WINDOW_DESKTOP;
break;
}
else if (atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_DOCK))
{
priv->type = META_COMP_WINDOW_DOCK;
break;
}
2008-08-20 04:33:39 -04:00
else if (atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_TOOLBAR) ||
atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_MENU) ||
atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_DIALOG) ||
atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_NORMAL) ||
atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_UTILITY) ||
atoms[i] ==
meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE_SPLASH))
{
priv->type = META_COMP_WINDOW_NORMAL;
break;
}
}
meta_XFree (atoms);
}
static gboolean
is_shaped (MetaDisplay *display, Window xwindow)
{
Display *xdisplay = meta_display_get_xdisplay (display);
2008-08-20 04:33:39 -04:00
gint xws, yws, xbs, ybs;
guint wws, hws, wbs, hbs;
gint bounding_shaped, clip_shaped;
if (meta_display_has_shape (display))
{
XShapeQueryExtents (xdisplay, xwindow, &bounding_shaped,
&xws, &yws, &wws, &hws, &clip_shaped,
&xbs, &ybs, &wbs, &hbs);
return (bounding_shaped != 0);
}
return FALSE;
}
static gboolean
meta_comp_window_has_shadow (MetaCompWindow *self)
{
MetaCompWindowPrivate * priv = self->priv;
/*
* Do not add shadows to ARGB windows (since they are probably transparent
*/
2008-08-20 04:33:39 -04:00
if (priv->argb32 || priv->opacity != 0xff)
{
meta_verbose ("Window has no shadow as it is ARGB\n");
return FALSE;
}
2008-08-20 04:33:39 -04:00
/*
* Add shadows to override redirect windows (e.g., Gtk menus).
*/
if (priv->attrs.override_redirect)
{
meta_verbose ("Window has shadow because it is override redirect.\n");
return TRUE;
}
2008-08-20 04:33:39 -04:00
/*
* Always put a shadow around windows with a frame - This should override
* the restriction about not putting a shadow around shaped windows
* as the frame might be the reason the window is shaped
*/
if (priv->window)
{
2008-08-20 04:33:39 -04:00
if (meta_window_get_frame (priv->window))
{
meta_verbose ("Window has shadow because it has a frame\n");
return TRUE;
}
}
2008-08-20 04:33:39 -04:00
/*
* Never put a shadow around shaped windows
*/
if (priv->shaped)
{
meta_verbose ("Window has no shadow as it is shaped\n");
return FALSE;
}
2008-08-20 04:33:39 -04:00
/*
* Don't put shadow around DND icon windows
*/
if (priv->type == META_COMP_WINDOW_DND ||
2008-08-20 04:33:39 -04:00
priv->type == META_COMP_WINDOW_DESKTOP)
{
meta_verbose ("Window has no shadow as it is DND or Desktop\n");
return FALSE;
}
if (priv->type == META_COMP_WINDOW_MENU
#if 0
|| priv->type == META_COMP_WINDOW_DROP_DOWN_MENU
#endif
2008-08-20 04:33:39 -04:00
)
{
meta_verbose ("Window has shadow as it is a menu\n");
return TRUE;
}
#if 0
2008-08-20 04:33:39 -04:00
if (priv->type == META_COMP_WINDOW_TOOLTIP)
{
meta_verbose ("Window has shadow as it is a tooltip\n");
return TRUE;
}
#endif
meta_verbose ("Window has no shadow as it fell through\n");
return FALSE;
}
static void
clutter_cmp_destroy (MetaCompositor *compositor)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
#endif
}
static void
2008-08-20 03:26:52 -04:00
meta_comp_window_detach (MetaCompWindow *self)
{
MetaCompWindowPrivate *priv = self->priv;
MetaScreen *screen = priv->screen;
MetaDisplay *display = meta_screen_get_display (screen);
Display *xdisplay = meta_display_get_xdisplay (display);
2008-08-20 03:26:52 -04:00
if (priv->back_pixmap)
{
XFreePixmap (xdisplay, priv->back_pixmap);
priv->back_pixmap = None;
}
}
static void
2008-08-20 03:26:52 -04:00
destroy_win (MetaDisplay *display, Window xwindow)
{
MetaCompWindow *cw;
cw = find_window_in_display (display, xwindow);
if (cw == NULL)
return;
meta_verbose ("destroying a window... 0x%x (%p)\n", (guint) xwindow, cw);
2008-08-20 03:26:52 -04:00
clutter_actor_destroy (CLUTTER_ACTOR (cw));
}
static void
2008-08-20 04:33:39 -04:00
restack_win (MetaCompWindow *cw, Window above)
{
MetaCompWindowPrivate *priv = cw->priv;
2008-08-20 04:33:39 -04:00
MetaScreen *screen = priv->screen;
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
Window previous_above;
GList *sibling, *next;
sibling = g_list_find (info->windows, (gconstpointer) cw);
next = g_list_next (sibling);
previous_above = None;
if (next)
{
2008-08-20 04:33:39 -04:00
MetaCompWindow *ncw = next->data;
previous_above = ncw->priv->xwindow;
}
/* If above is set to None, the window whose state was changed is on
* the bottom of the stack with respect to sibling.
*/
if (above == None)
{
/* Insert at bottom of window stack */
info->windows = g_list_delete_link (info->windows, sibling);
info->windows = g_list_append (info->windows, cw);
clutter_actor_raise_top (CLUTTER_ACTOR (cw));
}
else if (previous_above != above)
{
GList *index;
2008-08-20 04:33:39 -04:00
for (index = info->windows; index; index = index->next)
{
MetaCompWindow *cw2 = (MetaCompWindow *) index->data;
if (cw2->priv->xwindow == above)
break;
}
if (index != NULL)
{
2008-08-20 04:33:39 -04:00
ClutterActor *above_win = index->data;
info->windows = g_list_delete_link (info->windows, sibling);
info->windows = g_list_insert_before (info->windows, index, cw);
2008-08-20 04:33:39 -04:00
clutter_actor_raise (CLUTTER_ACTOR (cw), above_win);
}
}
}
static void
resize_win (MetaCompWindow *cw,
int x,
int y,
int width,
int height,
int border_width,
gboolean override_redirect)
{
MetaCompWindowPrivate *priv = cw->priv;
priv->attrs.x = x;
priv->attrs.y = y;
clutter_actor_set_position (CLUTTER_ACTOR (cw), x, y);
/* Note, let named named pixmap resync actually resize actor */
if (priv->attrs.width != width || priv->attrs.height != height)
2008-08-20 04:33:39 -04:00
meta_comp_window_detach (cw);
2008-08-20 04:33:39 -04:00
priv->attrs.width = width;
priv->attrs.height = height;
priv->attrs.border_width = border_width;
priv->attrs.override_redirect = override_redirect;
}
static void
2008-08-20 04:33:39 -04:00
map_win (MetaDisplay *display, MetaScreen *screen, Window id)
{
2008-08-20 04:33:39 -04:00
MetaCompWindow *cw = find_window_for_screen (screen, id);
MetaCompWindowPrivate *priv;
if (cw == NULL)
return;
priv = cw->priv;
priv->attrs.map_state = IsViewable;
priv->minimize_in_progress = FALSE;
clutter_actor_show (CLUTTER_ACTOR (cw));
}
static void
2008-08-20 04:33:39 -04:00
unmap_win (MetaDisplay *display, MetaScreen *screen, Window id)
{
2008-08-20 04:33:39 -04:00
MetaCompWindow *cw = find_window_for_screen (screen, id);
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
MetaCompWindowPrivate *priv;
if (cw == NULL)
return;
priv = cw->priv;
if (priv->window && priv->window == info->focus_window)
info->focus_window = NULL;
priv->attrs.map_state = IsUnmapped;
2008-08-20 03:26:52 -04:00
meta_comp_window_detach (cw);
if (!priv->minimize_in_progress)
2008-08-20 04:33:39 -04:00
clutter_actor_hide (CLUTTER_ACTOR (cw));
}
static void
2008-08-20 04:33:39 -04:00
add_win (MetaScreen *screen, MetaWindow *window, Window xwindow)
{
2008-08-20 04:33:39 -04:00
MetaDisplay *display = meta_screen_get_display (screen);
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
MetaCompWindow *cw;
MetaCompWindowPrivate *priv;
if (info == NULL)
return;
if (xwindow == info->output)
return;
meta_verbose ("add window: Meta %p, xwin 0x%x\n", window, (guint) xwindow);
cw = g_object_new (META_TYPE_COMP_WINDOW,
"meta-window", window,
"x-window", xwindow,
"meta-screen", screen,
NULL);
priv = cw->priv;
clutter_actor_set_position (CLUTTER_ACTOR (cw),
priv->attrs.x, priv->attrs.y);
clutter_container_add_actor (CLUTTER_CONTAINER (info->stage),
CLUTTER_ACTOR (cw));
clutter_actor_hide (CLUTTER_ACTOR (cw));
/* Only add the window to the list of docks if it needs a shadow */
if (priv->type == META_COMP_WINDOW_DOCK)
{
meta_verbose ("Appending 0x%x to dock windows\n", (guint)xwindow);
info->dock_windows = g_slist_append (info->dock_windows, cw);
}
#if 0
printf ("added 0x%x (%p) type:", (guint)xwindow, cw);
switch (cw->type)
{
case META_COMP_WINDOW_NORMAL:
printf("normal"); break;
case META_COMP_WINDOW_DND:
printf("dnd"); break;
case META_COMP_WINDOW_DESKTOP:
printf("desktop"); break;
case META_COMP_WINDOW_DOCK:
printf("dock"); break;
case META_COMP_WINDOW_MENU:
printf("menu"); break;
case META_COMP_WINDOW_DROP_DOWN_MENU:
printf("menu"); break;
case META_COMP_WINDOW_TOOLTIP:
printf("tooltip"); break;
default:
printf("unknown");
break;
}
if (window && meta_window_get_frame (window))
printf(" *HAS FRAME* ");
printf("\n");
#endif
2008-08-20 04:33:39 -04:00
/*
* Add this to the list at the top of the stack before it is mapped so that
* map_win can find it again
*/
info->windows = g_list_prepend (info->windows, cw);
g_hash_table_insert (info->windows_by_xid, (gpointer) xwindow, cw);
if (priv->attrs.map_state == IsViewable)
map_win (display, screen, xwindow);
}
static void
repair_win (MetaCompWindow *cw)
{
2008-08-20 04:33:39 -04:00
MetaCompWindowPrivate *priv = cw->priv;
MetaScreen *screen = priv->screen;
MetaDisplay *display = meta_screen_get_display (screen);
Display *xdisplay = meta_display_get_xdisplay (display);
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
Window xwindow = priv->xwindow;
if (xwindow == meta_screen_get_xroot (screen) ||
xwindow == clutter_x11_get_stage_window (CLUTTER_STAGE (info->stage)))
return;
meta_error_trap_push (display);
if (priv->back_pixmap == None)
{
gint pxm_width, pxm_height;
priv->back_pixmap = XCompositeNameWindowPixmap (xdisplay, xwindow);
if (priv->back_pixmap == None)
{
meta_verbose ("Unable to get named pixmap for %p\n", cw);
return;
}
clutter_x11_texture_pixmap_set_pixmap
(CLUTTER_X11_TEXTURE_PIXMAP (priv->actor),
priv->back_pixmap);
g_object_get (priv->actor,
"pixmap-width", &pxm_width,
"pixmap-height", &pxm_height,
NULL);
clutter_actor_set_size (priv->actor, pxm_width, pxm_height);
if (priv->shadow)
clutter_actor_set_size (priv->shadow, pxm_width, pxm_height);
}
/*
* TODO -- on some gfx hardware updating the whole texture instead of
* the individual rectangles is actually quicker, so we might want to
* make this a configurable option (on desktop HW with multiple pipelines
* it is usually quicker to just update the damaged parts).
*
* If we are using TFP we update the whole texture (this simply trigers
* the texture rebind).
*/
if (CLUTTER_GLX_IS_TEXTURE_PIXMAP (priv->actor) &&
clutter_glx_texture_pixmap_using_extension (
CLUTTER_GLX_TEXTURE_PIXMAP (priv->actor)))
{
XDamageSubtract (xdisplay, priv->damage, None, None);
clutter_x11_texture_pixmap_update_area
(CLUTTER_X11_TEXTURE_PIXMAP (priv->actor),
0,
0,
clutter_actor_get_width (priv->actor),
clutter_actor_get_height (priv->actor));
}
else
{
XRectangle *r_damage;
XRectangle r_bounds;
XserverRegion parts;
int i, r_count;
parts = XFixesCreateRegion (xdisplay, 0, 0);
XDamageSubtract (xdisplay, priv->damage, None, parts);
r_damage = XFixesFetchRegionAndBounds (xdisplay,
parts,
&r_count,
&r_bounds);
if (r_damage)
{
for (i = 0; i < r_count; ++i)
{
clutter_x11_texture_pixmap_update_area
(CLUTTER_X11_TEXTURE_PIXMAP (priv->actor),
r_damage[i].x,
r_damage[i].y,
r_damage[i].width,
r_damage[i].height);
}
}
XFree (r_damage);
XFixesDestroyRegion (xdisplay, parts);
}
meta_error_trap_pop (display, FALSE);
}
static void
process_create (MetaCompositorClutter *compositor,
XCreateWindowEvent *event,
MetaWindow *window)
{
MetaScreen *screen;
screen = meta_display_screen_for_root (compositor->display, event->parent);
2008-08-20 04:33:39 -04:00
if (screen == NULL)
return;
2008-08-20 04:33:39 -04:00
/*
* This is quite silly as we end up creating windows as then immediatly
* destroying them as they (likely) become framed and thus reparented.
2008-08-20 04:33:39 -04:00
*/
if (!find_window_in_display (compositor->display, event->window))
add_win (screen, window, event->window);
}
static void
process_reparent (MetaCompositorClutter *compositor,
XReparentEvent *event,
MetaWindow *window)
{
MetaScreen *screen;
screen = meta_display_screen_for_root (compositor->display, event->parent);
2008-08-20 04:33:39 -04:00
if (screen != NULL)
{
meta_verbose ("reparent: adding a new window 0x%x\n",
(guint)event->window);
add_win (screen, window, event->window);
}
else
{
meta_verbose ("reparent: destroying a window 0%x\n",
(guint)event->window);
2008-08-20 03:26:52 -04:00
destroy_win (compositor->display, event->window);
}
}
static void
process_destroy (MetaCompositorClutter *compositor,
XDestroyWindowEvent *event)
{
2008-08-20 03:26:52 -04:00
destroy_win (compositor->display, event->window);
}
static void
process_damage (MetaCompositorClutter *compositor,
XDamageNotifyEvent *event)
{
XEvent next;
Display *dpy = event->display;
Drawable drawable = event->drawable;
MetaCompWindowPrivate *priv;
MetaCompWindow *cw = find_window_in_display (compositor->display, drawable);
if (!cw)
return;
priv = cw->priv;
if (priv->destroy_pending)
return;
if (XCheckTypedWindowEvent (dpy, drawable, DestroyNotify, &next))
{
priv->destroy_pending = TRUE;
process_destroy (compositor, (XDestroyWindowEvent *) &next);
return;
}
repair_win (cw);
}
static void
process_configure_notify (MetaCompositorClutter *compositor,
XConfigureEvent *event)
{
MetaDisplay *display = compositor->display;
MetaCompWindow *cw = find_window_in_display (display, event->window);
if (cw)
{
restack_win (cw, event->above);
resize_win (cw,
event->x, event->y, event->width, event->height,
event->border_width, event->override_redirect);
}
else
{
/* FIXME -- handle root window resizing */
}
}
static void
process_circulate_notify (MetaCompositorClutter *compositor,
XCirculateEvent *event)
{
MetaCompWindow *cw = find_window_in_display (compositor->display,
event->window);
MetaCompWindow *top;
MetaCompScreen *info;
2008-08-20 04:33:39 -04:00
Window above;
MetaCompWindowPrivate *priv;
if (!cw)
return;
priv = cw->priv;
2008-08-20 04:33:39 -04:00
info = meta_screen_get_compositor_data (priv->screen);
top = info->windows->data;
if ((event->place == PlaceOnTop) && top)
above = top->priv->xwindow;
else
above = None;
restack_win (cw, above);
}
static void
process_unmap (MetaCompositorClutter *compositor,
XUnmapEvent *event)
{
MetaCompWindow *cw;
Window xwin = event->window;
Display *dpy = event->display;
if (event->from_configure)
{
/* Ignore unmap caused by parent's resize */
return;
}
cw = find_window_in_display (compositor->display, xwin);
if (cw)
{
XEvent next;
MetaCompWindowPrivate *priv = cw->priv;
if (priv->destroy_pending)
return;
if (XCheckTypedWindowEvent (dpy, xwin, DestroyNotify, &next))
{
priv->destroy_pending = TRUE;
process_destroy (compositor, (XDestroyWindowEvent *) &next);
return;
}
meta_verbose ("processing unmap of 0x%x (%p)\n", (guint)xwin, cw);
unmap_win (compositor->display, priv->screen, xwin);
}
}
static void
process_map (MetaCompositorClutter *compositor,
XMapEvent *event,
MetaWindow *window)
{
MetaCompWindow *cw = find_window_in_display (compositor->display,
event->window);
if (cw)
map_win (compositor->display, cw->priv->screen, event->window);
}
static void
process_property_notify (MetaCompositorClutter *compositor,
XPropertyEvent *event)
{
MetaDisplay *display = compositor->display;
/* Check for the opacity changing */
if (event->atom == compositor->atom_net_wm_window_opacity)
{
MetaCompWindow *cw = find_window_in_display (display, event->window);
2008-08-20 04:33:39 -04:00
gulong value;
if (!cw)
{
/* Applications can set this for their toplevel windows, so
* this must be propagated to the window managed by the compositor
*/
cw = find_window_for_child_window_in_display (display,
event->window);
}
if (!cw)
return;
if (meta_prop_get_cardinal (display, event->window,
compositor->atom_net_wm_window_opacity,
&value) == FALSE)
{
guint8 opacity;
opacity = (guint8)((gfloat)value * 255.0 / ((gfloat)0xffffffff));
cw->priv->opacity = opacity;
clutter_actor_set_opacity (CLUTTER_ACTOR (cw), opacity);
}
return;
}
2008-08-20 04:33:39 -04:00
else if (event->atom == meta_display_get_atom (display,
META_ATOM__NET_WM_WINDOW_TYPE))
{
MetaCompWindow *cw = find_window_in_display (display, event->window);
if (!cw)
return;
meta_comp_window_get_window_type (cw);
return;
}
}
static void
2008-08-20 04:33:39 -04:00
show_overlay_window (MetaScreen *screen, Window cow)
{
2008-08-20 04:33:39 -04:00
MetaDisplay *display = meta_screen_get_display (screen);
Display *xdisplay = meta_display_get_xdisplay (display);
XserverRegion region;
region = XFixesCreateRegion (xdisplay, NULL, 0);
XFixesSetWindowShapeRegion (xdisplay, cow, ShapeBounding, 0, 0, 0);
XFixesSetWindowShapeRegion (xdisplay, cow, ShapeInput, 0, 0, region);
XFixesDestroyRegion (xdisplay, region);
}
static Window
get_output_window (MetaScreen *screen)
{
MetaDisplay *display = meta_screen_get_display (screen);
2008-08-20 04:33:39 -04:00
Display *xdisplay = meta_display_get_xdisplay (display);
Window output, xroot;
xroot = meta_screen_get_xroot (screen);
output = XCompositeGetOverlayWindow (xdisplay, xroot);
XSelectInput (xdisplay, output, ExposureMask);
return output;
}
static void
clutter_cmp_manage_screen (MetaCompositor *compositor,
MetaScreen *screen)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
MetaCompScreen *info;
2008-08-20 04:33:39 -04:00
MetaDisplay *display = meta_screen_get_display (screen);
Display *xdisplay = meta_display_get_xdisplay (display);
int screen_number = meta_screen_get_screen_number (screen);
Window xroot = meta_screen_get_xroot (screen);
Window xwin;
gint width, height;
guchar *data;
/* Check if the screen is already managed */
if (meta_screen_get_compositor_data (screen))
return;
2008-08-20 04:33:39 -04:00
meta_error_trap_push_with_return (display);
XCompositeRedirectSubwindows (xdisplay, xroot, CompositeRedirectManual);
XSync (xdisplay, FALSE);
2008-08-20 04:33:39 -04:00
if (meta_error_trap_pop_with_return (display, FALSE))
{
g_warning ("Another compositing manager is running on screen %i",
screen_number);
return;
}
info = g_new0 (MetaCompScreen, 1);
info->screen = screen;
meta_screen_set_compositor_data (screen, info);
info->output = get_output_window (screen);
info->windows = NULL;
info->windows_by_xid = g_hash_table_new (g_direct_hash, g_direct_equal);
info->focus_window = meta_display_get_focus_window (display);
XClearArea (xdisplay, info->output, 0, 0, 0, 0, TRUE);
meta_screen_set_cm_selection (screen);
info->stage = clutter_stage_get_default ();
meta_screen_get_size (screen, &width, &height);
clutter_actor_set_size (info->stage, width, height);
xwin = clutter_x11_get_stage_window (CLUTTER_STAGE (info->stage));
XReparentWindow (xdisplay, xwin, info->output, 0, 0);
/* Shadow setup */
data = shadow_gaussian_make_tile ();
info->shadow_src = clutter_texture_new ();
clutter_texture_set_from_rgb_data (CLUTTER_TEXTURE (info->shadow_src),
data,
TRUE,
TILE_WIDTH,
TILE_HEIGHT,
TILE_WIDTH*4,
4,
0,
NULL);
free (data);
clutter_actor_show_all (info->stage);
/* Now we're up and running we can show the output if needed */
show_overlay_window (screen, info->output);
info->destroy_effect
= clutter_effect_template_new (clutter_timeline_new_for_duration (
DESTROY_TIMEOUT),
CLUTTER_ALPHA_SINE_INC);
info->minimize_effect
= clutter_effect_template_new (clutter_timeline_new_for_duration (
MINIMIZE_TIMEOUT),
CLUTTER_ALPHA_SINE_INC);
#endif
}
static void
clutter_cmp_unmanage_screen (MetaCompositor *compositor,
MetaScreen *screen)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
#endif
}
static void
clutter_cmp_add_window (MetaCompositor *compositor,
MetaWindow *window,
Window xwindow,
XWindowAttributes *attrs)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
MetaCompositorClutter *xrc = (MetaCompositorClutter *) compositor;
2008-08-20 04:33:39 -04:00
MetaScreen *screen = meta_screen_for_x_screen (attrs->screen);
meta_error_trap_push (xrc->display);
add_win (screen, window, xwindow);
meta_error_trap_pop (xrc->display, FALSE);
#endif
}
static void
clutter_cmp_remove_window (MetaCompositor *compositor,
Window xwindow)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
#endif
}
static void
clutter_cmp_set_updates (MetaCompositor *compositor,
MetaWindow *window,
gboolean update)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
#endif
}
static void
clutter_cmp_process_event (MetaCompositor *compositor,
XEvent *event,
MetaWindow *window)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
MetaCompositorClutter *xrc = (MetaCompositorClutter *) compositor;
/*
* This trap is so that none of the compositor functions cause
* X errors. This is really a hack, but I'm afraid I don't understand
* enough about Metacity/X to know how else you are supposed to do it
*/
meta_error_trap_push (xrc->display);
switch (event->type)
{
case CirculateNotify:
process_circulate_notify (xrc, (XCirculateEvent *) event);
break;
case ConfigureNotify:
process_configure_notify (xrc, (XConfigureEvent *) event);
break;
case PropertyNotify:
process_property_notify (xrc, (XPropertyEvent *) event);
break;
case Expose:
break;
case UnmapNotify:
process_unmap (xrc, (XUnmapEvent *) event);
break;
case MapNotify:
process_map (xrc, (XMapEvent *) event, window);
break;
case ReparentNotify:
process_reparent (xrc, (XReparentEvent *) event, window);
break;
case CreateNotify:
process_create (xrc, (XCreateWindowEvent *) event, window);
break;
case DestroyNotify:
process_destroy (xrc, (XDestroyWindowEvent *) event);
break;
default:
if (event->type == meta_display_get_damage_event_base (xrc->display) + XDamageNotify)
{
process_damage (xrc, (XDamageNotifyEvent *) event);
}
/* else if (event->type == meta_display_get_shape_event_base (xrc->display) + ShapeNotify)
process_shape (xrc, (XShapeEvent *) event);
else
{
meta_error_trap_pop (xrc->display, FALSE);
return;
}
*/
break;
}
meta_error_trap_pop (xrc->display, FALSE);
#endif
}
static Pixmap
clutter_cmp_get_window_pixmap (MetaCompositor *compositor,
MetaWindow *window)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
return None;
#else
return None;
#endif
}
static void
clutter_cmp_set_active_window (MetaCompositor *compositor,
MetaScreen *screen,
MetaWindow *window)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
#endif
}
static void
on_destroy_effect_complete (ClutterActor *actor,
gpointer user_data)
{
2008-08-20 03:26:52 -04:00
clutter_actor_destroy (actor);
}
static void
clutter_cmp_destroy_window (MetaCompositor *compositor,
MetaWindow *window)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
2008-08-20 04:33:39 -04:00
MetaCompWindow *cw = NULL;
MetaScreen *screen = meta_window_get_screen (window);
MetaCompScreen *info = meta_screen_get_compositor_data (screen);
MetaFrame *f = meta_window_get_frame (window);
/* Chances are we actually get the window frame here */
cw = find_window_for_screen (screen,
f ? meta_frame_get_xwindow (f) :
meta_window_get_xwindow (window));
if (!cw)
return;
2008-08-20 04:33:39 -04:00
/*
* We remove the window from internal lookup hashes and thus any other
* unmap events etc fail
2008-08-20 04:33:39 -04:00
*/
info->windows = g_list_remove (info->windows, (gconstpointer) cw);
g_hash_table_remove (info->windows_by_xid,
(gpointer) (f ? meta_frame_get_xwindow (f) :
meta_window_get_xwindow (window)));
clutter_actor_move_anchor_point_from_gravity (CLUTTER_ACTOR (cw),
CLUTTER_GRAVITY_CENTER);
clutter_effect_fade (info->destroy_effect,
CLUTTER_ACTOR (cw),
0,
on_destroy_effect_complete,
(gpointer)cw);
clutter_effect_scale (info->destroy_effect ,
CLUTTER_ACTOR (cw),
1.0,
0.0,
NULL,
NULL);
#endif
}
static void
on_minimize_effect_complete (ClutterActor *actor,
gpointer user_data)
{
MetaCompWindow *cw = (MetaCompWindow *)user_data;
/*
* Must reverse the effect of the effect once we hide the actor.
*/
clutter_actor_hide (CLUTTER_ACTOR (cw));
clutter_actor_set_opacity (CLUTTER_ACTOR (cw), cw->priv->opacity);
clutter_actor_set_scale (CLUTTER_ACTOR (cw), 1.0, 1.0);
clutter_actor_move_anchor_point_from_gravity (CLUTTER_ACTOR (cw),
CLUTTER_GRAVITY_NORTH_WEST);
}
static void
2008-08-20 04:33:39 -04:00
clutter_cmp_minimize_window (MetaCompositor *compositor, MetaWindow *window)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
2008-08-20 04:33:39 -04:00
MetaCompWindow *cw;
MetaCompScreen *info;
2008-08-20 04:33:39 -04:00
MetaScreen *screen;
MetaFrame *f = meta_window_get_frame (window);
screen = meta_window_get_screen (window);
info = meta_screen_get_compositor_data (screen);
/* Chances are we actually get the window frame here */
cw = find_window_for_screen (screen,
f ? meta_frame_get_xwindow (f) :
meta_window_get_xwindow (window));
if (!cw)
return;
meta_verbose ("Animating minimize of 0x%x\n",
(guint)meta_window_get_xwindow (window));
cw->priv->minimize_in_progress = TRUE;
clutter_actor_move_anchor_point_from_gravity (CLUTTER_ACTOR (cw),
CLUTTER_GRAVITY_SOUTH_WEST);
clutter_effect_fade (info->minimize_effect,
CLUTTER_ACTOR (cw),
0,
on_minimize_effect_complete,
(gpointer)cw);
clutter_effect_scale (info->minimize_effect,
CLUTTER_ACTOR (cw),
0.0,
0.0,
NULL,
NULL);
#endif
}
static MetaCompositor comp_info = {
clutter_cmp_destroy,
clutter_cmp_manage_screen,
clutter_cmp_unmanage_screen,
clutter_cmp_add_window,
clutter_cmp_remove_window,
clutter_cmp_set_updates,
clutter_cmp_process_event,
clutter_cmp_get_window_pixmap,
clutter_cmp_set_active_window,
clutter_cmp_destroy_window,
clutter_cmp_minimize_window
};
MetaCompositor *
meta_compositor_clutter_new (MetaDisplay *display)
{
#ifdef HAVE_COMPOSITE_EXTENSIONS
char *atom_names[] = {
"_XROOTPMAP_ID",
"_XSETROOT_ID",
"_NET_WM_WINDOW_OPACITY",
};
2008-08-20 04:33:39 -04:00
Atom atoms[G_N_ELEMENTS(atom_names)];
MetaCompositorClutter *clc;
2008-08-20 04:33:39 -04:00
MetaCompositor *compositor;
Display *xdisplay = meta_display_get_xdisplay (display);
if (!composite_at_least_version (display, 0, 3))
return NULL;
clc = g_new (MetaCompositorClutter, 1);
clc->compositor = comp_info;
compositor = (MetaCompositor *) clc;
clc->display = display;
meta_verbose ("Creating %d atoms\n", (int) G_N_ELEMENTS (atom_names));
XInternAtoms (xdisplay, atom_names, G_N_ELEMENTS (atom_names),
False, atoms);
clc->atom_x_root_pixmap = atoms[0];
clc->atom_x_set_root = atoms[1];
clc->atom_net_wm_window_opacity = atoms[2];
return compositor;
#else
return NULL;
#endif
}
/* ------------------------------- */
/* Shadow Generation */
typedef struct GaussianMap
{
int size;
double * data;
} GaussianMap;
static double
gaussian (double r, double x, double y)
{
return ((1 / (sqrt (2 * M_PI * r))) *
exp ((- (x * x + y * y)) / (2 * r * r)));
}
static GaussianMap *
make_gaussian_map (double r)
{
GaussianMap *c;
int size = ((int) ceil ((r * 3)) + 1) & ~1;
int center = size / 2;
int x, y;
double t = 0.0;
double g;
c = malloc (sizeof (GaussianMap) + size * size * sizeof (double));
c->size = size;
c->data = (double *) (c + 1);
for (y = 0; y < size; y++)
for (x = 0; x < size; x++)
{
g = gaussian (r, (double) (x - center), (double) (y - center));
t += g;
c->data[y * size + x] = g;
}
for (y = 0; y < size; y++)
for (x = 0; x < size; x++)
c->data[y*size + x] /= t;
return c;
}
static unsigned char
sum_gaussian (GaussianMap * map, double opacity,
int x, int y, int width, int height)
{
int fx, fy;
double * g_data;
double * g_line = map->data;
int g_size = map->size;
int center = g_size / 2;
int fx_start, fx_end;
int fy_start, fy_end;
double v;
unsigned int r;
/*
* Compute set of filter values which are "in range",
* that's the set with:
* 0 <= x + (fx-center) && x + (fx-center) < width &&
* 0 <= y + (fy-center) && y + (fy-center) < height
*
* 0 <= x + (fx - center) x + fx - center < width
* center - x <= fx fx < width + center - x
*/
fx_start = center - x;
if (fx_start < 0)
fx_start = 0;
fx_end = width + center - x;
if (fx_end > g_size)
fx_end = g_size;
fy_start = center - y;
if (fy_start < 0)
fy_start = 0;
fy_end = height + center - y;
if (fy_end > g_size)
fy_end = g_size;
g_line = g_line + fy_start * g_size + fx_start;
v = 0;
for (fy = fy_start; fy < fy_end; fy++)
{
g_data = g_line;
g_line += g_size;
for (fx = fx_start; fx < fx_end; fx++)
v += *g_data++;
}
if (v > 1)
v = 1;
v *= (opacity * 255.0);
r = (unsigned int) v;
return (unsigned char) r;
}
static unsigned char *
shadow_gaussian_make_tile ()
{
unsigned char * data;
int size;
int center;
int x, y;
unsigned char d;
int pwidth, pheight;
double opacity = SHADOW_OPACITY;
static GaussianMap * gaussian_map = NULL;
struct _mypixel
{
unsigned char r;
unsigned char g;
unsigned char b;
unsigned char a;
} * _d;
if (!gaussian_map)
gaussian_map =
make_gaussian_map (SHADOW_RADIUS);
size = gaussian_map->size;
center = size / 2;
/* Top & bottom */
pwidth = MAX_TILE_SZ;
pheight = MAX_TILE_SZ;
data = g_malloc0 (4 * TILE_WIDTH * TILE_HEIGHT);
_d = (struct _mypixel*) data;
/* N */
for (y = 0; y < pheight; y++)
{
d = sum_gaussian (gaussian_map, opacity,
center, y - center,
TILE_WIDTH, TILE_HEIGHT);
for (x = 0; x < pwidth; x++)
{
_d[y*3*pwidth + x + pwidth].r = 0;
_d[y*3*pwidth + x + pwidth].g = 0;
_d[y*3*pwidth + x + pwidth].b = 0;
_d[y*3*pwidth + x + pwidth].a = d;
}
}
/* S */
pwidth = MAX_TILE_SZ;
pheight = MAX_TILE_SZ;
for (y = 0; y < pheight; y++)
{
d = sum_gaussian (gaussian_map, opacity,
center, y - center,
TILE_WIDTH, TILE_HEIGHT);
for (x = 0; x < pwidth; x++)
{
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x + pwidth].r = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x + pwidth].g = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x + pwidth].b = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x + pwidth].a = d;
}
}
/* w */
pwidth = MAX_TILE_SZ;
pheight = MAX_TILE_SZ;
for (x = 0; x < pwidth; x++)
{
d = sum_gaussian (gaussian_map, opacity,
x - center, center,
TILE_WIDTH, TILE_HEIGHT);
for (y = 0; y < pheight; y++)
{
_d[y*3*pwidth + 3*pwidth*pheight + x].r = 0;
_d[y*3*pwidth + 3*pwidth*pheight + x].g = 0;
_d[y*3*pwidth + 3*pwidth*pheight + x].b = 0;
_d[y*3*pwidth + 3*pwidth*pheight + x].a = d;
}
}
/* E */
for (x = 0; x < pwidth; x++)
{
d = sum_gaussian (gaussian_map, opacity,
x - center, center,
TILE_WIDTH, TILE_HEIGHT);
for (y = 0; y < pheight; y++)
{
_d[y*3*pwidth + 3*pwidth*pheight + (pwidth-x-1) + 2*pwidth].r = 0;
_d[y*3*pwidth + 3*pwidth*pheight + (pwidth-x-1) + 2*pwidth].g = 0;
_d[y*3*pwidth + 3*pwidth*pheight + (pwidth-x-1) + 2*pwidth].b = 0;
_d[y*3*pwidth + 3*pwidth*pheight + (pwidth-x-1) + 2*pwidth].a = d;
}
}
/* NW */
pwidth = MAX_TILE_SZ;
pheight = MAX_TILE_SZ;
for (x = 0; x < pwidth; x++)
for (y = 0; y < pheight; y++)
{
d = sum_gaussian (gaussian_map, opacity,
x-center, y-center,
TILE_WIDTH, TILE_HEIGHT);
_d[y*3*pwidth + x].r = 0;
_d[y*3*pwidth + x].g = 0;
_d[y*3*pwidth + x].b = 0;
_d[y*3*pwidth + x].a = d;
}
/* SW */
for (x = 0; x < pwidth; x++)
for (y = 0; y < pheight; y++)
{
d = sum_gaussian (gaussian_map, opacity,
x-center, y-center,
TILE_WIDTH, TILE_HEIGHT);
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x].r = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x].g = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x].b = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + x].a = d;
}
/* SE */
for (x = 0; x < pwidth; x++)
for (y = 0; y < pheight; y++)
{
d = sum_gaussian (gaussian_map, opacity,
x-center, y-center,
TILE_WIDTH, TILE_HEIGHT);
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + (pwidth-x-1) +
2*pwidth].r = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + (pwidth-x-1) +
2*pwidth].g = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + (pwidth-x-1) +
2*pwidth].b = 0;
_d[(pheight-y-1)*3*pwidth + 6*pwidth*pheight + (pwidth-x-1) +
2*pwidth].a = d;
}
/* NE */
for (x = 0; x < pwidth; x++)
for (y = 0; y < pheight; y++)
{
d = sum_gaussian (gaussian_map, opacity,
x-center, y-center,
TILE_WIDTH, TILE_HEIGHT);
_d[y*3*pwidth + (pwidth - x - 1) + 2*pwidth].r = 0;
_d[y*3*pwidth + (pwidth - x - 1) + 2*pwidth].g = 0;
_d[y*3*pwidth + (pwidth - x - 1) + 2*pwidth].b = 0;
_d[y*3*pwidth + (pwidth - x - 1) + 2*pwidth].a = d;
}
/* center */
pwidth = MAX_TILE_SZ;
pheight = MAX_TILE_SZ;
d = sum_gaussian (gaussian_map, opacity,
center, center, TILE_WIDTH, TILE_HEIGHT);
for (x = 0; x < pwidth; x++)
for (y = 0; y < pheight; y++)
{
_d[y*3*pwidth + 3*pwidth*pheight + x + pwidth].r = 0;
_d[y*3*pwidth + 3*pwidth*pheight + x + pwidth].g = 0;
_d[y*3*pwidth + 3*pwidth*pheight + x + pwidth].b = 0;
_d[y*3*pwidth + 3*pwidth*pheight + x + pwidth].a = d;
}
return data;
}
#define TIDY_TYPE_TEXTURE_FRAME (tidy_texture_frame_get_type ())
#define TIDY_TEXTURE_FRAME(obj) \
(G_TYPE_CHECK_INSTANCE_CAST ((obj), \
TIDY_TYPE_TEXTURE_FRAME, TidyTextureFrame))
#define TIDY_TEXTURE_FRAME_CLASS(klass) \
(G_TYPE_CHECK_CLASS_CAST ((klass), \
TIDY_TYPE_TEXTURE_FRAME, TidyTextureFrameClass))
#define TIDY_IS_TEXTURE_FRAME(obj) \
(G_TYPE_CHECK_INSTANCE_TYPE ((obj), \
TIDY_TYPE_TEXTURE_FRAME))
#define TIDY_IS_TEXTURE_FRAME_CLASS(klass) \
(G_TYPE_CHECK_CLASS_TYPE ((klass), \
TIDY_TYPE_TEXTURE_FRAME))
#define TIDY_TEXTURE_FRAME_GET_CLASS(obj) \
(G_TYPE_INSTANCE_GET_CLASS ((obj), \
TIDY_TYPE_TEXTURE_FRAME, TidyTextureFrameClass))
typedef struct _TidyTextureFrame TidyTextureFrame;
typedef struct _TidyTextureFramePrivate TidyTextureFramePrivate;
typedef struct _TidyTextureFrameClass TidyTextureFrameClass;
struct _TidyTextureFrame
{
ClutterCloneTexture parent;
/*< priv >*/
TidyTextureFramePrivate *priv;
};
struct _TidyTextureFrameClass
{
ClutterCloneTextureClass parent_class;
/* padding for future expansion */
void (*_clutter_box_1) (void);
void (*_clutter_box_2) (void);
void (*_clutter_box_3) (void);
void (*_clutter_box_4) (void);
};
GType tidy_texture_frame_get_type (void) G_GNUC_CONST;
#define TIDY_PARAM_READABLE \
(G_PARAM_READABLE | \
G_PARAM_STATIC_NICK | G_PARAM_STATIC_NAME | G_PARAM_STATIC_BLURB)
#define TIDY_PARAM_READWRITE \
(G_PARAM_READABLE | G_PARAM_WRITABLE | \
G_PARAM_STATIC_NICK | G_PARAM_STATIC_NAME | G_PARAM_STATIC_BLURB)
enum
{
PROP_0,
PROP_LEFT,
PROP_TOP,
PROP_RIGHT,
PROP_BOTTOM
};
G_DEFINE_TYPE (TidyTextureFrame,
tidy_texture_frame,
CLUTTER_TYPE_CLONE_TEXTURE);
#define TIDY_TEXTURE_FRAME_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE ((obj), TIDY_TYPE_TEXTURE_FRAME, TidyTextureFramePrivate))
struct _TidyTextureFramePrivate
{
gint left, top, right, bottom;
};
static void
tidy_texture_frame_paint (ClutterActor *self)
{
TidyTextureFramePrivate *priv = TIDY_TEXTURE_FRAME (self)->priv;
ClutterCloneTexture *clone_texture = CLUTTER_CLONE_TEXTURE (self);
ClutterTexture *parent_texture;
guint width, height;
guint tex_width, tex_height;
guint ex, ey;
ClutterFixed tx1, ty1, tx2, ty2;
ClutterColor col = { 0xff, 0xff, 0xff, 0xff };
CoglHandle cogl_texture;
priv = TIDY_TEXTURE_FRAME (self)->priv;
/* no need to paint stuff if we don't have a texture */
parent_texture = clutter_clone_texture_get_parent_texture (clone_texture);
if (!parent_texture)
return;
/* parent texture may have been hidden, so need to make sure it gets
* realized
*/
if (!CLUTTER_ACTOR_IS_REALIZED (parent_texture))
clutter_actor_realize (CLUTTER_ACTOR (parent_texture));
cogl_texture = clutter_texture_get_cogl_texture (parent_texture);
if (cogl_texture == COGL_INVALID_HANDLE)
return;
cogl_push_matrix ();
tex_width = cogl_texture_get_width (cogl_texture);
tex_height = cogl_texture_get_height (cogl_texture);
clutter_actor_get_size (self, &width, &height);
tx1 = CLUTTER_INT_TO_FIXED (priv->left) / tex_width;
tx2 = CLUTTER_INT_TO_FIXED (tex_width - priv->right) / tex_width;
ty1 = CLUTTER_INT_TO_FIXED (priv->top) / tex_height;
ty2 = CLUTTER_INT_TO_FIXED (tex_height - priv->bottom) / tex_height;
2008-08-18 07:02:14 -04:00
col.alpha = clutter_actor_get_paint_opacity (self);
cogl_color (&col);
ex = width - priv->right;
if (ex < 0)
ex = priv->right; /* FIXME ? */
ey = height - priv->bottom;
if (ey < 0)
ey = priv->bottom; /* FIXME ? */
#define FX(x) CLUTTER_INT_TO_FIXED(x)
/* top left corner */
cogl_texture_rectangle (cogl_texture,
0,
0,
FX(priv->left), /* FIXME: clip if smaller */
FX(priv->top),
0,
0,
tx1,
ty1);
/* top middle */
cogl_texture_rectangle (cogl_texture,
FX(priv->left),
FX(priv->top),
FX(ex),
0,
tx1,
0,
tx2,
ty1);
/* top right */
cogl_texture_rectangle (cogl_texture,
FX(ex),
0,
FX(width),
FX(priv->top),
tx2,
0,
CFX_ONE,
ty1);
/* mid left */
cogl_texture_rectangle (cogl_texture,
0,
FX(priv->top),
FX(priv->left),
FX(ey),
0,
ty1,
tx1,
ty2);
/* center */
cogl_texture_rectangle (cogl_texture,
FX(priv->left),
FX(priv->top),
FX(ex),
FX(ey),
tx1,
ty1,
tx2,
ty2);
/* mid right */
cogl_texture_rectangle (cogl_texture,
FX(ex),
FX(priv->top),
FX(width),
FX(ey),
tx2,
ty1,
CFX_ONE,
ty2);
/* bottom left */
cogl_texture_rectangle (cogl_texture,
0,
FX(ey),
FX(priv->left),
FX(height),
0,
ty2,
tx1,
CFX_ONE);
/* bottom center */
cogl_texture_rectangle (cogl_texture,
FX(priv->left),
FX(ey),
FX(ex),
FX(height),
tx1,
ty2,
tx2,
CFX_ONE);
/* bottom right */
cogl_texture_rectangle (cogl_texture,
FX(ex),
FX(ey),
FX(width),
FX(height),
tx2,
ty2,
CFX_ONE,
CFX_ONE);
cogl_pop_matrix ();
}
static void
tidy_texture_frame_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
TidyTextureFrame *ctexture = TIDY_TEXTURE_FRAME (object);
TidyTextureFramePrivate *priv = ctexture->priv;
switch (prop_id)
{
case PROP_LEFT:
priv->left = g_value_get_int (value);
break;
case PROP_TOP:
priv->top = g_value_get_int (value);
break;
case PROP_RIGHT:
priv->right = g_value_get_int (value);
break;
case PROP_BOTTOM:
priv->bottom = g_value_get_int (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
tidy_texture_frame_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
TidyTextureFrame *ctexture = TIDY_TEXTURE_FRAME (object);
TidyTextureFramePrivate *priv = ctexture->priv;
switch (prop_id)
{
case PROP_LEFT:
g_value_set_int (value, priv->left);
break;
case PROP_TOP:
g_value_set_int (value, priv->top);
break;
case PROP_RIGHT:
g_value_set_int (value, priv->right);
break;
case PROP_BOTTOM:
g_value_set_int (value, priv->bottom);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
tidy_texture_frame_class_init (TidyTextureFrameClass *klass)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
ClutterActorClass *actor_class = CLUTTER_ACTOR_CLASS (klass);
actor_class->paint = tidy_texture_frame_paint;
gobject_class->set_property = tidy_texture_frame_set_property;
gobject_class->get_property = tidy_texture_frame_get_property;
g_object_class_install_property
(gobject_class,
PROP_LEFT,
g_param_spec_int ("left",
"left",
"",
0, G_MAXINT,
0,
TIDY_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_TOP,
g_param_spec_int ("top",
"top",
"",
0, G_MAXINT,
0,
TIDY_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_BOTTOM,
g_param_spec_int ("bottom",
"bottom",
"",
0, G_MAXINT,
0,
TIDY_PARAM_READWRITE));
g_object_class_install_property
(gobject_class,
PROP_RIGHT,
g_param_spec_int ("right",
"right",
"",
0, G_MAXINT,
0,
TIDY_PARAM_READWRITE));
g_type_class_add_private (gobject_class, sizeof (TidyTextureFramePrivate));
}
static void
tidy_texture_frame_init (TidyTextureFrame *self)
{
TidyTextureFramePrivate *priv;
self->priv = priv = TIDY_TEXTURE_FRAME_GET_PRIVATE (self);
}
2008-08-20 04:33:39 -04:00
static ClutterActor*
tidy_texture_frame_new (ClutterTexture *texture,
gint left,
gint top,
gint right,
gint bottom)
{
g_return_val_if_fail (texture == NULL || CLUTTER_IS_TEXTURE (texture), NULL);
return g_object_new (TIDY_TYPE_TEXTURE_FRAME,
"parent-texture", texture,
"left", left,
"top", top,
"right", right,
"bottom", bottom,
NULL);
}