aboutsummaryrefslogblamecommitdiffstats
path: root/calendar/modules/e-memo-shell-content.c
blob: a7b50fc9671450649401e9b8c839f98ea093a5dc (plain) (tree)



























































































                                                                           





                                                                         
                                                    




















































































                                                                                    
















                                                                            
                                                    























































































































































































































































































































































                                                                                 
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* e-memo-shell-content.h
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * 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., 51 Franklin Street, Fifth Floor,
 * Boston, MA 02110-1301, USA.
 */

#include "e-memo-shell-content.h"

#include <glib/gi18n.h>

#include "e-util/gconf-bridge.h"

#include "calendar/gui/calendar-config.h"
#include "calendar/gui/e-memo-table.h"
#include "calendar/gui/e-memo-table-config.h"

#include "widgets/menus/gal-view-etable.h"

#define E_MEMO_SHELL_CONTENT_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), E_TYPE_MEMO_SHELL_CONTENT, EMemoShellContentPrivate))

#define E_MEMO_TABLE_DEFAULT_STATE \
    "<?xml version=\"1.0\"?>" \
    "<ETableState>" \
    "  <column source=\"1\"/>" \
    "  <column source=\"0\"/>" \
    "  <column source=\"2\"/>" \
    "  <grouping/>" \
    "</ETableState>"

struct _EMemoShellContentPrivate {
    GtkWidget *paned;
    GtkWidget *memo_table;
    GtkWidget *memo_preview;

    EMemoTableConfig *table_config;
    GalViewInstance *view_instance;

    gchar *current_uid;
};

enum {
    PROP_0,
    PROP_PREVIEW_VISIBLE
};

enum {
    TARGET_VCALENDAR
};

static GtkTargetEntry drag_types[] = {
    { "text/calendar", 0, TARGET_VCALENDAR },
    { "text/x-calendar", 0, TARGET_VCALENDAR }
};

static gpointer parent_class;

static void
memo_shell_content_changed_cb (EMemoShellContent *memo_shell_content,
                               GalViewInstance *view_instance)
{
    EShellView *shell_view;
    EShellContent *shell_content;
    gchar *view_id;

    shell_content = E_SHELL_CONTENT (memo_shell_content);
    shell_view = e_shell_content_get_shell_view (shell_content);
    view_id = gal_view_instance_get_current_view_id (view_instance);
    e_shell_view_set_view_id (shell_view, view_id);
    g_free (view_id);
}

static void
memo_shell_content_display_view_cb (EMemoShellContent *memo_shell_content,
                                    GalView *gal_view)
{
    EMemoTable *memo_table;
    ETable *table;

    if (!GAL_IS_VIEW_ETABLE (gal_view))
        return;

    memo_table = E_MEMO_TABLE (memo_shell_content->priv->memo_table);
    table = e_memo_table_get_table (memo_table);

    gal_view_etable_attach_table (GAL_VIEW_ETABLE (gal_view), table);
}

static void
memo_shell_content_table_drag_data_get_cb (EMemoShellContent *memo_shell_content,
                                           gint row,
                                           gint col,
                                           GdkDragContext *context,
                                           GtkSelectionData *selection_data,
                                           guint info,
                                           guint time)
{
    /* FIXME */
}

static void
memo_shell_content_table_drag_data_delete_cb (EMemoShellContent *memo_shell_content,
                                              gint row,
                                              gint col,
                                              GdkDragContext *context)
{
    /* Moved components are deleted from source immediately when moved,
     * because some of them can be part of destination source, and we
     * don't want to delete not-moved tasks.  There is no such information
     * which event has been moved and which not, so skip this method. */
}

