aboutsummaryrefslogblamecommitdiffstats
path: root/shell/e-folder-list.c
blob: fb199be2a98064325daaf2fbe3c1d4740d80db3b (plain) (tree)























                                                                           




                         
                                
 

                             
 
                        


                                             



                                      
                            
 
                      













                                            


                            









                                       
                                                               




                                 



































                                                                               
                                                                               





                                                    
                                                                                 









                                                                     


                                        
                                                 


           
                                       
 
                                                  
 
                                
                                                
                                                   


                                   
 

                                                                    


















                                                                                



                                                


                                                 

                              
                                          
                                                                         

                                      
                                 
                                                       
                                                                                    

                      
                                                                               




                      



                                              


                                                 


                                                             
                      

                                                                                                    

                      
                                                                               






                                                  
                                   

                                 


                                                      
 
                     
 


                                                                




                                          







                                                                                      

                           






                                                                          


                                                                        
                                                                  
                                                                      




                                                                                  
                                                                                


                                                                                   






                                                             







                                                








                                                
                                      
                          





                                                

                                    






















                                                                                         
 

                                                                  
                                                                      










                                                                                                           
                                                      



                                                                                              





                                                                                                              
                                            
                                        
                                        




































                                                                       

                                                                          
                                                                    











                                                                       
        






















                                                                                          
                                                                                    












                                                                                                      
                                                                                                                          

                                                                 
                                                                   
                                                                    
                                                                      
                                                                        
                                                                          


                                                                                                               



                                                               
 

                                                                   
                                                                     
 




                                         
                                         
 
                           




                               






                                                                






                                                   






                                                                               






                                                                                                                      






















































                                                                                            
          
                                                                 
 

                                                                                        




                                                                      
                                                                                         
 
                              
                                   
 
                                                                                                           









                                                                  



                                               


                                                                

                                                                                                  





                                                                                                            
                                        
                                        














                                                                           


                                                                                                                   







                                     
                                                         





















                                               
                                                                                         








                                                                               
                                                                                      
 


                          
 
                                    
 



                                                                   
 


                                                                                             












                                                                                      


                                                                                                                  
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* 
 * e-folder-list.c
 * Copyright 2000, 2001, Ximian, Inc.
 *
 * Authors:
 *   Chris Lahey <clahey@ximian.com>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Library General Public
 * License, version 2, as published by the Free Software Foundation.
 *
 * This library 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
 * Library General Public License for more details.
 *
 * You should have received a copy of the GNU Library General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
 * 02111-1307, USA.
 */

#include <config.h>

#include <string.h>

#include <gtk/gtkframe.h>

#include <libgnome/gnome-i18n.h>

#include <libxml/parser.h>
#include <libxml/xmlmemory.h>

#include <glade/glade.h>

#include <gal/e-table/e-table-memory-store.h>
#include <gal/util/e-xml-utils.h>
#include <gal/widgets/e-gui-utils.h>
#include <gal/widgets/e-option-menu.h>

#include "e-folder-list.h"
#include "e-shell-marshal.h"

#include "Evolution.h"

static GtkVBoxClass *parent_class = NULL;
#define PARENT_TYPE (gtk_vbox_get_type ())

enum {
    CHANGED,
    OPTION_MENU_CHANGED,
    LAST_SIGNAL
};

static guint signals [LAST_SIGNAL] = { 0, };

/* The arguments we take */
enum {
    PROP_0,
    PROP_TITLE,
    PROP_POSSIBLE_TYPES,
};

struct _EFolderListPrivate {
    GladeXML *gui;

    ETableScrolled *scrolled_table;
    char *title;
    GtkFrame *frame;
    ETableMemoryStore *model;
    EvolutionShellClient *client;
    GNOME_Evolution_StorageRegistry corba_storage_registry;
    EOptionMenu *option_menu;

    char **possible_types;
};


