aboutsummaryrefslogblamecommitdiffstats
path: root/mail/e-mail-shell-content.c
blob: f6b07ea7286d9d3fb9353696c0d1b386f9b95fe3 (plain) (tree)























                                                                               
                              
                                              

                                

                                            
 
                           
                              

                        
 





                                                                     
                               
 
                                     

                                       














                                    













                                                                                

























































                                                                          


                                                   






                                               




                                                     


















                                                         


                                          


                             
                                           






                                                            




                                                                    










                                                              

                                                  
                                              

                                                                         










                                                             

                           
                                    
                                                                      


                                                                     
 




                                                   








                                                                             



                                                             

















                                               
                          
                 
 
                                                                  






























































































                                                                                



                     



                                                             
                                                










                                                                            

                                                                        
                                                                          
 




































































                                                                       








                                                                            








                                                                               

































                                                                                












                                                                               



























































                                                                              





















                                                                                 





                                                             

















































































































                                                                             
/*
 * e-mail-shell-content.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/>  
 *
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

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

#include <glib/gi18n.h>
#include <camel/camel-store.h>
#include <libedataserver/e-data-server-util.h>

#include "e-util/gconf-bridge.h"
#include "widgets/menus/gal-view-etable.h"
#include "widgets/menus/gal-view-instance.h"

#include "em-folder-view.h"
#include "em-search-context.h"
#include "em-utils.h"
#include "mail-config.h"

#define E_MAIL_SHELL_CONTENT_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), E_TYPE_MAIL_SHELL_CONTENT, EMailShellContentPrivate))

struct _EMailShellContentPrivate {
    GtkWidget *paned;
    GtkWidget *folder_view;

    EMFormatHTMLDisplay *preview;
    GalViewInstance *view_instance;

    guint paned_binding_id;

    guint preview_visible   : 1;
    guint vertical_view : 1;
};

enum {
    PROP_0,
    PROP_PREVIEW_VISIBLE,
    PROP_VERTICAL_VIEW
};

static gpointer parent_class;

static void
mail_shell_content_display_view_cb (EMailShellContent *mail_shell_content,
                                    GalView *gal_view)
{
    EMFolderView *folder_view;

    folder_view = e_mail_shell_content_get_folder_view (mail_shell_content);

    if (GAL_IS_VIEW_ETABLE (gal_view))
        gal_view_etable_attach_tree (
            GAL_VIEW_ETABLE (gal_view),
            folder_view->list->tree);
}

static void
mail_shell_content_set_property (GObject *object,
                                 guint property_id,
                                 const GValue *value,
                                 GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_PREVIEW_VISIBLE:
            e_mail_shell_content_set_preview_visible (
                E_MAIL_SHELL_CONTENT (object),
                g_value_get_boolean (value));
            return;

        case PROP_VERTICAL_VIEW:
            e_mail_shell_content_set_vertical_view (
                E_MAIL_SHELL_CONTENT (object),
                g_value_get_boolean (value));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
mail_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_mail_shell_content_get_preview_visible (
                E_MAIL_SHELL_CONTENT (object)));
            return;

        case PROP_VERTICAL_VIEW:
            g_value_set_boolean (
                value,
                e_mail_shell_content_get_vertical_view (
                E_MAIL_SHELL_CONTENT (object)));
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
mail_shell_content_dispose (GObject *object)
{
    EMailShellContentPrivate *priv;

    priv = E_MAIL_SHELL_CONTENT_GET_PRIVATE (object);

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

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

    if (priv->preview != NULL) {
        g_object_unref (priv->preview);
        priv->preview = 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
mail_shell_content_finalize (GObject *object)
{
    EMailShellContentPrivate *priv;

    priv = E_MAIL_SHELL_CONTENT_GET_PRIVATE (object);

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

static void
mail_shell_content_constructed (GObject *object)
{
    EMailShellContentPrivate *priv;
    EShellContent *shell_content;
    EShellView *shell_view;
    EShellViewClass *shell_view_class;
    GConfBridge *bridge;
    GtkWidget *container;
    GtkWidget *widget;
    GalViewCollection *view_collection;
    const gchar *key;

    priv = E_MAIL_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;

    /* 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;

    /* XXX Kill EMFolderView? */
    priv->folder_view = em_folder_view_new ();
    g_object_ref_sink (priv->folder_view);
    widget = GTK_WIDGET (((EMFolderView *) priv->folder_view)->list);
    gtk_paned_add1 (GTK_PANED (container), 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);
    gtk_widget_show (widget);

    container = widget;

    /* XXX Kill EMFolderView? */
    priv->preview = ((EMFolderView *) priv->folder_view)->preview;
    widget = GTK_WIDGET (((EMFormatHTML *) priv->preview)->html);
    gtk_container_add (GTK_CONTAINER (container), widget);
    gtk_widget_show (widget);

    /* Load the view instance. */

    e_mail_shell_content_update_view_instance (
        E_MAIL_SHELL_CONTENT (object));

    /* Bind GObject properties to GConf keys. */

    bridge = gconf_bridge_get ();

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

