aboutsummaryrefslogblamecommitdiffstats
path: root/e-util/e-plugin-ui.c
blob: 9a9a4f23ed926f1d4ed902012848fd0cbbc62ef4 (plain) (tree)
1
2
3
4
5
6
7
8
9
  



                                                                



                                                                    
                                                  

                                                                   
                                                                             

                                                        


                        
 
                   
                         


                   



                                                             

                                                                    


                              
                                                        


                       
                                                        
                                                          


                                              




                                      
                                                 






                                                                            
 

















                                                                              







                                                                         


                              







                                                                           
                                                                      




































































                                                                          

  


                             
                                                       
                                                          
 
                             

                                                     
                                        
                                                   


           
                                                     
                                                        
                                                 
                                                    
 


                               
                               
                               
 
                                                
 


















                                                                            
                       

                           
                                                    
                                                         
 
                                        
                                                                





                                                    
                                                                       
         

 

                                             
                                                
                                         
 

                                   


                             



                                                             
                                                    
                                                   
 




                                                 
                        


           
                                              
                                                 
                                          
 

                                   

                    
                                          
                                                                  
 

                               
 



                                                            
 
                                                    
 


                               
 
                                
                                                       
 




                                                                     
 

                                 
 
                                                            

                                                            
 



                                                                         

 
           
                                               
                                                  

                                           
 




                                          
                                                                  













                                                            
 










                                                                     

                                                                       
 




                                                                              


           
                                           
 


                               
 
                                                     
 

                                                   
 
                                                            
                                             
                                                                  
         


           
                                            
 


                               
 
                                                      
 

                                                   
 
                                                            
                                             
                                                                          
         





                                         
                            
                            


                                                     

                                                               
                                                                 
                                     
                                                            

                                                                   
                                                    
                                               
                                              




















                                                                           


                                                              

                                    

                                  
                                










                                                                          





                                                                

                                                    

                                                                
                                                                                  
                                                                     


                                                                         


                                             
                                                            











                                         
                                                                
            
                                                                 























                                                                        
                              
                             





                                                




                                           

                                                                    

                                                         
                                          
                                        




























                                                                            
                                                     
                                              

                                                 

                            
                                                        
                                      






                                                    

                                                         


                                                                             
                                               



                                                        

                                                                

                                                                    
                                                                         



                                                                 
                                                                 
                 



         
                                                   



                                            
                                                        






                                                    

                                                         







                                                                             
                                                                        
                 


         
    
                                                    
                                             
 

                            
                                                        






                                                    
 




                                                                             
 


                                                        
                                                                               

                 
 
/*
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 */

#include "e-plugin-ui.h"

#include "e-util.h"
#include "e-ui-manager.h"

#include <string.h>

#define E_PLUGIN_UI_HOOK_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), E_TYPE_PLUGIN_UI_HOOK, EPluginUIHookPrivate))

#define E_PLUGIN_UI_DEFAULT_FUNC    "e_plugin_ui_init"
#define E_PLUGIN_UI_HOOK_CLASS_ID   "org.gnome.evolution.ui:1.0"

struct _EPluginUIHookPrivate {

    /* Table of GtkUIManager ID's to UI definitions.
     *
     * For example:
     *
     *     <hook class="org.gnome.evolution.ui:1.0">
     *       <ui-manager id="org.gnome.evolution.foo">
     *               ... UI definition ...
     *       </ui-manager>
     *     </hook>
     *
     * Results in:
     *
     *     g_hash_table_insert (
     *             ui_definitions,
     *             "org.gnome.evolution.foo",
     *             "... UI definition ...");
     *
     * See http://library.gnome.org/devel/gtk/unstable/GtkUIManager.html
     * for more information about UI definitions.  Note: the <ui> tag is
     * optional.
     */
    GHashTable *ui_definitions;

    /* Table of GtkUIManager ID's to callback function names.
     *
     * This stores the optional "callback" attribute in the <ui-manager>
     * element.  If not specified, it defaults to "e_plugin_ui_init".
     *
     * This is useful when extending the UI of multiple GtkUIManager IDs
     * from a single plugin.
     *
     * For example:
     *
     *     <hook class="org.gnome.evolution.ui:1.0">
     *       <ui-manager id="org.gnome.evolution.foo" callback="init_foo">
     *         ...
     *       </ui-manager>
     *       <ui-manager id="org.gnome.evolution.bar" callback="init_bar">
     *         ...
     *       </ui-manager>
     *     </hook>
     *
     * Results in:
     *
     *     g_hash_table_insert (
     *             callbacks, "org.gnome.evolution.foo", "init_foo");
     *
     *     g_hash_table_insert (
     *             callbacks, "org.gnome.evolution.bar", "init_bar");
     */
    GHashTable *callbacks;