static GNOME_Evolution_Folder *
get_folder_for_uri (EFolderList *efl,
            const char *uri)
{
    EFolderListPrivate *priv = efl->priv;
    CORBA_Environment ev;
    GNOME_Evolution_Folder *folder;

    CORBA_exception_init (&ev);
    folder = GNOME_Evolution_StorageRegistry_getFolderByUri (
        priv->corba_storage_registry, uri, &ev);
    if (ev._major != CORBA_NO_EXCEPTION)
        folder = CORBA_OBJECT_NIL;
    CORBA_exception_free (&ev);

    return folder;
}

static char *
create_display_string (EFolderList *efl, char *folder_uri, char *folder_name)
{
    char *storage_lname, *p;
    char *label_text;

    storage_lname = NULL;
    p = strchr (folder_uri, '/');
    if (p) {
        p = strchr (p + 1, '/');
        if (p) {
            GNOME_Evolution_Folder *storage_folder;
            char *storage_uri;

            storage_uri = g_strndup (folder_uri,
                         p - folder_uri);
            storage_folder = get_folder_for_uri (efl, storage_uri);
            storage_lname = g_strdup (storage_folder->displayName);
            CORBA_free (storage_folder);
            g_free (storage_uri);
        }
    }

    if (storage_lname) {
        label_text = g_strdup_printf (_("\"%s\" in \"%s\""), folder_name,
                          storage_lname);
        g_free (storage_lname);
    } else
        label_text = g_strdup_printf ("\"%s\"", folder_name);

    return label_text;
}



static void
e_folder_list_changed (EFolderList *efl)
{
    g_signal_emit (efl, signals[CHANGED], 0);
}

static void
e_folder_list_dispose (GObject *object)
{
    EFolderList *efl = E_FOLDER_LIST (object);

    if (efl->priv != NULL) {
        g_object_unref (efl->priv->gui);
        g_object_unref (efl->priv->client);
        g_free (efl->priv);
        efl->priv = NULL;
    }

    if (G_OBJECT_CLASS (parent_class)->dispose)
        (* G_OBJECT_CLASS (parent_class)->dispose) (object);
}

static void
set_frame_label (EFolderList *efl)
{
    GtkFrame *frame = efl->priv->frame;
    char *title = efl->priv->title;
    if (frame) {
        if (title) {
            gtk_frame_set_label (frame, title);
            gtk_frame_set_shadow_type (frame, GTK_SHADOW_ETCHED_IN);
        } else {
            gtk_frame_set_label (frame, "");
            gtk_frame_set_shadow_type (frame, GTK_SHADOW_NONE);
        }
    }
}

static void
e_folder_list_set_property (GObject *object,
                guint property_id,
                const GValue *value,
                GParamSpec *pspec)
{
    EFolderList *efl = E_FOLDER_LIST(object);

    switch (property_id) {
    case PROP_TITLE:
        g_free (efl->priv->title);
        efl->priv->title = g_strdup (g_value_get_string (value));
        set_frame_label (efl);
        break;
    case PROP_POSSIBLE_TYPES:
        g_strfreev (efl->priv->possible_types);
        efl->priv->possible_types = e_strdupv (g_value_get_pointer (value));
        break;
    default:
        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
        break;
    }
}

static void
e_folder_list_get_property (GObject *object,
                guint property_id,
                GValue *value,
                GParamSpec *pspec)
{
    EFolderList *efl = E_FOLDER_LIST(object);

    switch (property_id) {
    case PROP_TITLE:
        g_value_set_string (value, efl->priv->title);
        break;
    case PROP_POSSIBLE_TYPES:
        g_value_set_pointer (value, e_strdupv ((const gchar **) efl->priv->possible_types));
        break;
    default:
        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
        break;
    }
}