static void
memo_shell_content_cursor_change_cb (EMemoShellContent *memo_shell_content,
                                     gint row,
                                     ETable *table)
{
    EMemoPreview *memo_preview;
    EMemoTable *memo_table;
    ECalModel *model;
    ECalModelComponent *comp_data;
    ECalComponent *comp;
    const gchar *uid;

    memo_preview = E_MEMO_PREVIEW (memo_shell_content->priv->memo_preview);
    memo_table = E_MEMO_TABLE (memo_shell_content->priv->memo_table);

    if (e_table_selected_count (table) != 1) {
        e_memo_preview_clear (memo_preview);
        return;
    }

    model = e_memo_table_get_model (memo_table);
    row = e_table_get_cursor_row (table);
    comp_data = e_cal_model_get_component_at (model, row);

    comp = e_cal_component_new ();
    e_cal_component_set_icalcomponent (
        comp, icalcomponent_new_clone (comp_data->icalcomp));
    e_memo_preview_display (memo_preview, comp_data->client, comp);

    e_cal_component_get_uid (comp, &uid);
    g_free (memo_shell_content->priv->current_uid);
    memo_shell_content->priv->current_uid = g_strdup (uid);

    g_object_unref (comp);
}

static void
memo_shell_content_selection_change_cb (EMemoShellContent *memo_shell_content,
                                        ETable *table)
{
    EMemoPreview *memo_preview;

    memo_preview = E_MEMO_PREVIEW (memo_shell_content->priv->memo_preview);

    /* XXX Old code emits a "selection-changed" signal here. */

    if (e_table_selected_count (table) != 1)
        e_memo_preview_clear (memo_preview);
}

static void
memo_shell_content_model_row_changed_cb (EMemoShellContent *memo_shell_content,
                                         gint row,
                                         ETableModel *model)
{
    ECalModelComponent *comp_data;
    EMemoTable *memo_table;
    ETable *table;
    const gchar *current_uid;
    const gchar *uid;

    current_uid = memo_shell_content->priv->current_uid;
    if (current_uid == NULL)
        return;

    comp_data = e_cal_model_get_component_at (E_CAL_MODEL (model), row);
    if (comp_data == NULL)
        return;

    uid = icalcomponent_get_uid (comp_data->icalcomp);
    if (g_strcmp0 (uid, current_uid) != 0)
        return;

    memo_table = E_MEMO_TABLE (memo_shell_content->priv->memo_table);
    table = e_memo_table_get_table (memo_table);

    memo_shell_content_cursor_change_cb (memo_shell_content, 0, table);
}