static guint32
mail_shell_content_check_state (EShellContent *shell_content)
{
    EMailShellContent *mail_shell_content;
    EMFolderView *folder_view;
    GPtrArray *uids;
    CamelFolder *folder;
    CamelStore *store;
    const gchar *folder_uri;
    const gchar *tag;
    gboolean can_clear_flags = FALSE;
    gboolean can_flag_completed = FALSE;
    gboolean can_flag_for_followup = FALSE;
    gboolean has_deleted = FALSE;
    gboolean has_important = FALSE;
    gboolean has_junk = FALSE;
    gboolean has_not_junk = FALSE;
    gboolean has_read = FALSE;
    gboolean has_undeleted = FALSE;
    gboolean has_unimportant = FALSE;
    gboolean has_unread = FALSE;
    gboolean draft_or_outbox;
    guint32 state = 0;
    guint ii;

    mail_shell_content = E_MAIL_SHELL_CONTENT (shell_content);
    folder_view = e_mail_shell_content_get_folder_view (mail_shell_content);
    uids = message_list_get_selected (folder_view->list);
    folder_uri = folder_view->folder_uri;
    folder = folder_view->folder;
    store = CAMEL_STORE (folder->parent_store);

    draft_or_outbox =
        em_utils_folder_is_drafts (folder, folder_uri) ||
        em_utils_folder_is_outbox (folder, folder_uri);
    if (!draft_or_outbox) {
        has_junk = !(store->flags & CAMEL_STORE_VJUNK);
        has_not_junk = TRUE;
    }

    for (ii = 0; ii < uids->len; ii++) {
        CamelMessageInfo *info;
        guint32 flags;

        info = camel_folder_get_message_info (
            folder, uids->pdata[ii]);
        if (info == NULL)
            continue;

        flags = camel_message_info_flags (info);

        if (flags & CAMEL_MESSAGE_SEEN)
            has_read = TRUE;
        else
            has_unread = TRUE;

        if (flags & CAMEL_MESSAGE_DELETED)
            has_deleted = TRUE;
        else
            has_undeleted = TRUE;

        if (flags & CAMEL_MESSAGE_FLAGGED)
            has_important = TRUE;
        else
            has_unimportant = TRUE;

        tag = camel_message_info_user_tag (info, "follow-up");
        if (tag != NULL && *tag != '\0') {
            can_clear_flags = TRUE;
            tag = camel_message_info_user_tag (
                info, "completed-on");
            if (tag != NULL && *tag != '\0')
                can_flag_completed = TRUE;
        } else
            can_flag_for_followup = TRUE;
    }

    if (uids->len == 1)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_SINGLE;
    if (uids->len > 1)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_MULTIPLE;
    if (!draft_or_outbox && uids->len == 1)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_CAN_ADD_SENDER;
#if 0  /* FIXME */
    if (can_edit)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_CAN_EDIT;
#endif
    if (can_clear_flags)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_FLAG_CLEAR;
    if (can_flag_completed)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_FLAG_COMPLETED;
    if (can_flag_for_followup)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_FLAG_FOLLOWUP;
    if (has_deleted)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_DELETED;
    if (has_important)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_IMPORTANT;
    if (has_junk)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_JUNK;
    if (has_not_junk)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_NOT_JUNK;
    if (has_read)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_READ;
    if (has_undeleted)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_UNDELETED;
    if (has_unimportant)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_UNIMPORTANT;
    if (has_unread)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_UNREAD;
#if 0  /* FIXME */
    if (has_callto_uri)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_URI_CALLTO;
    if (has_http_uri)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_URI_HTTP;
    if (has_mailto_uri)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_HAS_URI_MAILTO;
    if (is_mailing_list)
        state |= E_MAIL_SHELL_CONTENT_SELECTION_IS_MAILING_LIST;
#endif

    em_utils_uids_free (uids);

    return state;
}