static void
e_folder_list_class_init (EFolderListClass *klass)
{
    GObjectClass *object_class;
    GtkVBoxClass *vbox_class;

    object_class = (GObjectClass*) klass;
    vbox_class = (GtkVBoxClass *) klass;
    parent_class = g_type_class_ref (PARENT_TYPE);

    glade_init();

    object_class->set_property = e_folder_list_set_property;
    object_class->get_property = e_folder_list_get_property;
    object_class->dispose      = e_folder_list_dispose;

    klass->changed             = NULL;
    klass->option_menu_changed = NULL;

    signals [OPTION_MENU_CHANGED] =
        g_signal_new ("option_menu_changed",
                  E_OBJECT_CLASS_TYPE (object_class),
                  G_SIGNAL_RUN_LAST,
                  G_STRUCT_OFFSET (EFolderListClass, option_menu_changed),
                  NULL, NULL,
                  e_shell_marshal_NONE__INT,
                  G_TYPE_NONE, 1,
                  G_TYPE_INT);

    signals [CHANGED] =
        g_signal_new ("changed",
                  E_OBJECT_CLASS_TYPE (object_class),
                  G_SIGNAL_RUN_LAST,
                  G_STRUCT_OFFSET (EFolderListClass, changed),
                  NULL, NULL,
                  e_shell_marshal_NONE__NONE,
                  G_TYPE_NONE, 0);

    E_OBJECT_CLASS_ADD_SIGNALS (object_class, signals, LAST_SIGNAL);

    g_object_class_install_property (object_class, PROP_TITLE,
                     g_param_spec_string ("title",
                                  NULL,
                                  NULL,
                                  NULL,
                                  G_PARAM_READWRITE));
    g_object_class_install_property (object_class, PROP_POSSIBLE_TYPES,
                     g_param_spec_pointer ("possible_types",
                                   NULL,
                                   NULL,
                                   G_PARAM_READWRITE));
}

#define SPEC    "<ETableSpecification cursor-mode=\"line\"" \
    "             selection-mode=\"browse\"" \
    "                     no-headers=\"true\"" \
        ">" \
    "  <ETableColumn model_col=\"0\"" \
    "            expansion=\"0.0\"" \
    "                cell=\"pixbuf\"" \
    "                minimum_width=\"18\"" \
    "                resizable=\"false\"" \
    "            _title=\"icon\"" \
    "                compare=\"string\"" \
        "                search=\"string\"/>" \
    "  <ETableColumn model_col=\"1\"" \
    "            expansion=\"1.0\"" \
    "                cell=\"string\"" \
    "                minimum_width=\"32\"" \
    "                resizable=\"true\"" \
    "            _title=\"blah\"" \
    "                compare=\"string\"" \
        "                search=\"string\"/>" \
    "  <ETableState>" \
    "    <column source=\"0\"/>" \
    "    <column source=\"1\"/>" \
    "    <grouping>" \
    "    </grouping>" \
    "  </ETableState>" \
    "</ETableSpecification>"


static ETableMemoryStoreColumnInfo columns[] = {
    E_TABLE_MEMORY_STORE_PIXBUF,
    E_TABLE_MEMORY_STORE_STRING,
    E_TABLE_MEMORY_STORE_STRING,
    E_TABLE_MEMORY_STORE_STRING,
    E_TABLE_MEMORY_STORE_STRING,
    E_TABLE_MEMORY_STORE_TERMINATOR
};

GtkWidget *
create_custom_optionmenu (char *name, char *string1, char *string2, int int1, int int2);

GtkWidget *
create_custom_optionmenu (char *name, char *string1, char *string2, int int1, int int2)
{
    return e_option_menu_new (NULL);
}

GtkWidget *
create_custom_folder_list (char *name, char *string1, char *string2, int int1, int int2);

GtkWidget *
create_custom_folder_list (char *name, char *string1, char *string2, int int1, int int2)
{
    ETableModel *model;
    GtkWidget *scrolled;

    model = e_table_memory_store_new (columns);
    scrolled = e_table_scrolled_new (model, NULL, SPEC, NULL);
    g_object_set_data (G_OBJECT (scrolled), "table-model", model);
    return scrolled;
}