static void
memo_shell_content_set_property (GObject *object,
                                 guint property_id,
                                 const GValue *value,
                                 GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_PREVIEW_VISIBLE:
            e_memo_shell_content_set_preview_visible (
                E_MEMO_SHELL_CONTENT (object),
                g_value_get_boolean (value));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
memo_shell_content_get_property (GObject *object,
                                 guint property_id,
                                 GValue *value,
                                 GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_PREVIEW_VISIBLE:
            g_value_set_boolean (
                value, e_memo_shell_content_get_preview_visible (
                E_MEMO_SHELL_CONTENT (object)));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
memo_shell_content_dispose (GObject *object)
{
    EMemoShellContentPrivate *priv;

    priv = E_MEMO_SHELL_CONTENT_GET_PRIVATE (object);

    if (priv->paned != NULL) {
        g_object_unref (priv->paned);
        priv->paned = NULL;
    }

    if (priv->memo_table != NULL) {
        g_object_unref (priv->memo_table);
        priv->memo_table = NULL;
    }

    if (priv->memo_preview != NULL) {
        g_object_unref (priv->memo_preview);
        priv->memo_preview = NULL;
    }

    if (priv->table_config != NULL) {
        g_object_unref (priv->table_config);
        priv->table_config = NULL;
    }

    if (priv->view_instance != NULL) {
        g_object_unref (priv->view_instance);
        priv->view_instance = NULL;
    }

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

static void
memo_shell_content_finalize (GObject *object)
{
    EMemoShellContentPrivate *priv;

    priv = E_MEMO_SHELL_CONTENT_GET_PRIVATE (object);

    g_free (priv->current_uid);

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

static void
memo_shell_content_constructed (GObject *object)
{
    EMemoShellContentPrivate *priv;
    EShellContent *shell_content;
    EShellView *shell_view;
    EShellViewClass *shell_view_class;
    GalViewCollection *view_collection;
    GalViewInstance *view_instance;
    ECalModel *model;
    ETable *table;
    GConfBridge *bridge;
    GtkWidget *container;
    GtkWidget *widget;
    const gchar *key;

    priv = E_MEMO_SHELL_CONTENT_GET_PRIVATE (object);

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

    shell_content = E_SHELL_CONTENT (object);
    shell_view = e_shell_content_get_shell_view (shell_content);
    shell_view_class = E_SHELL_VIEW_GET_CLASS (shell_view);
    view_collection = shell_view_class->view_collection;

    /* Load the view instance. */

    view_instance = gal_view_instance_new (view_collection, NULL);
    g_signal_connect_swapped (
        view_instance, "changed",
        G_CALLBACK (memo_shell_content_changed_cb),
        object);
    g_signal_connect_swapped (
        view_instance, "display-view",
        G_CALLBACK (memo_shell_content_display_view_cb),
        object);
    gal_view_instance_load (view_instance);
    priv->view_instance = view_instance;

    /* Build content widgets. */

    container = GTK_WIDGET (object);

    widget = gtk_vpaned_new ();
    gtk_container_add (GTK_CONTAINER (container), widget);
    priv->paned = g_object_ref (widget);
    gtk_widget_show (widget);

    container = widget;

    widget = e_memo_table_new (shell_view);
    gtk_paned_add1 (GTK_PANED (container), widget);
    priv->memo_table = g_object_ref (widget);
    gtk_widget_show (widget);

    widget = gtk_scrolled_window_new (NULL, NULL);
    gtk_scrolled_window_set_policy (
        GTK_SCROLLED_WINDOW (widget),
        GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
    gtk_scrolled_window_set_shadow_type (
        GTK_SCROLLED_WINDOW (widget), GTK_SHADOW_IN);
    gtk_paned_add2 (GTK_PANED (container), widget);

    container = widget;

    widget = e_memo_preview_new ();
    e_memo_preview_set_default_timezone (
        E_MEMO_PREVIEW (widget),
        calendar_config_get_icaltimezone ());
    gtk_container_add (GTK_CONTAINER (container), widget);
    priv->memo_preview = g_object_ref (widget);
    gtk_widget_show (widget);

    /* Configure the memo table. */

    widget = E_MEMO_TABLE (priv->memo_table)->etable;
    table = e_table_scrolled_get_table (E_TABLE_SCROLLED (widget));
    model = e_memo_table_get_model (E_MEMO_TABLE (priv->memo_table));

    priv->table_config = e_memo_table_config_new (
        E_MEMO_TABLE (priv->memo_table));

    e_table_set_state (table, E_MEMO_TABLE_DEFAULT_STATE);

    e_table_drag_source_set (
        table, GDK_BUTTON1_MASK,
        drag_types, G_N_ELEMENTS (drag_types),
        GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_ASK);

    g_signal_connect_swapped (
        table, "table-drag-data-get",
        G_CALLBACK (memo_shell_content_table_drag_data_get_cb),
        object);

    g_signal_connect_swapped (
        table, "table-drag-data-delete",
        G_CALLBACK (memo_shell_content_table_drag_data_delete_cb),
        object);

    g_signal_connect_swapped (
        table, "cursor-change",
        G_CALLBACK (memo_shell_content_cursor_change_cb),
        object);

    g_signal_connect_swapped (
        table, "selection-change",
        G_CALLBACK (memo_shell_content_selection_change_cb),
        object);

    g_signal_connect_swapped (
        model, "model-row-changed",
        G_CALLBACK (memo_shell_content_model_row_changed_cb),
        object);

    /* Bind GObject properties to GConf keys. */

    bridge = gconf_bridge_get ();

    object = G_OBJECT (priv->paned);
    key = "/apps/evolution/calendar/display/memo_vpane_position";
    gconf_bridge_bind_property_delayed (bridge, key, object, "position");
}

static void
memo_shell_content_class_init (EMemoShellContentClass *class)
{
    GObjectClass *object_class;

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

    object_class = G_OBJECT_CLASS (class);
    object_class->set_property = memo_shell_content_set_property;
    object_class->get_property = memo_shell_content_get_property;
    object_class->dispose = memo_shell_content_dispose;
    object_class->finalize = memo_shell_content_finalize;
    object_class->constructed = memo_shell_content_constructed;

    g_object_class_install_property (
        object_class,
        PROP_PREVIEW_VISIBLE,
        g_param_spec_boolean (
            "preview-visible",
            _("Preview is Visible"),
            _("Whether the preview pane is visible"),
            TRUE,
            G_PARAM_READWRITE));
}

static void
memo_shell_content_init (EMemoShellContent *memo_shell_content)
{
    memo_shell_content->priv =
        E_MEMO_SHELL_CONTENT_GET_PRIVATE (memo_shell_content);

    /* Postpone widget construction until we have a shell view. */
}

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

    if (G_UNLIKELY (type == 0)) {
        static const GTypeInfo type_info = {
            sizeof (EMemoShellContentClass),
            (GBaseInitFunc) NULL,
            (GBaseFinalizeFunc) NULL,
            (GClassInitFunc) memo_shell_content_class_init,
            (GClassFinalizeFunc) NULL,
            NULL,  /* class_data */
            sizeof (EMemoShellContent),
            0,     /* n_preallocs */
            (GInstanceInitFunc) memo_shell_content_init,
            NULL   /* value_table */
        };

        type = g_type_register_static (
            E_TYPE_SHELL_CONTENT, "EMemoShellContent",
            &type_info, 0);
    }

    return type;
}

GtkWidget *
e_memo_shell_content_new (EShellView *shell_view)
{
    g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL);

    return g_object_new (
        E_TYPE_MEMO_SHELL_CONTENT,
        "shell-view", shell_view, NULL);
}

EMemoPreview *
e_memo_shell_content_get_memo_preview (EMemoShellContent *memo_shell_content)
{
    g_return_val_if_fail (
        E_IS_MEMO_SHELL_CONTENT (memo_shell_content), NULL);

    return E_MEMO_PREVIEW (memo_shell_content->priv->memo_preview);
}

EMemoTable *
e_memo_shell_content_get_memo_table (EMemoShellContent *memo_shell_content)
{
    g_return_val_if_fail (
        E_IS_MEMO_SHELL_CONTENT (memo_shell_content), NULL);

    return E_MEMO_TABLE (memo_shell_content->priv->memo_table);
}

GalViewInstance *
e_memo_shell_content_get_view_instance (EMemoShellContent *memo_shell_content)
{
    g_return_val_if_fail (
        E_IS_MEMO_SHELL_CONTENT (memo_shell_content), NULL);

    return memo_shell_content->priv->view_instance;
}

gboolean
e_memo_shell_content_get_preview_visible (EMemoShellContent *memo_shell_content)
{
    GtkPaned *paned;
    GtkWidget *child;

    g_return_val_if_fail (
        E_IS_MEMO_SHELL_CONTENT (memo_shell_content), FALSE);

    paned = GTK_PANED (memo_shell_content->priv->paned);
    child = gtk_paned_get_child2 (paned);

    return GTK_WIDGET_VISIBLE (child);
}

void
e_memo_shell_content_set_preview_visible (EMemoShellContent *memo_shell_content,
                                          gboolean preview_visible)
{
    GtkPaned *paned;
    GtkWidget *child;

    g_return_if_fail (E_IS_MEMO_SHELL_CONTENT (memo_shell_content));

    paned = GTK_PANED (memo_shell_content->priv->paned);
    child = gtk_paned_get_child2 (paned);

    if (preview_visible)
        gtk_widget_show (child);
    else
        gtk_widget_hide (child);

    g_object_notify (G_OBJECT (memo_shell_content), "preview-visible");
}