aboutsummaryrefslogblamecommitdiffstats
path: root/e-util/e-extension.c
blob: 75caf5adf46595774191f912ec13bc37e9d617eb (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

















                                                                             






















                                                                      














                                                     



                        






















































































































                                                                          







                                              






                                                                
/*
 * e-extension.c
 *
 * 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/>
 *
 */

/**
 * SECTION: e-extension
 * @short_description: abstract base class for extensions
 * @include: e-util/e-extension.h
 *
 * #EExtension provides a way to extend the functionality of objects
 * that implement the #EExtensible interface.  #EExtension subclasses
 * can target a particular extensible object type.  New instances of
 * an extensible object type get paired with a new instance of each
 * #EExtension subclass that targets the extensible object type.
 *
 * The first steps of writing a new extension are as follows:
 *
 * 1. Subclass #EExtension.
 *
 * 2. In the class initialization function, specify the #GType being
 *    extended.  The #GType must implement the #EExtensible interface.
 *
 * 3. Register the extension's own #GType.  If the extension is to
 *    be loaded dynamically using #GTypeModule, the type should be
 *    registered in the library module's e_module_load() function.
 **/

#include "e-extension.h"

#define E_EXTENSION_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), E_TYPE_EXTENSION, EExtensionPrivate))

struct _EExtensionPrivate {
    gpointer extensible;  /* weak pointer */
};

enum {
    PROP_0,
    PROP_EXTENSIBLE
};

G_DEFINE_ABSTRACT_TYPE (
    EExtension,
    e_extension,
    G_TYPE_OBJECT)

static void
extension_set_extensible (EExtension *extension,
                          EExtensible *extensible)
{
    EExtensionClass *class;
    GType extensible_type;

    g_return_if_fail (E_IS_EXTENSIBLE (extensible));
    g_return_if_fail (extension->priv->extensible == NULL);

    class = E_EXTENSION_GET_CLASS (extension);
    extensible_type = G_OBJECT_TYPE (extensible);

    /* Verify the EExtensible object is the type we want. */
    if (!g_type_is_a (extensible_type, class->extensible_type)) {
        g_warning ("%s is meant to extend %s but was given an %s",
            G_OBJECT_TYPE_NAME (extension),
            g_type_name (class->extensible_type),
            g_type_name (extensible_type));
        return;
    }

    extension->priv->extensible = extensible;

    g_object_add_weak_pointer (
        G_OBJECT (extensible), &extension->priv->extensible);
}

static void
extension_set_property (GObject *object,
                        guint property_id,
                        const GValue *value,
                        GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_EXTENSIBLE:
            extension_set_extensible (
                E_EXTENSION (object),
                g_value_get_object (value));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
extension_get_property (GObject *object,
                        guint property_id,
                        GValue *value,
                        GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_EXTENSIBLE:
            g_value_set_object (
                value, e_extension_get_extensible (
                E_EXTENSION (object)));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
extension_constructed (GObject *object)
{
    /* This allows subclasses to chain up safely since GObject
     * does not implement this method, and we might want to do
     * something here in the future. */
}

static void
extension_dispose (GObject *object)
{
    EExtensionPrivate *priv;

    priv = E_EXTENSION_GET_PRIVATE (object);

    if (priv->extensible != NULL) {
        g_object_remove_weak_pointer (
            G_OBJECT (priv->extensible), &priv->extensible);
        priv->extensible = NULL;
    }

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

static void
e_extension_class_init (EExtensionClass *class)
{
    GObjectClass *object_class;

    g_type_class_add_private (class, sizeof (EExtensionPrivate));

    object_class = G_OBJECT_CLASS (class);
    object_class->set_property = extension_set_property;
    object_class->get_property = extension_get_property;
    object_class->constructed = extension_constructed;
    object_class->dispose = extension_dispose;

    g_object_class_install_property (
        object_class,
        PROP_EXTENSIBLE,
        g_param_spec_object (
            "extensible",
            "Extensible Object",
            "The object being extended",
            E_TYPE_EXTENSIBLE,
            G_PARAM_READWRITE |
            G_PARAM_CONSTRUCT_ONLY));
}

static void
e_extension_init (EExtension *extension)
{
    extension->priv = E_EXTENSION_GET_PRIVATE (extension);
}

/**
 * e_extension_get_extensible:
 * @extension: an #EExtension
 *
 * Returns the object that @extension extends.
 *
 * Returns: the object being extended
 **/
EExtensible *
e_extension_get_extensible (EExtension *extension)
{
    g_return_val_if_fail (E_IS_EXTENSION (extension), NULL);

    return E_EXTENSIBLE (extension->priv->extensible);
}