mutter/src/compositor/meta-plugin.c

567 lines
16 KiB
C
Raw Normal View History

2008-12-17 04:33:56 -05:00
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
/*
* Copyright (c) 2008 Intel Corp.
*
* Author: Tomas Frydrych <tf@linux.intel.com>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*/
#include "meta-plugin.h"
#include "meta-plugin-manager.h"
2008-12-17 04:33:56 -05:00
#include "screen.h"
#include "display.h"
#include <string.h>
#include <X11/Xlib.h>
#include <X11/extensions/Xfixes.h>
#include <X11/extensions/shape.h>
#include <clutter/x11/clutter-x11.h>
#include "compositor-private.h"
#include "meta-window-actor-private.h"
G_DEFINE_ABSTRACT_TYPE (MetaPlugin, meta_plugin, G_TYPE_OBJECT);
2008-12-17 04:33:56 -05:00
#define META_PLUGIN_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE ((obj), META_TYPE_PLUGIN, MetaPluginPrivate))
2008-12-17 04:33:56 -05:00
enum
{
PROP_0,
PROP_SCREEN,
PROP_FEATURES,
PROP_DISABLED,
PROP_DEBUG_MODE,
};
struct _MetaPluginPrivate
2008-12-17 04:33:56 -05:00
{
MetaScreen *screen;
gulong features;
gint running;
gboolean disabled : 1;
gboolean debug : 1;
};
static void
meta_plugin_set_features (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = plugin->priv;
MetaPluginClass *klass = META_PLUGIN_GET_CLASS (plugin);
2008-12-17 04:33:56 -05:00
priv->features = 0;
/*
* Feature flags: identify events that the plugin can handle; a plugin can
* handle one or more events.
*/
2008-12-17 04:33:56 -05:00
if (klass->minimize)
priv->features |= META_PLUGIN_MINIMIZE;
2008-12-17 04:33:56 -05:00
if (klass->maximize)
priv->features |= META_PLUGIN_MAXIMIZE;
2008-12-17 04:33:56 -05:00
if (klass->unmaximize)
priv->features |= META_PLUGIN_UNMAXIMIZE;
2008-12-17 04:33:56 -05:00
if (klass->map)
priv->features |= META_PLUGIN_MAP;
2008-12-17 04:33:56 -05:00
if (klass->destroy)
priv->features |= META_PLUGIN_DESTROY;
2008-12-17 04:33:56 -05:00
if (klass->switch_workspace)
priv->features |= META_PLUGIN_SWITCH_WORKSPACE;
2008-12-17 04:33:56 -05:00
}
static void
meta_plugin_constructed (GObject *object)
{
meta_plugin_set_features (META_PLUGIN (object));
if (G_OBJECT_CLASS (meta_plugin_parent_class)->constructed)
G_OBJECT_CLASS (meta_plugin_parent_class)->constructed (object);
}
static void
meta_plugin_dispose (GObject *object)
{
G_OBJECT_CLASS (meta_plugin_parent_class)->dispose (object);
}
static void
meta_plugin_finalize (GObject *object)
{
G_OBJECT_CLASS (meta_plugin_parent_class)->finalize (object);
}
2008-12-17 04:33:56 -05:00
static void
meta_plugin_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (object)->priv;
2008-12-17 04:33:56 -05:00
switch (prop_id)
{
case PROP_SCREEN:
priv->screen = g_value_get_object (value);
break;
case PROP_DISABLED:
priv->disabled = g_value_get_boolean (value);
break;
case PROP_DEBUG_MODE:
priv->debug = g_value_get_boolean (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
meta_plugin_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (object)->priv;
2008-12-17 04:33:56 -05:00
switch (prop_id)
{
case PROP_SCREEN:
g_value_set_object (value, priv->screen);
break;
case PROP_DISABLED:
g_value_set_boolean (value, priv->disabled);
break;
case PROP_DEBUG_MODE:
g_value_set_boolean (value, priv->debug);
break;
case PROP_FEATURES:
g_value_set_ulong (value, priv->features);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
meta_plugin_class_init (MetaPluginClass *klass)
2008-12-17 04:33:56 -05:00
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
gobject_class->constructed = meta_plugin_constructed;
gobject_class->finalize = meta_plugin_finalize;
gobject_class->dispose = meta_plugin_dispose;
gobject_class->set_property = meta_plugin_set_property;
gobject_class->get_property = meta_plugin_get_property;
2008-12-17 04:33:56 -05:00
g_object_class_install_property (gobject_class,
PROP_SCREEN,
g_param_spec_object ("screen",
"MetaScreen",
"MetaScreen",
META_TYPE_SCREEN,
G_PARAM_READWRITE));
2008-12-17 04:33:56 -05:00
g_object_class_install_property (gobject_class,
PROP_FEATURES,
g_param_spec_ulong ("features",
"Features",
"Plugin Features",
0 , G_MAXULONG, 0,
G_PARAM_READABLE));
g_object_class_install_property (gobject_class,
PROP_DISABLED,
g_param_spec_boolean ("disabled",
"Plugin disabled",
"Plugin disabled",
FALSE,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_DEBUG_MODE,
g_param_spec_boolean ("debug-mode",
"Debug Mode",
"Debug Mode",
FALSE,
G_PARAM_READABLE));
g_type_class_add_private (gobject_class, sizeof (MetaPluginPrivate));
2008-12-17 04:33:56 -05:00
}
static void
meta_plugin_init (MetaPlugin *self)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv;
2008-12-17 04:33:56 -05:00
self->priv = priv = META_PLUGIN_GET_PRIVATE (self);
2008-12-17 04:33:56 -05:00
}
gulong
meta_plugin_features (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return priv->features;
}
gboolean
meta_plugin_disabled (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return priv->disabled;
}
gboolean
meta_plugin_running (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return (priv->running > 0);
}
gboolean
meta_plugin_debug_mode (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return priv->debug;
}
const MetaPluginInfo *
meta_plugin_get_info (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginClass *klass = META_PLUGIN_GET_CLASS (plugin);
2008-12-17 04:33:56 -05:00
if (klass && klass->plugin_info)
return klass->plugin_info (plugin);
return NULL;
}
ClutterActor *
meta_plugin_get_overlay_group (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return meta_get_overlay_group_for_screen (priv->screen);
2008-12-17 04:33:56 -05:00
}
ClutterActor *
meta_plugin_get_stage (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return meta_get_stage_for_screen (priv->screen);
2008-12-17 04:33:56 -05:00
}
ClutterActor *
meta_plugin_get_window_group (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return meta_get_window_group_for_screen (priv->screen);
2008-12-17 04:33:56 -05:00
}
ClutterActor *
meta_plugin_get_background_actor (MetaPlugin *plugin)
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
return meta_get_background_actor_for_screen (priv->screen);
}
/**
* _meta_plugin_effect_started:
* @plugin: the plugin
*
* Mark that an effect has started for the plugin. This is called
* internally by MetaPluginManager.
*/
void
_meta_plugin_effect_started (MetaPlugin *plugin)
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
priv->running++;
}
2008-12-17 04:33:56 -05:00
void
meta_plugin_switch_workspace_completed (MetaPlugin *plugin)
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
MetaScreen *screen = meta_plugin_get_screen (plugin);
if (priv->running-- < 0)
{
g_warning ("Error in running effect accounting, adjusting.");
priv->running = 0;
}
meta_switch_workspace_completed (screen);
}
static void
meta_plugin_window_effect_completed (MetaPlugin *plugin,
MetaWindowActor *actor,
unsigned long event)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
if (priv->running-- < 0)
{
g_warning ("Error in running effect accounting, adjusting.");
priv->running = 0;
}
if (!actor)
{
const MetaPluginInfo *info;
2008-12-17 04:33:56 -05:00
const gchar *name = NULL;
if (plugin && (info = meta_plugin_get_info (plugin)))
2008-12-17 04:33:56 -05:00
name = info->name;
g_warning ("Plugin [%s] passed NULL for actor!",
name ? name : "unknown");
}
meta_window_actor_effect_completed (actor, event);
}
void
meta_plugin_minimize_completed (MetaPlugin *plugin,
MetaWindowActor *actor)
{
meta_plugin_window_effect_completed (plugin, actor, META_PLUGIN_MINIMIZE);
}
void
meta_plugin_maximize_completed (MetaPlugin *plugin,
MetaWindowActor *actor)
{
meta_plugin_window_effect_completed (plugin, actor, META_PLUGIN_MAXIMIZE);
}
void
meta_plugin_unmaximize_completed (MetaPlugin *plugin,
MetaWindowActor *actor)
{
meta_plugin_window_effect_completed (plugin, actor, META_PLUGIN_UNMAXIMIZE);
}
void
meta_plugin_map_completed (MetaPlugin *plugin,
MetaWindowActor *actor)
{
meta_plugin_window_effect_completed (plugin, actor, META_PLUGIN_MAP);
}
void
meta_plugin_destroy_completed (MetaPlugin *plugin,
MetaWindowActor *actor)
{
meta_plugin_window_effect_completed (plugin, actor, META_PLUGIN_DESTROY);
2008-12-17 04:33:56 -05:00
}
void
meta_plugin_query_screen_size (MetaPlugin *plugin,
int *width,
int *height)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
meta_screen_get_size (priv->screen, width, height);
}
void
meta_plugin_set_stage_reactive (MetaPlugin *plugin,
gboolean reactive)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
MetaScreen *screen = priv->screen;
if (reactive)
meta_set_stage_input_region (screen, None);
2008-12-17 04:33:56 -05:00
else
meta_empty_stage_input_region (screen);
2008-12-17 04:33:56 -05:00
}
void
meta_plugin_set_stage_input_area (MetaPlugin *plugin,
gint x, gint y, gint width, gint height)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
MetaScreen *screen = priv->screen;
MetaDisplay *display = meta_screen_get_display (screen);
Display *xdpy = meta_display_get_xdisplay (display);
XRectangle rect;
XserverRegion region;
rect.x = x;
rect.y = y;
rect.width = width;
rect.height = height;
region = XFixesCreateRegion (xdpy, &rect, 1);
meta_set_stage_input_region (screen, region);
2008-12-17 04:33:56 -05:00
XFixesDestroyRegion (xdpy, region);
}
void
meta_plugin_set_stage_input_region (MetaPlugin *plugin,
XserverRegion region)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
MetaScreen *screen = priv->screen;
2008-12-17 04:33:56 -05:00
meta_set_stage_input_region (screen, region);
2008-12-17 04:33:56 -05:00
}
/**
* meta_plugin_get_window_actors:
* @plugin: A #MetaPlugin
*
* This function returns all of the #MetaWindowActor objects referenced by Mutter, including
* override-redirect windows. The returned list is a snapshot of Mutter's current
* stacking order, with the topmost window last.
*
* The 'restacked' signal of #MetaScreen signals when this value has changed.
*
* Returns: (transfer none) (element-type MetaWindowActor): Windows in stacking order, topmost last
*/
2008-12-17 04:33:56 -05:00
GList *
meta_plugin_get_window_actors (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return meta_get_window_actors (priv->screen);
2008-12-17 04:33:56 -05:00
}
/**
* meta_plugin_begin_modal:
* @plugin: a #MetaPlugin
* @grab_window: the X window to grab the keyboard and mouse on
* @cursor: the cursor to use for the pointer grab, or None,
* to use the normal cursor for the grab window and
* its descendants.
* @options: flags that modify the behavior of the modal grab
* @timestamp: the timestamp used for establishing grabs
*
* This function is used to grab the keyboard and mouse for the exclusive
* use of the plugin. Correct operation requires that both the keyboard
* and mouse are grabbed, or thing will break. (In particular, other
* passive X grabs in Meta can trigger but not be handled by the normal
* keybinding handling code.) However, the plugin can establish the keyboard
* and/or mouse grabs ahead of time and pass in the
* %META_MODAL_POINTER_ALREADY_GRABBED and/or %META_MODAL_KEYBOARD_ALREADY_GRABBED
* options. This facility is provided for two reasons: first to allow using
* this function to establish modality after a passive grab, and second to
* allow using obscure features of XGrabPointer() and XGrabKeyboard() without
* having to add them to this API.
*
* Return value: whether we successfully grabbed the keyboard and
* mouse and made the plugin modal.
*/
gboolean
meta_plugin_begin_modal (MetaPlugin *plugin,
Window grab_window,
Cursor cursor,
MetaModalOptions options,
guint32 timestamp)
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
return meta_begin_modal_for_plugin (priv->screen, plugin,
grab_window, cursor, options, timestamp);
}
/**
* meta_plugin_end_modal
* @plugin: a #MetaPlugin
* @timestamp: the time used for releasing grabs
*
* Ends the modal operation begun with meta_plugin_begin_modal(). This
* ungrabs both the mouse and keyboard even when
* %META_MODAL_POINTER_ALREADY_GRABBED or
* %META_MODAL_KEYBOARD_ALREADY_GRABBED were provided as options
* when beginnning the modal operation.
*/
void
meta_plugin_end_modal (MetaPlugin *plugin,
guint32 timestamp)
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
meta_end_modal_for_plugin (priv->screen, plugin, timestamp);
}
2008-12-17 04:33:56 -05:00
Display *
meta_plugin_get_xdisplay (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
MetaDisplay *display = meta_screen_get_display (priv->screen);
Display *xdpy = meta_display_get_xdisplay (display);
2008-12-17 04:33:56 -05:00
return xdpy;
}
/**
* meta_plugin_get_screen:
* @plugin: a #MetaPlugin
*
* Gets the #MetaScreen corresponding to a plugin. Each plugin instance
* is associated with exactly one screen; if Metacity is managing
* multiple screens, multiple plugin instances will be created.
*
* Return value: (transfer none): the #MetaScreen for the plugin
*/
2008-12-17 04:33:56 -05:00
MetaScreen *
meta_plugin_get_screen (MetaPlugin *plugin)
2008-12-17 04:33:56 -05:00
{
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
2008-12-17 04:33:56 -05:00
return priv->screen;
}
/**
* meta_plugin_type_register:
* @plugin_type: a #MetaPlugin type
*
* Register @plugin_type as a compositor plugin type to be used.
* You must call this before calling meta_init().
*/
void
meta_plugin_type_register (GType plugin_type)
{
MetaPluginManager *plugin_manager;
plugin_manager = meta_plugin_manager_get_default ();
meta_plugin_manager_register (plugin_manager, plugin_type);
}