    /* The registry is the heart of EPluginUI.  It tracks GtkUIManager
     * instances, GtkUIManager IDs, and UI merge IDs as a hash table of
     * hash tables:
     *
     *     GtkUIManager instance -> GtkUIManager ID -> UI Merge ID
     *
     * A GtkUIManager instance and ID form a unique key for looking up
     * UI merge IDs.  The reason both are needed is because the same
     * GtkUIManager instance can be registered under multiple IDs.
     *
     * This is done primarily to support shell views, which share a
     * common GtkUIManager instance for a particular shell window.
     * Each shell view registers the same GtkUIManager instance under
     * a unique ID:
     *
     *     "org.gnome.evolution.mail"      }
     *     "org.gnome.evolution.contacts"  }  aliases for a common
     *     "org.gnome.evolution.calendar"  }  GtkUIManager instance
     *     "org.gnome.evolution.memos"     }
     *     "org.gnome.evolution.tasks"     }
     *
     * Note: The shell window also registers the same GtkUIManager
     *       instance as "org.gnome.evolution.shell".
     *
     * This way, plugins that extend a shell view's UI will follow the
     * merging and unmerging of the shell view automatically.
     *
     * The presence or absence of GtkUIManager IDs in the registry is
     * significant.  Presence of a (instance, ID) pair indicates that
     * UI manager is active, absence indicates inactive.  Furthermore,
     * a non-zero merge ID for an active UI manager indicates the
     * plugin is enabled.  Zero indicates disabled.
     *
     * Here's a quick scenario to illustrate:
     *
     * Suppose we have a plugin that extends the mail shell view UI.
     * Its EPlugin definition file has this section:
     *
     *     <hook class="org.gnome.evolution.ui:1.0">
     *       <ui-manager id="org.gnome.evolution.mail">
     *               ... UI definition ...
     *       </ui-manager>
     *     </hook>
     *
     * The plugin is enabled and the active shell view is "mail".
     * Let "ManagerA" denote the common GtkUIManager instance for
     * this shell window.  Here's what happens to the registry as
     * the user performs various actions;
     *
     *     - Initial State                            Merge ID
     *                                                   V
     *       { "ManagerA", { "org.gnome.evolution.mail", 3 } }
     *
     *     - User Disables the Plugin
     *
     *       { "ManagerA", { "org.gnome.evolution.mail", 0 } }
     *
     *     - User Enables the Plugin
     *
     *       { "ManagerA", { "org.gnome.evolution.mail", 4 } }
     *
     *     - User Switches to Calendar View
     *
     *       { "ManagerA", { } }
     *
     *     - User Disables the Plugin
     *
     *       { "ManagerA", { } }
     *
     *     - User Switches to Mail View
     *
     *       { "ManagerA", { "org.gnome.evolution.mail", 0 } }
     *
     *     - User Enables the Plugin
     *
     *       { "ManagerA", { "org.gnome.evolution.mail", 5 } }
     */
    GHashTable *registry;
};

static gpointer parent_class;

static void
plugin_ui_hook_unregister_manager (EPluginUIHook *hook,
                                   EUIManager *ui_manager)
{
    GHashTable *registry;

    /* Note: Manager may already be finalized. */
    registry = hook->priv->registry;
    g_hash_table_remove (registry, ui_manager);
}

static void
plugin_ui_hook_register_manager (EPluginUIHook *hook,
                                 EUIManager *ui_manager,
                                 const gchar *id,
                                 gpointer user_data)
{
    EPlugin *plugin;
    EPluginUIInitFunc func;
    GHashTable *registry;
    GHashTable *hash_table;
    const gchar *func_name;

    plugin = ((EPluginHook *) hook)->plugin;

    hash_table = hook->priv->callbacks;
    func_name = g_hash_table_lookup (hash_table, id);

    if (func_name == NULL)
        func_name = E_PLUGIN_UI_DEFAULT_FUNC;

    func = e_plugin_get_symbol (plugin, func_name);

    if (func == NULL) {
        g_critical (
            "Plugin \"%s\" is missing a function named %s()",
            plugin->name, func_name);
        return;
    }

    /* Pass the manager and user_data to the plugin's callback function.
     * The plugin should install whatever GtkActions and GtkActionGroups
     * are neccessary to implement the actions in its UI definition. */
    if (!func (ui_manager, user_data))
        return;

    g_object_weak_ref (
        G_OBJECT (ui_manager), (GWeakNotify)
        plugin_ui_hook_unregister_manager, hook);

    registry = hook->priv->registry;
    hash_table = g_hash_table_lookup (registry, ui_manager);

    if (hash_table == NULL) {
        hash_table = g_hash_table_new_full (
            g_str_hash, g_str_equal,
            (GDestroyNotify) g_free,
            (GDestroyNotify) NULL);
        g_hash_table_insert (registry, ui_manager, hash_table);
    }
}

