f9454e29db
The "multiple plugins loaded at once" strategy was always a big fiction: while it may be viable if you're super careful, it's fragile and requires a bit of infrastructure that we would be better off without. Note that for simplicity, we're keeping the MetaPluginManager, but it only manages one plugin. A possible future cleanup would be to remove it entirely. https://bugzilla.gnome.org/show_bug.cgi?id=676855
334 lines
9.4 KiB
C
334 lines
9.4 KiB
C
/* -*- 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/meta-plugin.h>
|
|
#include "meta-plugin-manager.h"
|
|
#include <meta/screen.h>
|
|
#include <meta/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);
|
|
|
|
#define META_PLUGIN_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE ((obj), META_TYPE_PLUGIN, MetaPluginPrivate))
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_SCREEN,
|
|
PROP_DEBUG_MODE,
|
|
};
|
|
|
|
struct _MetaPluginPrivate
|
|
{
|
|
MetaScreen *screen;
|
|
|
|
gint running;
|
|
gboolean debug : 1;
|
|
};
|
|
|
|
static void
|
|
meta_plugin_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (object)->priv;
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_SCREEN:
|
|
priv->screen = g_value_get_object (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)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (object)->priv;
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_SCREEN:
|
|
g_value_set_object (value, priv->screen);
|
|
break;
|
|
case PROP_DEBUG_MODE:
|
|
g_value_set_boolean (value, priv->debug);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
meta_plugin_class_init (MetaPluginClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->set_property = meta_plugin_set_property;
|
|
gobject_class->get_property = meta_plugin_get_property;
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_SCREEN,
|
|
g_param_spec_object ("screen",
|
|
"MetaScreen",
|
|
"MetaScreen",
|
|
META_TYPE_SCREEN,
|
|
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));
|
|
}
|
|
|
|
static void
|
|
meta_plugin_init (MetaPlugin *self)
|
|
{
|
|
MetaPluginPrivate *priv;
|
|
|
|
self->priv = priv = META_PLUGIN_GET_PRIVATE (self);
|
|
}
|
|
|
|
gboolean
|
|
meta_plugin_running (MetaPlugin *plugin)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
|
|
|
|
return (priv->running > 0);
|
|
}
|
|
|
|
gboolean
|
|
meta_plugin_debug_mode (MetaPlugin *plugin)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
|
|
|
|
return priv->debug;
|
|
}
|
|
|
|
const MetaPluginInfo *
|
|
meta_plugin_get_info (MetaPlugin *plugin)
|
|
{
|
|
MetaPluginClass *klass = META_PLUGIN_GET_CLASS (plugin);
|
|
|
|
if (klass && klass->plugin_info)
|
|
return klass->plugin_info (plugin);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* _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++;
|
|
}
|
|
|
|
void
|
|
meta_plugin_switch_workspace_completed (MetaPlugin *plugin)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
|
|
|
|
MetaScreen *screen = priv->screen;
|
|
|
|
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)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
|
|
|
|
if (priv->running-- < 0)
|
|
{
|
|
g_warning ("Error in running effect accounting, adjusting.");
|
|
priv->running = 0;
|
|
}
|
|
|
|
if (!actor)
|
|
{
|
|
const MetaPluginInfo *info;
|
|
const gchar *name = NULL;
|
|
|
|
if (plugin && (info = meta_plugin_get_info (plugin)))
|
|
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);
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
|
|
/**
|
|
* 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
|
|
*/
|
|
MetaScreen *
|
|
meta_plugin_get_screen (MetaPlugin *plugin)
|
|
{
|
|
MetaPluginPrivate *priv = META_PLUGIN (plugin)->priv;
|
|
|
|
return priv->screen;
|
|
}
|