/* -*- 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_gnome_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 (char *xml)
{
xmlDoc *doc;
xmlNode *root;
xmlNode *node;
int i;
EFolderListItem *items;
if (xml == NULL || *xml == 0) {
items = g_new (EFolderListItem, 1);
items[0].uri = NULL;
items[0].physical_uri = NULL;
items[0].display_name = NULL;
return items;
}
doc = xmlParseMemory (xml, strlen (xml));
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, i++) {
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", "");
}
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, 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, 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, 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, gchar **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, gchar *first_label, ...)
{
char **array;
GET_STRING_ARRAY_FROM_ELLIPSIS (array, first_label);
e_folder_list_set_option_menu_strings_from_array (efl, array);
g_free (array);
}
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)