static guint
plugin_ui_hook_merge_ui (EPluginUIHook *hook,
                         EUIManager *ui_manager,
                         const gchar *id)
{
    GHashTable *hash_table;
    const gchar *ui_definition;
    guint merge_id;
    GError *error = NULL;

    hash_table = hook->priv->ui_definitions;
    ui_definition = g_hash_table_lookup (hash_table, id);
    g_return_val_if_fail (ui_definition != NULL, 0);

    merge_id = e_ui_manager_add_ui_from_string (
        ui_manager, ui_definition, &error);

    if (error != NULL) {
        g_warning ("%s", error->message);
        g_error_free (error);
    }

    return merge_id;
}

static void
plugin_ui_enable_manager (EPluginUIHook *hook,
                          EUIManager *ui_manager,
                          const gchar *id)
{
    GHashTable *hash_table;
    GHashTable *ui_definitions;
    GList *keys;

    hash_table = hook->priv->registry;
    hash_table = g_hash_table_lookup (hash_table, ui_manager);

    if (hash_table == NULL)
        return;

    if (id != NULL)
        keys = g_list_prepend (NULL, (gpointer) id);
    else
        keys = g_hash_table_get_keys (hash_table);

    ui_definitions = hook->priv->ui_definitions;

    while (keys != NULL) {
        guint merge_id;
        gpointer data;

        id = keys->data;
        keys = g_list_delete_link (keys, keys);

        if (g_hash_table_lookup (ui_definitions, id) == NULL)
            continue;

        data = g_hash_table_lookup (hash_table, id);
        merge_id = GPOINTER_TO_UINT (data);

        if (merge_id > 0)
            continue;

        if (((EPluginHook *) hook)->plugin->enabled)
            merge_id = plugin_ui_hook_merge_ui (
                hook, ui_manager, id);

        /* Merge ID will be 0 on error, which is what we want. */
        data = GUINT_TO_POINTER (merge_id);
        g_hash_table_insert (hash_table, g_strdup (id), data);
    }
}

static void
plugin_ui_disable_manager (EPluginUIHook *hook,
                           EUIManager *ui_manager,
                           const gchar *id,
                           gboolean remove)
{
    GHashTable *hash_table;
    GHashTable *ui_definitions;
    GList *keys;

    hash_table = hook->priv->registry;
    hash_table = g_hash_table_lookup (hash_table, ui_manager);

    if (hash_table == NULL)
        return;

    if (id != NULL)
        keys = g_list_prepend (NULL, (gpointer) id);
    else
        keys = g_hash_table_get_keys (hash_table);

    ui_definitions = hook->priv->ui_definitions;

    while (keys != NULL) {
        guint merge_id;
        gpointer data;

        id = keys->data;
        keys = g_list_delete_link (keys, keys);

        if (g_hash_table_lookup (ui_definitions, id) == NULL)
            continue;

        data = g_hash_table_lookup (hash_table, id);
        merge_id = GPOINTER_TO_UINT (data);

        /* Merge ID could be 0 if the plugin is disabled. */
        if (merge_id > 0)
            gtk_ui_manager_remove_ui (
                GTK_UI_MANAGER (ui_manager), merge_id);

        if (remove)
            g_hash_table_remove (hash_table, id);
        else
            g_hash_table_insert (hash_table, g_strdup (id), NULL);
    }
}

static void
plugin_ui_enable_hook (EPluginUIHook *hook)
{
    GHashTable *hash_table;
    GHashTableIter iter;
    gpointer key;

    /* Enable all GtkUIManagers for this hook. */

    hash_table = hook->priv->registry;
    g_hash_table_iter_init (&iter, hash_table);

    while (g_hash_table_iter_next (&iter, &key, NULL)) {
        EUIManager *ui_manager = key;
        plugin_ui_enable_manager (hook, ui_manager, NULL);
    }
}