static void
add_clicked (GtkButton *button, EFolderList *efl)
{
    GNOME_Evolution_Folder *folder;

    evolution_shell_client_user_select_folder (efl->priv->client,
                           GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (efl))),
                           _("Add a Folder"),
                           "",
                           (const gchar **) efl->priv->possible_types,
                           &folder);

    if (folder != NULL) {
        GdkPixbuf *pixbuf;
        char *display_string = create_display_string (efl, folder->evolutionUri, folder->displayName);

        pixbuf = evolution_shell_client_get_pixbuf_for_type (efl->priv->client, folder->type, TRUE);
        e_table_memory_store_insert (efl->priv->model, -1, NULL, pixbuf, display_string,
                         folder->displayName, folder->evolutionUri, folder->physicalUri);
        e_folder_list_changed (efl);
        g_object_unref (pixbuf);
        g_free (display_string);
    }
}

static void
make_list (int model_row, gpointer closure)
{
    GList **list = closure;
    *list = g_list_prepend (*list, GINT_TO_POINTER (model_row));
}

static void
remove_row (gpointer data, gpointer user_data)
{
    ETableMemoryStore *etms = user_data;
    int row = GPOINTER_TO_INT (data);

    e_table_memory_store_remove (etms, row);
}

static void
remove_clicked (GtkButton *button, EFolderList *efl)
{
    GList *list = NULL;
    ETable *table;

    table = e_table_scrolled_get_table (efl->priv->scrolled_table);

    e_table_selected_row_foreach (table, make_list, &list);

    g_list_foreach (list, remove_row, efl->priv->model);

    g_list_free (list);

    e_folder_list_changed (efl);
}

static void
optionmenu_changed (EOptionMenu *option_menu, int value, EFolderList *efl)
{
    g_signal_emit (efl, signals[OPTION_MENU_CHANGED], 0, value);
}

static void
update_buttons (EFolderList *efl)
{
    int cursor_row;
    int selection_count;
    ETable *table;

    table = e_table_scrolled_get_table (efl->priv->scrolled_table);
    cursor_row = e_table_get_cursor_row (table);
    selection_count = e_table_selected_count (table);
    
    e_glade_xml_set_sensitive (efl->priv->gui, "button-remove", selection_count >= 1);
}

static void
cursor_changed (ESelectionModel *selection_model, int row, int col, EFolderList *efl)
{
    update_buttons (efl);
}

static void
selection_changed (ESelectionModel *selection_model, EFolderList *efl)
{
    update_buttons (efl);
}

static void
e_folder_list_init (EFolderList *efl)
{
    GladeXML *gui;
    ESelectionModel *selection_model;

    efl->priv = g_new (EFolderListPrivate, 1);

    gui = glade_xml_new (EVOLUTION_GLADEDIR "/e-folder-list.glade", NULL, NULL);
    efl->priv->gui = gui;

    efl->priv->title = NULL;
    efl->priv->frame = GTK_FRAME (glade_xml_get_widget(gui, "frame-toplevel"));
    if (efl->priv->frame == NULL)
        return;

    gtk_widget_reparent(GTK_WIDGET (efl->priv->frame),
                GTK_WIDGET (efl));

    efl->priv->option_menu = E_OPTION_MENU(glade_xml_get_widget (gui, "custom-optionmenu"));
    e_folder_list_set_option_menu_strings (efl, NULL);
    efl->priv->scrolled_table = E_TABLE_SCROLLED(glade_xml_get_widget(gui, "custom-folder-list"));
    efl->priv->model = E_TABLE_MEMORY_STORE (g_object_get_data (G_OBJECT (efl->priv->scrolled_table), "table-model"));

    e_glade_xml_connect_widget (gui, "button-add", "clicked",
                    G_CALLBACK (add_clicked), efl);
    e_glade_xml_connect_widget (gui, "button-remove", "clicked",
                    G_CALLBACK (remove_clicked), efl);
    e_glade_xml_connect_widget (gui, "custom-optionmenu", "changed",
                    G_CALLBACK (optionmenu_changed), efl);

    selection_model = e_table_get_selection_model (e_table_scrolled_get_table (efl->priv->scrolled_table));

    g_signal_connect (selection_model, "selection_changed",
              G_CALLBACK (selection_changed), efl);
    g_signal_connect (selection_model, "cursor_changed",
              G_CALLBACK (cursor_changed), efl);

    /* XXX libglade2 seems to not show custom widgets even when
       they're flagged Visible.*/
    gtk_widget_show_all (GTK_WIDGET (efl->priv->scrolled_table));

    efl->priv->possible_types = NULL;
    set_frame_label (efl);
}