static void
mail_shell_content_class_init (EMailShellContentClass *class)
{
    GObjectClass *object_class;
    EShellContentClass *shell_content_class;

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

    object_class = G_OBJECT_CLASS (class);
    object_class->set_property = mail_shell_content_set_property;
    object_class->get_property = mail_shell_content_get_property;
    object_class->dispose = mail_shell_content_dispose;
    object_class->finalize = mail_shell_content_finalize;
    object_class->constructed = mail_shell_content_constructed;

    shell_content_class = E_SHELL_CONTENT_CLASS (class);
    shell_content_class->new_search_context = em_search_context_new;
    shell_content_class->check_state = mail_shell_content_check_state;

    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));

    g_object_class_install_property (
        object_class,
        PROP_VERTICAL_VIEW,
        g_param_spec_boolean (
            "vertical-view",
            _("Vertical View"),
            _("Whether vertical view is enabled"),
            FALSE,
            G_PARAM_READWRITE));
}

static void
mail_shell_content_init (EMailShellContent *mail_shell_content)
{
    mail_shell_content->priv =
        E_MAIL_SHELL_CONTENT_GET_PRIVATE (mail_shell_content);

    mail_shell_content->priv->preview_visible = TRUE;

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

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

    if (G_UNLIKELY (type == 0)) {
        static const GTypeInfo type_info = {
            sizeof (EMailShellContentClass),
            (GBaseInitFunc) NULL,
            (GBaseFinalizeFunc) NULL,
            (GClassInitFunc) mail_shell_content_class_init,
            (GClassFinalizeFunc) NULL,
            NULL,  /* class_data */
            sizeof (EMailShellContent),
            0,     /* n_preallocs */
            (GInstanceInitFunc) mail_shell_content_init,
            NULL   /* value_table */
        };

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

    return type;
}

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

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

EMFolderView *
e_mail_shell_content_get_folder_view (EMailShellContent *mail_shell_content)
{
    g_return_val_if_fail (
        E_IS_MAIL_SHELL_CONTENT (mail_shell_content), NULL);

    return EM_FOLDER_VIEW (mail_shell_content->priv->folder_view);
}

EMFormatHTMLDisplay *
e_mail_shell_content_get_preview_format (EMailShellContent *mail_shell_content)
{
    g_return_val_if_fail (
        E_IS_MAIL_SHELL_CONTENT (mail_shell_content), NULL);

    return mail_shell_content->priv->preview;
}

gboolean
e_mail_shell_content_get_preview_visible (EMailShellContent *mail_shell_content)
{
    g_return_val_if_fail (
        E_IS_MAIL_SHELL_CONTENT (mail_shell_content), FALSE);

    return mail_shell_content->priv->preview_visible;
}

void
e_mail_shell_content_set_preview_visible (EMailShellContent *mail_shell_content,
                                          gboolean preview_visible)
{
    GtkPaned *paned;
    GtkWidget *child;

    g_return_if_fail (E_IS_MAIL_SHELL_CONTENT (mail_shell_content));

    if (preview_visible == mail_shell_content->priv->preview_visible)
        return;

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

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

    mail_shell_content->priv->preview_visible = preview_visible;

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

GtkWidget *
e_mail_shell_content_get_preview_widget (EMailShellContent *mail_shell_content)
{
    EMFormatHTML *format;

    g_return_val_if_fail (
        E_IS_MAIL_SHELL_CONTENT (mail_shell_content), NULL);

    format = (EMFormatHTML *) mail_shell_content->priv->preview;

    return GTK_WIDGET (format->html);
}

gboolean
e_mail_shell_content_get_vertical_view (EMailShellContent *mail_shell_content)
{
    g_return_val_if_fail (
        E_IS_MAIL_SHELL_CONTENT (mail_shell_content), FALSE);

    return mail_shell_content->priv->vertical_view;
}

void
e_mail_shell_content_set_vertical_view (EMailShellContent *mail_shell_content,
                                        gboolean vertical_view)
{
    GConfBridge *bridge;
    GtkWidget *old_paned;
    GtkWidget *new_paned;
    GtkWidget *child1;
    GtkWidget *child2;
    guint binding_id;
    const gchar *key;

    g_return_if_fail (E_IS_MAIL_SHELL_CONTENT (mail_shell_content));

    if (vertical_view == mail_shell_content->priv->vertical_view)
        return;

    bridge = gconf_bridge_get ();
    old_paned = mail_shell_content->priv->paned;
    binding_id = mail_shell_content->priv->paned_binding_id;

    child1 = gtk_paned_get_child1 (GTK_PANED (old_paned));
    child2 = gtk_paned_get_child2 (GTK_PANED (old_paned));

    if (binding_id > 0)
        gconf_bridge_unbind (bridge, binding_id);

    if (vertical_view) {
        new_paned = gtk_hpaned_new ();
        key = "/apps/evolution/mail/display/hpaned_size";
    } else {
        new_paned = gtk_vpaned_new ();
        key = "/apps/evolution/mail/display/paned_size";
    }

    gtk_widget_reparent (child1, new_paned);
    gtk_widget_reparent (child2, new_paned);
    gtk_widget_show (new_paned);

    gtk_widget_destroy (old_paned);
    gtk_container_add (GTK_CONTAINER (mail_shell_content), new_paned);

    binding_id = gconf_bridge_bind_property_delayed (
        bridge, key, G_OBJECT (new_paned), "position");

    mail_shell_content->priv->vertical_view = vertical_view;
    mail_shell_content->priv->paned_binding_id = binding_id;
    mail_shell_content->priv->paned = g_object_ref (new_paned);

    g_object_notify (G_OBJECT (mail_shell_content), "vertical-view");
}

void
e_mail_shell_content_update_view_instance (EMailShellContent *mail_shell_content)
{
    EShellContent *shell_content;
    EShellView *shell_view;
    EShellViewClass *shell_view_class;
    GalViewCollection *view_collection;
    GalViewInstance *view_instance;
    EMFolderView *folder_view;
    gboolean outgoing_folder;
    gboolean show_vertical_view;
    gchar *view_id;

    g_return_if_fail (E_IS_MAIL_SHELL_CONTENT (mail_shell_content));

    shell_content = E_SHELL_CONTENT (mail_shell_content);
    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;

    folder_view = e_mail_shell_content_get_folder_view (mail_shell_content);

    /* If no folder is selected, return silently. */
    if (folder_view->folder == NULL)
        return;

    /* If we have a folder, we should also have a URI. */
    g_return_if_fail (folder_view->folder_uri != NULL);

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

    /* TODO: Should this go through the mail-config API? */
    view_id = mail_config_folder_to_safe_url (folder_view->folder);
    view_instance = e_shell_view_new_view_instance (shell_view, view_id);
    mail_shell_content->priv->view_instance = view_instance;

    show_vertical_view = folder_view->list_active &&
        e_mail_shell_content_get_vertical_view (mail_shell_content);

    if (show_vertical_view) {
        gchar *filename;
        gchar *safe_view_id;

        /* Force the view instance into vertical view. */

        g_free (view_instance->custom_filename);
        g_free (view_instance->current_view_filename);

        safe_view_id = g_strdup (view_id);
        e_filename_make_safe (safe_view_id);

        filename = g_strdup_printf (
            "custom_wide_view-%s.xml", safe_view_id);
        view_instance->custom_filename = g_build_filename (
            view_collection->local_dir, filename, NULL);
        g_free (filename);

        filename = g_strdup_printf (
            "current_wide_view-%s.xml", safe_view_id);
        view_instance->current_view_filename = g_build_filename (
            view_collection->local_dir, filename, NULL);
        g_free (filename);

        g_free (safe_view_id);
    }

    g_free (view_id);

    outgoing_folder =
        em_utils_folder_is_drafts (
            folder_view->folder, folder_view->folder_uri) ||
        em_utils_folder_is_outbox (
            folder_view->folder, folder_view->folder_uri) ||
        em_utils_folder_is_sent (
            folder_view->folder, folder_view->folder_uri);

    if (outgoing_folder) {
        if (show_vertical_view)
            gal_view_instance_set_default_view (
                view_instance, "Wide_View_Sent");
        else
            gal_view_instance_set_default_view (
                view_instance, "As_Sent_Folder");
    } else if (show_vertical_view) {
        gal_view_instance_set_default_view (
            view_instance, "Wide_View_Normal");
    }

    gal_view_instance_load (view_instance);

    if (!gal_view_instance_exists (view_instance)) {
        gchar *state_filename;

        state_filename = mail_config_folder_to_cachename (
            folder_view->folder, "et-header-");

        if (g_file_test (state_filename, G_FILE_TEST_IS_REGULAR)) {
            ETableSpecification *spec;
            ETableState *state;
            GalView *view;
            gchar *spec_filename;

            spec = e_table_specification_new ();
            spec_filename = g_build_filename (
                EVOLUTION_ETSPECDIR,
                "message-list.etspec",
                NULL);
            e_table_specification_load_from_file (
                spec, spec_filename);
            g_free (spec_filename);

            state = e_table_state_new ();
            view = gal_view_etable_new (spec, "");

            e_table_state_load_from_file (
                state, state_filename);
            gal_view_etable_set_state (
                GAL_VIEW_ETABLE (view), state);
            gal_view_instance_set_custom_view (
                view_instance, view);

            g_object_unref (state);
            g_object_unref (view);
            g_object_unref (spec);
        }

        g_free (state_filename);
    }

    g_signal_connect (
        view_instance, "display-view",
        G_CALLBACK (mail_shell_content_display_view_cb),
        mail_shell_content);

    mail_shell_content_display_view_cb (
        mail_shell_content,
        gal_view_instance_get_current_view (view_instance));
}