static void
plugin_ui_disable_hook (EPluginUIHook *hook)
{
    GHashTable *hash_table;
    GHashTableIter iter;
    gpointer key;

    /* Disable all GtkUIManagers for this hook. */

    hash_table = hook->priv->registry;
    g_hash_table_iter_init (&iter, hash_table);

    while (g_hash_table_iter_next (&iter, &key, NULL)) {
        EUIManager *ui_manager = key;
        plugin_ui_disable_manager (hook, ui_manager, NULL, FALSE);
    }
}

static void
plugin_ui_hook_finalize (GObject *object)
{
    EPluginUIHookPrivate *priv;
    GHashTableIter iter;
    gpointer ui_manager;

    priv = E_PLUGIN_UI_HOOK_GET_PRIVATE (object);

    /* Remove weak reference callbacks to GtkUIManagers. */
    g_hash_table_iter_init (&iter, priv->registry);
    while (g_hash_table_iter_next (&iter, &ui_manager, NULL))
        g_object_weak_unref (
            G_OBJECT (ui_manager), (GWeakNotify)
            plugin_ui_hook_unregister_manager, object);

    g_hash_table_destroy (priv->ui_definitions);
    g_hash_table_destroy (priv->callbacks);
    g_hash_table_destroy (priv->registry);

    /* Chain up to parent's dispose() method. */
    G_OBJECT_CLASS (parent_class)->dispose (object);
}

static gint
plugin_ui_hook_construct (EPluginHook *hook,
                          EPlugin *plugin,
                          xmlNodePtr node)
{
    EPluginUIHookPrivate *priv;

    priv = E_PLUGIN_UI_HOOK_GET_PRIVATE (hook);

    /* XXX The EPlugin should be a property of EPluginHookClass.
     *     Then it could be passed directly to g_object_new() and
     *     we wouldn't have to chain up here. */

    /* Chain up to parent's construct() method. */
    E_PLUGIN_HOOK_CLASS (parent_class)->construct (hook, plugin, node);

    for (node = xmlFirstElementChild (node); node != NULL;
        node = xmlNextElementSibling (node)) {

        xmlNodePtr child;
        xmlBufferPtr buffer;
        GString *content;
        const gchar *temp;
        gchar *callback;
        gchar *id;

        if (strcmp ((gchar *) node->name, "ui-manager") != 0)
            continue;

        id = e_plugin_xml_prop (node, "id");
        if (id == NULL) {
            g_warning ("<ui-manager> requires 'id' property");
            continue;
        }

        callback = e_plugin_xml_prop (node, "callback");
        if (callback != NULL)
            g_hash_table_insert (
                priv->callbacks,
                g_strdup (id), callback);

        content = g_string_sized_new (1024);

        /* Extract the XML content below <ui-manager> */
        buffer = xmlBufferCreate ();
        for (child = node->children; child != NULL; child = child->next) {
            xmlNodeDump (buffer, node->doc, child, 2, 1);
            temp = (const gchar *) xmlBufferContent (buffer);
            g_string_append (content, temp);
        }

        g_hash_table_insert (
            priv->ui_definitions,
            id, g_string_free (content, FALSE));

        xmlBufferFree (buffer);
    }

    return 0;
}

static void
plugin_ui_hook_enable (EPluginHook *hook,
                       gint state)
{
    if (state)
        plugin_ui_enable_hook (E_PLUGIN_UI_HOOK (hook));
    else
        plugin_ui_disable_hook (E_PLUGIN_UI_HOOK (hook));
}

static void
plugin_ui_hook_class_init (EPluginUIHookClass *class)
{
    GObjectClass *object_class;
    EPluginHookClass *plugin_hook_class;

    parent_class = g_type_class_peek_parent (class);
    g_type_class_add_private (class, sizeof (EPluginUIHookPrivate));

    object_class = G_OBJECT_CLASS (class);
    object_class->finalize = plugin_ui_hook_finalize;

    plugin_hook_class = E_PLUGIN_HOOK_CLASS (class);
    plugin_hook_class->id = E_PLUGIN_UI_HOOK_CLASS_ID;
    plugin_hook_class->construct = plugin_ui_hook_construct;
    plugin_hook_class->enable = plugin_ui_hook_enable;
}