EFolderListItem *
e_folder_list_parse_xml (const char *xml)
{
    xmlDoc *doc = NULL;
    xmlNode *root;
    xmlNode *node;
    int i;
    EFolderListItem *items;

    if (xml && *xml) {
        doc = xmlParseMemory (xml, strlen (xml));
        if (!doc)
            g_warning ("malformed EFolderList xml");
    }

    if (!doc) {
        items = g_new (EFolderListItem, 1);
        items[0].uri = NULL;
        items[0].physical_uri = NULL;
        items[0].display_name = NULL;
        return items;
    }

    root = xmlDocGetRootElement (doc);

    for (node = root->xmlChildrenNode, i = 0; node; node = node->next, i++)
        /* Intentionally empty */;

    items = g_new (EFolderListItem, i + 1);

    for (node = root->xmlChildrenNode, i = 0; node; node = node->next) {
        if (node->name && !strcmp (node->name, "folder")) {
            items[i].uri = e_xml_get_string_prop_by_name_with_default (node, "uri", "");
            items[i].physical_uri = e_xml_get_string_prop_by_name_with_default (node, "physical-uri", "");
            items[i].display_name = e_xml_get_string_prop_by_name_with_default (node, "display-name", "");
            i++;
        }
    }
    items[i].uri = NULL;
    items[i].physical_uri = NULL;
    items[i].display_name = NULL;

    xmlFreeDoc (doc);

    return items;
}

char *
e_folder_list_create_xml (EFolderListItem *items)
{
    xmlDoc *doc;
    xmlNode *root;
    char *xml;
    xmlChar *temp;
    int length;
    int i;

    doc = xmlNewDoc ("1.0");

    root = xmlNewDocNode (doc, NULL, "EvolutionFolderList", NULL);

    xmlDocSetRootElement (doc, root);

    for (i = 0; items[i].uri; i++) {
        xmlNode *node = xmlNewChild (root, NULL, "folder", NULL);
        e_xml_set_string_prop_by_name (node, "uri", items[i].uri);
        e_xml_set_string_prop_by_name (node, "physical-uri", items[i].physical_uri);
        e_xml_set_string_prop_by_name (node, "display-name", items[i].display_name);
    }

    xmlDocDumpMemory (doc, &temp, &length);

    xml = g_strdup (temp);

    xmlFree (temp);
    xmlFreeDoc (doc);

    return xml;
}

void
e_folder_list_free_items (EFolderListItem *items)
{
    int i;
    for (i = 0; items[i].uri; i++) {
        g_free (items[i].uri);
        g_free (items[i].physical_uri);
        g_free (items[i].display_name);
    }
    g_free (items);
}

GtkWidget*
e_folder_list_new (EvolutionShellClient *client, const char *xml)
{
    GtkWidget *widget = GTK_WIDGET (g_object_new (e_folder_list_get_type (), NULL));

    e_folder_list_construct (E_FOLDER_LIST (widget), client, xml);
    return widget;
}

GtkWidget*
e_folder_list_construct (EFolderList *efl, EvolutionShellClient *client, const char *xml)
{
    g_object_ref (client);
    efl->priv->client = client;

    efl->priv->corba_storage_registry = evolution_shell_client_get_storage_registry_interface (client);
    e_folder_list_set_xml (efl, xml);
    return GTK_WIDGET (efl);
}

void
e_folder_list_set_items (EFolderList *efl, EFolderListItem *items)
{
    int i;
    e_table_memory_store_clear (efl->priv->model);
    for (i = 0; items[i].uri; i++) {
        GNOME_Evolution_Folder *folder;
        GdkPixbuf *pixbuf;
        char *display_string;

        folder = get_folder_for_uri (efl, items[i].uri);
        if (!folder)
            continue;
        display_string = create_display_string (efl, items[i].uri, items[i].display_name);

        pixbuf = evolution_shell_client_get_pixbuf_for_type (efl->priv->client, folder->type, TRUE);
        
        e_table_memory_store_insert (efl->priv->model, -1, NULL,
                         pixbuf, display_string,
                         items[i].display_name, items[i].uri, items[i].physical_uri);
        CORBA_free (folder);
        g_object_unref (pixbuf);
        g_free (display_string);
    }
}

EFolderListItem *
e_folder_list_get_items (EFolderList *efl)
{
    EFolderListItem *items;
    int count;
    int i;

    count = e_table_model_row_count (E_TABLE_MODEL (efl->priv->model));

    items = g_new (EFolderListItem, count + 1);

    for (i = 0; i < count; i++) {
        items[i].display_name = g_strdup (e_table_model_value_at (E_TABLE_MODEL (efl->priv->model), 2, i));
        items[i].uri = g_strdup (e_table_model_value_at (E_TABLE_MODEL (efl->priv->model), 3, i));
        items[i].physical_uri = g_strdup (e_table_model_value_at (E_TABLE_MODEL (efl->priv->model), 4, i));
    }
    items[i].uri = NULL;
    items[i].physical_uri = NULL;

    return items;
}

void
e_folder_list_set_xml (EFolderList *efl, const char *xml)
{
    EFolderListItem *items;

    items = e_folder_list_parse_xml (xml);
    e_folder_list_set_items (efl, items);
    e_folder_list_free_items (items);
}

char *
e_folder_list_get_xml (EFolderList *efl)
{
    EFolderListItem *items;
    char *xml;

    items = e_folder_list_get_items (efl);
    xml = e_folder_list_create_xml (items);
    e_folder_list_free_items (items);

    return xml;
}

void
e_folder_list_set_option_menu_strings_from_array (EFolderList *efl, const char **strings)
{
    e_option_menu_set_strings_from_array (efl->priv->option_menu, strings);
    if (strings && *strings)
        gtk_widget_show (GTK_WIDGET (efl->priv->option_menu));
    else
        gtk_widget_hide (GTK_WIDGET (efl->priv->option_menu));
}

void
e_folder_list_set_option_menu_strings (EFolderList *efl, const char *first_label, ...)
{
    GPtrArray *labels;
    va_list args;
    char *s;

    labels = g_ptr_array_new ();

    va_start (args, first_label);
    for (s = (char *)first_label; s; s = va_arg (args, char *))
        g_ptr_array_add (labels, s);
    va_end (args);

    e_folder_list_set_option_menu_strings_from_array (efl, (const char **)labels->pdata);

    g_ptr_array_free (labels, TRUE);
}

int
e_folder_list_get_option_menu_value (EFolderList *efl)
{
    return e_option_menu_get_value (efl->priv->option_menu);
}

void
e_folder_list_set_option_menu_value (EFolderList *efl, int value)
{
    gtk_option_menu_set_history (GTK_OPTION_MENU (efl->priv->option_menu), value);
}


E_MAKE_TYPE (e_folder_list, "EFolderList", EFolderList, e_folder_list_class_init, e_folder_list_init, PARENT_TYPE)