static void
plugin_ui_hook_init (EPluginUIHook *hook)
{
    GHashTable *ui_definitions;
    GHashTable *callbacks;
    GHashTable *registry;

    ui_definitions = g_hash_table_new_full (
        g_str_hash, g_str_equal,
        (GDestroyNotify) g_free,
        (GDestroyNotify) g_free);

    callbacks = g_hash_table_new_full (
        g_str_hash, g_str_equal,
        (GDestroyNotify) g_free,
        (GDestroyNotify) g_free);

    registry = g_hash_table_new (g_direct_hash, g_direct_equal);

    hook->priv = E_PLUGIN_UI_HOOK_GET_PRIVATE (hook);
    hook->priv->ui_definitions = ui_definitions;
    hook->priv->callbacks = callbacks;
    hook->priv->registry = registry;
}

GType
e_plugin_ui_hook_get_type (void)
{
    static GType type = 0;

    if (G_UNLIKELY (type == 0)) {
        static const GTypeInfo type_info = {
            sizeof (EPluginUIHookClass),
            (GBaseInitFunc) NULL,
            (GBaseFinalizeFunc) NULL,
            (GClassInitFunc) plugin_ui_hook_class_init,
            (GClassFinalizeFunc) NULL,
            NULL,  /* class_data */
            sizeof (EPluginUIHook),
            0,     /* n_preallocs */
            (GInstanceInitFunc) plugin_ui_hook_init,
            NULL   /* value_table */
        };

        type = g_type_register_static (
            E_TYPE_PLUGIN_HOOK, "EPluginUIHook", &type_info, 0);
    }

    return type;
}

void
e_plugin_ui_register_manager (EUIManager *ui_manager,
                              const gchar *id,
                              gpointer user_data)
{
    GSList *plugin_list;

    g_return_if_fail (E_IS_UI_MANAGER (ui_manager));
    g_return_if_fail (id != NULL);

    /* Loop over all installed plugins. */
    plugin_list = e_plugin_list_plugins ();
    while (plugin_list != NULL) {
        EPlugin *plugin = plugin_list->data;
        GSList *iter;

        plugin_list = g_slist_next (plugin_list);

        /* Look for hooks of type EPluginUIHook. */
        for (iter = plugin->hooks; iter != NULL; iter = iter->next) {
            EPluginUIHook *hook = iter->data;
            GHashTable *hash_table;

            if (!E_IS_PLUGIN_UI_HOOK (hook))
                continue;

            hash_table = hook->priv->ui_definitions;

            /* Check if the hook has a UI definition
             * for the GtkUIManager being registered. */
            if (g_hash_table_lookup (hash_table, id) == NULL)
                continue;

            /* Register the manager with the hook. */
            plugin_ui_hook_register_manager (
                hook, ui_manager, id, user_data);
        }
    }
}

void
e_plugin_ui_enable_manager (EUIManager *ui_manager,
                            const gchar *id)
{
    GSList *plugin_list;

    g_return_if_fail (E_IS_UI_MANAGER (ui_manager));
    g_return_if_fail (id != NULL);

    /* Loop over all installed plugins. */
    plugin_list = e_plugin_list_plugins ();
    while (plugin_list != NULL) {
        EPlugin *plugin = plugin_list->data;
        GSList *iter;

        plugin_list = g_slist_next (plugin_list);

        /* Look for hooks of type EPluginUIHook. */
        for (iter = plugin->hooks; iter != NULL; iter = iter->next) {
            EPluginUIHook *hook = iter->data;

            if (!E_IS_PLUGIN_UI_HOOK (hook))
                continue;

            plugin_ui_enable_manager (hook, ui_manager, id);
        }
    }
}

void
e_plugin_ui_disable_manager (EUIManager *ui_manager,
                             const gchar *id)
{
    GSList *plugin_list;

    g_return_if_fail (E_IS_UI_MANAGER (ui_manager));
    g_return_if_fail (id != NULL);

    /* Loop over all installed plugins. */
    plugin_list = e_plugin_list_plugins ();
    while (plugin_list != NULL) {
        EPlugin *plugin = plugin_list->data;
        GSList *iter;

        plugin_list = g_slist_next (plugin_list);

        /* Look for hooks of type EPluginUIHook. */
        for (iter = plugin->hooks; iter != NULL; iter = iter->next) {
            EPluginUIHook *hook = iter->data;

            if (!E_IS_PLUGIN_UI_HOOK (hook))
                continue;

            plugin_ui_disable_manager (hook, ui_manager, id, TRUE);
        }
    }
}