From bfe2227dd0f527b2c8da97d73ed2361a564f08bd Mon Sep 17 00:00:00 2001 From: Chris Toshok Date: Mon, 2 Oct 2000 20:06:11 +0000 Subject: subscribe-control.[ch] -> subscribe_dialog.[ch] 2000-10-02 Chris Toshok * Makefile.am (evolution_mail_SOURCES): subscribe-control.[ch] -> subscribe_dialog.[ch] * mail-callbacks.c (manage_subscriptions): subscribe_control -> subscribe_dialog. Also, pass the shell to subscribe_dialog_new. * mail-types.h: SubscribeControl -> SubscribeDialog. * subscribe-dialog.c, subscribe-dialog.h: rename from subscribe-control.[ch]. * subscribe-dialog.c (subscribe_dialog_construct): pass Evolution_Shell in. (subscribe_dialog_new): takes Evolution_Shell argument now. svn path=/trunk/; revision=5668 --- mail/ChangeLog | 15 ++ mail/Makefile.am | 4 +- mail/mail-callbacks.c | 4 +- mail/mail-types.h | 2 +- mail/subscribe-control.c | 586 ----------------------------------------------- mail/subscribe-control.h | 45 ---- mail/subscribe-dialog.c | 586 +++++++++++++++++++++++++++++++++++++++++++++++ mail/subscribe-dialog.h | 47 ++++ 8 files changed, 653 insertions(+), 636 deletions(-) delete mode 100644 mail/subscribe-control.c delete mode 100644 mail/subscribe-control.h create mode 100644 mail/subscribe-dialog.c create mode 100644 mail/subscribe-dialog.h diff --git a/mail/ChangeLog b/mail/ChangeLog index e93799d43e..90b3b125ea 100644 --- a/mail/ChangeLog +++ b/mail/ChangeLog @@ -1,3 +1,18 @@ +2000-10-02 Chris Toshok + + * Makefile.am (evolution_mail_SOURCES): subscribe-control.[ch] -> subscribe_dialog.[ch] + + * mail-callbacks.c (manage_subscriptions): subscribe_control -> + subscribe_dialog. Also, pass the shell to subscribe_dialog_new. + + * mail-types.h: SubscribeControl -> SubscribeDialog. + + * subscribe-dialog.c, subscribe-dialog.h: rename from subscribe-control.[ch]. + + * subscribe-dialog.c (subscribe_dialog_construct): pass + Evolution_Shell in. + (subscribe_dialog_new): takes Evolution_Shell argument now. + 2000-10-02 Chris Toshok * message-list.c (message_list_init_renderers): remove the 2 tree diff --git a/mail/Makefile.am b/mail/Makefile.am index 33cb58cf21..c25ae80838 100644 --- a/mail/Makefile.am +++ b/mail/Makefile.am @@ -78,8 +78,8 @@ evolution_mail_SOURCES = \ message-thread.c \ message-thread.h \ session.c \ - subscribe-control.c \ - subscribe-control.h \ + subscribe-dialog.c \ + subscribe-dialog.h \ mail.h evolution_mail_LDADD = \ diff --git a/mail/mail-callbacks.c b/mail/mail-callbacks.c index f4f1ffcb20..6545db3b17 100644 --- a/mail/mail-callbacks.c +++ b/mail/mail-callbacks.c @@ -37,7 +37,7 @@ #include "mail-ops.h" #include "mail-local.h" #include "folder-browser.h" -#include "subscribe-control.h" +#include "subscribe-dialog.h" #include "filter/filter-editor.h" #include "filter/filter-driver.h" #include @@ -827,7 +827,7 @@ void manage_subscriptions (BonoboUIHandler *uih, void *user_data, const char *path) { /* XXX pass in the selected storage */ - GtkWidget *subscribe = subscribe_control_new (); + GtkWidget *subscribe = subscribe_dialog_new ((FOLDER_BROWSER (user_data))->shell); gtk_widget_show (subscribe); } diff --git a/mail/mail-types.h b/mail/mail-types.h index 9f72b23800..74e650fd4f 100644 --- a/mail/mail-types.h +++ b/mail/mail-types.h @@ -29,7 +29,7 @@ extern "C" { typedef struct _FolderBrowser FolderBrowser; -typedef struct _SubscribeControl SubscribeControl; +typedef struct _SubscribeDialog SubscribeDialog; typedef struct _MessageList MessageList; typedef struct _MailDisplay MailDisplay; diff --git a/mail/subscribe-control.c b/mail/subscribe-control.c deleted file mode 100644 index 5204c10d6b..0000000000 --- a/mail/subscribe-control.c +++ /dev/null @@ -1,586 +0,0 @@ -/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ -/* - * subscribe-control.c: Subscribe control top level component - * - * Author: - * Chris Toshok (toshok@helixcode.com) - * - * (C) 2000 Helix Code, Inc. - */ -#include -#include -#include "subscribe-control.h" -#include "e-util/e-html-utils.h" -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "art/empty.xpm" -#include "art/mark.xpm" - -#define DEFAULT_STORAGE_SET_WIDTH 150 -#define DEFAULT_WIDTH 500 -#define DEFAULT_HEIGHT 300 - -#define PARENT_TYPE (gtk_object_get_type ()) - -#define ETABLE_SPEC " \ - \ - 0 \ - 1 \ - 2 \ - \ - \ -" - -enum { - COL_FOLDER_NAME, - COL_FOLDER_SUBSCRIBED, - COL_FOLDER_DESCRIPTION, - COL_LAST -}; - -/* - * Virtual Column list: - * 0 Folder Name - * 1 Subscribed - * 2 Description - */ -char *headers [COL_LAST] = { - "Folder", - "Subscribed", - "Description", -}; - -/* per node structure */ -typedef struct { - gboolean subscribed; -} SubscribeData; - - -static GtkObjectClass *subscribe_control_parent_class; - -static void subscribe_close (BonoboUIHandler *uih, void *user_data, const char *path); -static void subscribe_select_all (BonoboUIHandler *uih, void *user_data, const char *path); -static void subscribe_unselect_all (BonoboUIHandler *uih, void *user_data, const char *path); -static void subscribe_folder (GtkWidget *widget, gpointer user_data); -static void unsubscribe_folder (GtkWidget *widget, gpointer user_data); -static void subscribe_refresh_list (GtkWidget *widget, gpointer user_data); -static void subscribe_search (GtkWidget *widget, gpointer user_data); - -static BonoboUIVerb verbs [] = { - /* File Menu */ - BONOBO_UI_VERB ("FileCloseWin", subscribe_close), - - /* Edit Menu */ - BONOBO_UI_VERB ("EditSelectAll", subscribe_select_all), - BONOBO_UI_VERB ("EditUnSelectAll", subscribe_unselect_all), - - /* Folder Menu / Toolbar */ - BONOBO_UI_VERB ("SubscribeFolder", subscribe_folder), - BONOBO_UI_VERB ("UnsubscribeFolder", unsubscribe_folder), - - /* Toolbar Specific */ - BONOBO_UI_VERB ("RefreshList", subscribe_refresh_list), - - BONOBO_UI_VERB_END -}; - -static void -set_pixmap (Bonobo_UIContainer container, - const char *xml_path, - const char *icon) -{ - char *path; - GdkPixbuf *pixbuf; - - path = g_concat_dir_and_file (EVOLUTION_DATADIR "/images/evolution/buttons", icon); - - pixbuf = gdk_pixbuf_new_from_file (path); - g_return_if_fail (pixbuf != NULL); - - bonobo_ui_util_set_pixbuf (container, xml_path, pixbuf); - - gdk_pixbuf_unref (pixbuf); - - g_free (path); -} - -static void -update_pixmaps (Bonobo_UIContainer container) -{ - set_pixmap (container, "/Toolbar/SubscribeFolder", "fetch-mail.png"); /* XXX */ - set_pixmap (container, "/Toolbar/UnsubscribeFolder", "compose-message.png"); /* XXX */ - set_pixmap (container, "/Toolbar/RefreshList", "forward.png"); /* XXX */ -} - -static GtkWidget* -make_folder_search_widget (GtkSignalFunc start_search_func, - gpointer user_data_for_search) -{ - GtkWidget *search_hbox = gtk_hbox_new (FALSE, 0); - GtkWidget *search_entry = gtk_entry_new (); - - if (start_search_func) { - gtk_signal_connect (GTK_OBJECT (search_entry), "activate", - start_search_func, - user_data_for_search); - } - - /* add the search entry to the our search_vbox */ - gtk_box_pack_start (GTK_BOX (search_hbox), - gtk_label_new(_("Display folders containing:")), - FALSE, TRUE, 0); - gtk_box_pack_start (GTK_BOX (search_hbox), search_entry, - FALSE, TRUE, 3); - - return search_hbox; -} - - - -static void -subscribe_close (BonoboUIHandler *uih, - void *user_data, const char *path) -{ - SubscribeControl *sc = (SubscribeControl*)user_data; - - gtk_widget_destroy (sc->app); -} - -static void -subscribe_select_all (BonoboUIHandler *uih, - void *user_data, const char *path) -{ -} - -static void -subscribe_unselect_all (BonoboUIHandler *uih, - void *user_data, const char *path) -{ -} - -static void -subscribe_folder_foreach (int model_row, gpointer closure) -{ - SubscribeControl *sc = SUBSCRIBE_CONTROL (closure); - ETreePath *node = e_tree_model_node_at_row (sc->model, model_row); - SubscribeData *data = e_tree_model_node_get_data (sc->model, node); - - printf ("subscribe: row %d, node_data %p\n", model_row, - e_tree_model_node_get_data (sc->model, node)); - - data->subscribed = TRUE; - - e_tree_model_node_changed (sc->model, node); -} - -static void -subscribe_folder (GtkWidget *widget, gpointer user_data) -{ - SubscribeControl *sc = SUBSCRIBE_CONTROL (user_data); - - e_table_selected_row_foreach (E_TABLE_SCROLLED(sc->etable)->table, - subscribe_folder_foreach, sc); -} - -static void -unsubscribe_folder_foreach (int model_row, gpointer closure) -{ - SubscribeControl *sc = SUBSCRIBE_CONTROL (closure); - ETreePath *node = e_tree_model_node_at_row (sc->model, model_row); - SubscribeData *data = e_tree_model_node_get_data (sc->model, node); - - printf ("unsubscribe: row %d, node_data %p\n", model_row, - e_tree_model_node_get_data (sc->model, node)); - - data->subscribed = FALSE; - - e_tree_model_node_changed (sc->model, node); -} - - -static void -unsubscribe_folder (GtkWidget *widget, gpointer user_data) -{ - SubscribeControl *sc = SUBSCRIBE_CONTROL (user_data); - - e_table_selected_row_foreach (E_TABLE_SCROLLED(sc->etable)->table, - unsubscribe_folder_foreach, sc); -} - -static void -subscribe_refresh_list (GtkWidget *widget, gpointer user_data) -{ - printf ("subscribe_refresh_list\n"); -} - -static void -subscribe_search (GtkWidget *widget, gpointer user_data) -{ - char* search_pattern = e_utf8_gtk_entry_get_text(GTK_ENTRY(widget)); - - printf ("subscribe_search (%s)\n", search_pattern); - - g_free (search_pattern); -} - - -/* HTML Helpers */ -static void -html_size_req (GtkWidget *widget, GtkRequisition *requisition) -{ - if (GTK_LAYOUT (widget)->height > 90) - requisition->height = 90; - else - requisition->height = GTK_LAYOUT (widget)->height; -} - -/* Returns a GtkHTML which is already inside a GtkScrolledWindow. If - * @white is TRUE, the GtkScrolledWindow will be inside a GtkFrame. - */ -static GtkWidget * -html_new (gboolean white) -{ - GtkWidget *html, *scrolled, *frame; - GtkStyle *style; - - html = gtk_html_new (); - GTK_LAYOUT (html)->height = 0; - gtk_signal_connect (GTK_OBJECT (html), "size_request", - GTK_SIGNAL_FUNC (html_size_req), NULL); - gtk_html_set_editable (GTK_HTML (html), FALSE); - style = gtk_rc_get_style (html); - if (style) { - gtk_html_set_default_background_color (GTK_HTML (html), - white ? &style->white : - &style->bg[0]); - } - gtk_widget_set_sensitive (html, FALSE); - scrolled = gtk_scrolled_window_new (NULL, NULL); - gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled), - GTK_POLICY_NEVER, - GTK_POLICY_NEVER); - gtk_container_add (GTK_CONTAINER (scrolled), html); - - if (white) { - frame = gtk_frame_new (NULL); - gtk_frame_set_shadow_type (GTK_FRAME (frame), - GTK_SHADOW_ETCHED_IN); - gtk_container_add (GTK_CONTAINER (frame), scrolled); - gtk_widget_show_all (frame); - } else - gtk_widget_show_all (scrolled); - - return html; -} - -static void -put_html (GtkHTML *html, char *text) -{ - GtkHTMLStream *handle; - - text = e_text_to_html (text, (E_TEXT_TO_HTML_CONVERT_NL | - E_TEXT_TO_HTML_CONVERT_SPACES | - E_TEXT_TO_HTML_CONVERT_URLS)); - handle = gtk_html_begin (html); - gtk_html_write (html, handle, "", 12); - gtk_html_write (html, handle, text, strlen (text)); - gtk_html_write (html, handle, "", 14); - g_free (text); - gtk_html_end (html, handle, GTK_HTML_STREAM_OK); -} - - - -/* etable stuff for the subscribe ui */ - -static int -etable_col_count (ETableModel *etm, void *data) -{ - return COL_LAST; -} - -static void* -etable_duplicate_value (ETableModel *etm, int col, const void *val, void *data) -{ - return g_strdup (val); -} - -static void -etable_free_value (ETableModel *etm, int col, void *val, void *data) -{ - g_free (val); -} - -static void* -etable_init_value (ETableModel *etm, int col, void *data) -{ - return g_strdup (""); -} - -static gboolean -etable_value_is_empty (ETableModel *etm, int col, const void *val, void *data) -{ - return !(val && *(char *)val); -} - -static char* -etable_value_to_string (ETableModel *etm, int col, const void *val, void *data) -{ - return g_strdup(val); -} - -static GdkPixbuf* -etree_icon_at (ETreeModel *etree, ETreePath *path, void *model_data) -{ - return NULL; /* XXX no icons for now */ -} - -static void* -etree_value_at (ETreeModel *etree, ETreePath *path, int col, void *model_data) -{ - SubscribeData *data = e_tree_model_node_get_data (etree, path); - if (col == COL_FOLDER_NAME) - return "Folder Name"; - else if (col == COL_FOLDER_DESCRIPTION) - return "Folder Description"; - else /* COL_FOLDER_SUBSCRIBED */ - return GINT_TO_POINTER(data->subscribed); -} - -static void -etree_set_value_at (ETreeModel *etree, ETreePath *path, int col, const void *val, void *model_data) -{ - /* nothing */ -} - -static gboolean -etree_is_editable (ETreeModel *etree, ETreePath *path, int col, void *model_data) -{ - return FALSE; -} - - - -#define EXAMPLE_DESCR "And the beast shall come forth surrounded by a roiling cloud of vengeance.\n" \ -" The house of the unbelievers shall be razed and they shall be scorched to the\n" \ -" earth. Their tags shall blink until the end of days. \n" \ -" from The Book of Mozilla, 12:10" - -static void -subscribe_control_gui_init (SubscribeControl *sc) -{ - int i; - ECell *cells[3]; - ETableHeader *e_table_header; - GdkPixbuf *toggles[2]; - BonoboUIComponent *component; - Bonobo_UIContainer container; - GtkWidget *folder_search_widget; - BonoboControl *search_control; - GtkWidget *bonobo_win; - - /* Construct the app */ - bonobo_win = bonobo_win_new ("subscribe-dialog", "Subscribe"); - - sc->storage_set_view = gtk_label_new ("Storage Set View"); - sc->table = gtk_table_new (1, 2, FALSE); - - sc->hpaned = e_hpaned_new (); - e_paned_add1 (E_PANED (sc->hpaned), sc->storage_set_view); - e_paned_add2 (E_PANED (sc->hpaned), sc->table); - e_paned_set_position (E_PANED (sc->hpaned), DEFAULT_STORAGE_SET_WIDTH); - - bonobo_win_set_contents (BONOBO_WIN (bonobo_win), sc->hpaned); - gtk_widget_destroy (sc->app); - sc->app = bonobo_win; - - /* Build the menu and toolbar */ - sc->uih = bonobo_ui_handler_new (); - if (!sc->uih) { - g_message ("subscribe_control_gui_init(): eeeeek, could not create the UI handler!"); - return; - } - - bonobo_ui_handler_set_app (sc->uih, BONOBO_WIN (sc->app)); - - /* set up the bonobo stuff */ - component = bonobo_ui_compat_get_component (sc->uih); - container = bonobo_ui_compat_get_container (sc->uih); - - bonobo_ui_component_add_verb_list_with_data ( - component, verbs, sc); - - bonobo_ui_container_freeze (container, NULL); - - bonobo_ui_util_set_ui (component, container, - EVOLUTION_DATADIR, - "evolution-subscribe.xml", - "evolution-subscribe"); - - update_pixmaps (container); - - folder_search_widget = make_folder_search_widget (subscribe_search, sc); - gtk_widget_show_all (folder_search_widget); - search_control = bonobo_control_new (folder_search_widget); - - bonobo_ui_container_object_set (container, - "/Toolbar/FolderSearch", - bonobo_object_corba_objref (BONOBO_OBJECT (search_control)), - NULL); - - bonobo_ui_container_thaw (container, NULL); - - - /* set our our contents */ - sc->description = html_new (TRUE); - put_html (GTK_HTML (sc->description), EXAMPLE_DESCR); - - gtk_table_attach ( - GTK_TABLE (sc->table), sc->description->parent->parent, - 0, 1, 0, 1, - GTK_FILL | GTK_EXPAND, - 0, - 0, 0); - - sc->model = e_tree_simple_new (etable_col_count, - etable_duplicate_value, - etable_free_value, - etable_init_value, - etable_value_is_empty, - etable_value_to_string, - etree_icon_at, - etree_value_at, - etree_set_value_at, - etree_is_editable, - sc); - - sc->root = e_tree_model_node_insert (sc->model, NULL, - 0, NULL); - - e_tree_model_root_node_set_visible (sc->model, FALSE); - - - for (i = 0; i < 100; i ++) { - SubscribeData *data = g_new (SubscribeData, 1); - data->subscribed = FALSE; - e_tree_model_node_insert (sc->model, sc->root, - 0, data); - } - - e_table_header = e_table_header_new (); - - toggles[0] = gdk_pixbuf_new_from_xpm_data ((const char **)empty_xpm); - toggles[1] = gdk_pixbuf_new_from_xpm_data ((const char **)mark_xpm); - - cells[2] = e_cell_text_new (E_TABLE_MODEL(sc->model), NULL, GTK_JUSTIFY_LEFT); - cells[1] = e_cell_toggle_new (0, 2, toggles); - cells[0] = e_cell_tree_new (E_TABLE_MODEL(sc->model), - NULL, NULL, - /*tree_expanded_pixbuf, tree_unexpanded_pixbuf,*/ - TRUE, cells[2]); - - for (i = 0; i < COL_LAST; i++) { - /* Create the column. */ - ETableCol *ecol; - - if (i == 1) - ecol = e_table_col_new_with_pixbuf (i, toggles[1], - 0, gdk_pixbuf_get_width (toggles[1]), - cells[i], g_str_compare, FALSE); - else - ecol = e_table_col_new (i, headers [i], - 80, 20, - cells[i], - g_str_compare, TRUE); - /* Add it to the header. */ - e_table_header_add_column (e_table_header, ecol, i); - } - - sc->etable = e_table_scrolled_new (e_table_header, E_TABLE_MODEL(sc->model), ETABLE_SPEC); - - gtk_object_set (GTK_OBJECT (E_TABLE_SCROLLED (sc->etable)->table), - "cursor_mode", E_TABLE_CURSOR_LINE, - NULL); - - gtk_table_attach ( - GTK_TABLE (sc->table), sc->etable, - 0, 1, 1, 3, - GTK_FILL | GTK_EXPAND, - GTK_FILL | GTK_EXPAND, - 0, 0); - - gtk_widget_show (sc->description); - gtk_widget_show (sc->etable); - gtk_widget_show (sc->table); - gtk_widget_show (sc->storage_set_view); - gtk_widget_show (sc->hpaned); - - /* FIXME: Session management and stuff? */ - gtk_window_set_default_size ( - GTK_WINDOW (sc->app), - DEFAULT_WIDTH, DEFAULT_HEIGHT); -} - -static void -subscribe_control_destroy (GtkObject *object) -{ - SubscribeControl *subscribe_control; - - subscribe_control = SUBSCRIBE_CONTROL (object); - - subscribe_control_parent_class->destroy (object); -} - -static void -subscribe_control_class_init (GtkObjectClass *object_class) -{ - object_class->destroy = subscribe_control_destroy; - - subscribe_control_parent_class = gtk_type_class (PARENT_TYPE); -} - -static void -subscribe_control_init (GtkObject *object) -{ -} - -static void -subscribe_control_construct (GtkObject *object) -{ - SubscribeControl *sc = SUBSCRIBE_CONTROL (object); - - /* - * Our instance data - */ - - subscribe_control_gui_init (sc); -} - -GtkWidget * -subscribe_control_new () -{ - SubscribeControl *subscribe_control; - - subscribe_control = gtk_type_new (subscribe_control_get_type ()); - - subscribe_control_construct (GTK_OBJECT (subscribe_control)); - - return GTK_WIDGET (subscribe_control->app); -} - -E_MAKE_TYPE (subscribe_control, "SubscribeControl", SubscribeControl, subscribe_control_class_init, subscribe_control_init, PARENT_TYPE); - diff --git a/mail/subscribe-control.h b/mail/subscribe-control.h deleted file mode 100644 index 18a47dd73e..0000000000 --- a/mail/subscribe-control.h +++ /dev/null @@ -1,45 +0,0 @@ -/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ - - -#ifndef _SUBSCRIBE_CONTROL_H_ -#define _SUBSCRIBE_CONTROL_H_ - -#include "mail-types.h" -#include -#include -#include -#include -#include "shell/Evolution.h" - - -#define SUBSCRIBE_CONTROL_TYPE (subscribe_control_get_type ()) -#define SUBSCRIBE_CONTROL(o) (GTK_CHECK_CAST ((o), SUBSCRIBE_CONTROL_TYPE, SubscribeControl)) -#define SUBSCRIBE_CONTROL_CLASS(k) (GTK_CHECK_CLASS_CAST((k), SUBSCRIBE_CONTROL_TYPE, SubscribeControlClass)) -#define IS_SUBSCRIBE_CONTROL(o) (GTK_CHECK_TYPE ((o), SUBSCRIBE_CONTROL_TYPE)) -#define IS_SUBSCRIBE_CONTROL_CLASS(k) (GTK_CHECK_CLASS_TYPE ((k), SUBSCRIBE_CONTROL_TYPE)) - -struct _SubscribeControl { - GtkObject parent; - - BonoboUIHandler *uih; - - GtkWidget *app; - - GtkWidget *storage_set_view; - GtkWidget *hpaned; - GtkWidget *table; - GtkWidget *description; - GtkWidget *etable; - ETreeModel *model; - ETreePath *root; -}; - - -typedef struct { - GtkObjectClass parent_class; -} SubscribeControlClass; - -GtkType subscribe_control_get_type (void); -GtkWidget *subscribe_control_new (void); - -#endif /* _SUBSCRIBE_CONTROL_H_ */ diff --git a/mail/subscribe-dialog.c b/mail/subscribe-dialog.c new file mode 100644 index 0000000000..c95866e110 --- /dev/null +++ b/mail/subscribe-dialog.c @@ -0,0 +1,586 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* + * subscribe-dialog.c: Subscribe dialog + * + * Author: + * Chris Toshok (toshok@helixcode.com) + * + * (C) 2000 Helix Code, Inc. + */ +#include +#include +#include "subscribe-dialog.h" +#include "e-util/e-html-utils.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "art/empty.xpm" +#include "art/mark.xpm" + +#define DEFAULT_STORAGE_SET_WIDTH 150 +#define DEFAULT_WIDTH 500 +#define DEFAULT_HEIGHT 300 + +#define PARENT_TYPE (gtk_object_get_type ()) + +#define ETABLE_SPEC " \ + \ + 0 \ + 1 \ + 2 \ + \ + \ +" + +enum { + COL_FOLDER_NAME, + COL_FOLDER_SUBSCRIBED, + COL_FOLDER_DESCRIPTION, + COL_LAST +}; + +/* + * Virtual Column list: + * 0 Folder Name + * 1 Subscribed + * 2 Description + */ +char *headers [COL_LAST] = { + "Folder", + "Subscribed", + "Description", +}; + +/* per node structure */ +typedef struct { + gboolean subscribed; +} SubscribeData; + + +static GtkObjectClass *subscribe_dialog_parent_class; + +static void subscribe_close (BonoboUIHandler *uih, void *user_data, const char *path); +static void subscribe_select_all (BonoboUIHandler *uih, void *user_data, const char *path); +static void subscribe_unselect_all (BonoboUIHandler *uih, void *user_data, const char *path); +static void subscribe_folder (GtkWidget *widget, gpointer user_data); +static void unsubscribe_folder (GtkWidget *widget, gpointer user_data); +static void subscribe_refresh_list (GtkWidget *widget, gpointer user_data); +static void subscribe_search (GtkWidget *widget, gpointer user_data); + +static BonoboUIVerb verbs [] = { + /* File Menu */ + BONOBO_UI_VERB ("FileCloseWin", subscribe_close), + + /* Edit Menu */ + BONOBO_UI_VERB ("EditSelectAll", subscribe_select_all), + BONOBO_UI_VERB ("EditUnSelectAll", subscribe_unselect_all), + + /* Folder Menu / Toolbar */ + BONOBO_UI_VERB ("SubscribeFolder", subscribe_folder), + BONOBO_UI_VERB ("UnsubscribeFolder", unsubscribe_folder), + + /* Toolbar Specific */ + BONOBO_UI_VERB ("RefreshList", subscribe_refresh_list), + + BONOBO_UI_VERB_END +}; + +static void +set_pixmap (Bonobo_UIContainer container, + const char *xml_path, + const char *icon) +{ + char *path; + GdkPixbuf *pixbuf; + + path = g_concat_dir_and_file (EVOLUTION_DATADIR "/images/evolution/buttons", icon); + + pixbuf = gdk_pixbuf_new_from_file (path); + g_return_if_fail (pixbuf != NULL); + + bonobo_ui_util_set_pixbuf (container, xml_path, pixbuf); + + gdk_pixbuf_unref (pixbuf); + + g_free (path); +} + +static void +update_pixmaps (Bonobo_UIContainer container) +{ + set_pixmap (container, "/Toolbar/SubscribeFolder", "fetch-mail.png"); /* XXX */ + set_pixmap (container, "/Toolbar/UnsubscribeFolder", "compose-message.png"); /* XXX */ + set_pixmap (container, "/Toolbar/RefreshList", "forward.png"); /* XXX */ +} + +static GtkWidget* +make_folder_search_widget (GtkSignalFunc start_search_func, + gpointer user_data_for_search) +{ + GtkWidget *search_hbox = gtk_hbox_new (FALSE, 0); + GtkWidget *search_entry = gtk_entry_new (); + + if (start_search_func) { + gtk_signal_connect (GTK_OBJECT (search_entry), "activate", + start_search_func, + user_data_for_search); + } + + /* add the search entry to the our search_vbox */ + gtk_box_pack_start (GTK_BOX (search_hbox), + gtk_label_new(_("Display folders containing:")), + FALSE, TRUE, 0); + gtk_box_pack_start (GTK_BOX (search_hbox), search_entry, + FALSE, TRUE, 3); + + return search_hbox; +} + + + +static void +subscribe_close (BonoboUIHandler *uih, + void *user_data, const char *path) +{ + SubscribeDialog *sc = (SubscribeDialog*)user_data; + + gtk_widget_destroy (sc->app); +} + +static void +subscribe_select_all (BonoboUIHandler *uih, + void *user_data, const char *path) +{ +} + +static void +subscribe_unselect_all (BonoboUIHandler *uih, + void *user_data, const char *path) +{ +} + +static void +subscribe_folder_foreach (int model_row, gpointer closure) +{ + SubscribeDialog *sc = SUBSCRIBE_DIALOG (closure); + ETreePath *node = e_tree_model_node_at_row (sc->model, model_row); + SubscribeData *data = e_tree_model_node_get_data (sc->model, node); + + printf ("subscribe: row %d, node_data %p\n", model_row, + e_tree_model_node_get_data (sc->model, node)); + + data->subscribed = TRUE; + + e_tree_model_node_changed (sc->model, node); +} + +static void +subscribe_folder (GtkWidget *widget, gpointer user_data) +{ + SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data); + + e_table_selected_row_foreach (E_TABLE_SCROLLED(sc->etable)->table, + subscribe_folder_foreach, sc); +} + +static void +unsubscribe_folder_foreach (int model_row, gpointer closure) +{ + SubscribeDialog *sc = SUBSCRIBE_DIALOG (closure); + ETreePath *node = e_tree_model_node_at_row (sc->model, model_row); + SubscribeData *data = e_tree_model_node_get_data (sc->model, node); + + printf ("unsubscribe: row %d, node_data %p\n", model_row, + e_tree_model_node_get_data (sc->model, node)); + + data->subscribed = FALSE; + + e_tree_model_node_changed (sc->model, node); +} + + +static void +unsubscribe_folder (GtkWidget *widget, gpointer user_data) +{ + SubscribeDialog *sc = SUBSCRIBE_DIALOG (user_data); + + e_table_selected_row_foreach (E_TABLE_SCROLLED(sc->etable)->table, + unsubscribe_folder_foreach, sc); +} + +static void +subscribe_refresh_list (GtkWidget *widget, gpointer user_data) +{ + printf ("subscribe_refresh_list\n"); +} + +static void +subscribe_search (GtkWidget *widget, gpointer user_data) +{ + char* search_pattern = e_utf8_gtk_entry_get_text(GTK_ENTRY(widget)); + + printf ("subscribe_search (%s)\n", search_pattern); + + g_free (search_pattern); +} + + +/* HTML Helpers */ +static void +html_size_req (GtkWidget *widget, GtkRequisition *requisition) +{ + if (GTK_LAYOUT (widget)->height > 90) + requisition->height = 90; + else + requisition->height = GTK_LAYOUT (widget)->height; +} + +/* Returns a GtkHTML which is already inside a GtkScrolledWindow. If + * @white is TRUE, the GtkScrolledWindow will be inside a GtkFrame. + */ +static GtkWidget * +html_new (gboolean white) +{ + GtkWidget *html, *scrolled, *frame; + GtkStyle *style; + + html = gtk_html_new (); + GTK_LAYOUT (html)->height = 0; + gtk_signal_connect (GTK_OBJECT (html), "size_request", + GTK_SIGNAL_FUNC (html_size_req), NULL); + gtk_html_set_editable (GTK_HTML (html), FALSE); + style = gtk_rc_get_style (html); + if (style) { + gtk_html_set_default_background_color (GTK_HTML (html), + white ? &style->white : + &style->bg[0]); + } + gtk_widget_set_sensitive (html, FALSE); + scrolled = gtk_scrolled_window_new (NULL, NULL); + gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled), + GTK_POLICY_NEVER, + GTK_POLICY_NEVER); + gtk_container_add (GTK_CONTAINER (scrolled), html); + + if (white) { + frame = gtk_frame_new (NULL); + gtk_frame_set_shadow_type (GTK_FRAME (frame), + GTK_SHADOW_ETCHED_IN); + gtk_container_add (GTK_CONTAINER (frame), scrolled); + gtk_widget_show_all (frame); + } else + gtk_widget_show_all (scrolled); + + return html; +} + +static void +put_html (GtkHTML *html, char *text) +{ + GtkHTMLStream *handle; + + text = e_text_to_html (text, (E_TEXT_TO_HTML_CONVERT_NL | + E_TEXT_TO_HTML_CONVERT_SPACES | + E_TEXT_TO_HTML_CONVERT_URLS)); + handle = gtk_html_begin (html); + gtk_html_write (html, handle, "", 12); + gtk_html_write (html, handle, text, strlen (text)); + gtk_html_write (html, handle, "", 14); + g_free (text); + gtk_html_end (html, handle, GTK_HTML_STREAM_OK); +} + + + +/* etable stuff for the subscribe ui */ + +static int +etable_col_count (ETableModel *etm, void *data) +{ + return COL_LAST; +} + +static void* +etable_duplicate_value (ETableModel *etm, int col, const void *val, void *data) +{ + return g_strdup (val); +} + +static void +etable_free_value (ETableModel *etm, int col, void *val, void *data) +{ + g_free (val); +} + +static void* +etable_init_value (ETableModel *etm, int col, void *data) +{ + return g_strdup (""); +} + +static gboolean +etable_value_is_empty (ETableModel *etm, int col, const void *val, void *data) +{ + return !(val && *(char *)val); +} + +static char* +etable_value_to_string (ETableModel *etm, int col, const void *val, void *data) +{ + return g_strdup(val); +} + +static GdkPixbuf* +etree_icon_at (ETreeModel *etree, ETreePath *path, void *model_data) +{ + return NULL; /* XXX no icons for now */ +} + +static void* +etree_value_at (ETreeModel *etree, ETreePath *path, int col, void *model_data) +{ + SubscribeData *data = e_tree_model_node_get_data (etree, path); + if (col == COL_FOLDER_NAME) + return "Folder Name"; + else if (col == COL_FOLDER_DESCRIPTION) + return "Folder Description"; + else /* COL_FOLDER_SUBSCRIBED */ + return GINT_TO_POINTER(data->subscribed); +} + +static void +etree_set_value_at (ETreeModel *etree, ETreePath *path, int col, const void *val, void *model_data) +{ + /* nothing */ +} + +static gboolean +etree_is_editable (ETreeModel *etree, ETreePath *path, int col, void *model_data) +{ + return FALSE; +} + + + +#define EXAMPLE_DESCR "And the beast shall come forth surrounded by a roiling cloud of vengeance.\n" \ +" The house of the unbelievers shall be razed and they shall be scorched to the\n" \ +" earth. Their tags shall blink until the end of days. \n" \ +" from The Book of Mozilla, 12:10" + +static void +subscribe_dialog_gui_init (SubscribeDialog *sc) +{ + int i; + ECell *cells[3]; + ETableHeader *e_table_header; + GdkPixbuf *toggles[2]; + BonoboUIComponent *component; + Bonobo_UIContainer container; + GtkWidget *folder_search_widget; + BonoboControl *search_control; + GtkWidget *bonobo_win; + + /* Construct the app */ + bonobo_win = bonobo_win_new ("subscribe-dialog", "Subscribe"); + + sc->storage_set_view = gtk_label_new ("Storage Set View"); + sc->table = gtk_table_new (1, 2, FALSE); + + sc->hpaned = e_hpaned_new (); + e_paned_add1 (E_PANED (sc->hpaned), sc->storage_set_view); + e_paned_add2 (E_PANED (sc->hpaned), sc->table); + e_paned_set_position (E_PANED (sc->hpaned), DEFAULT_STORAGE_SET_WIDTH); + + bonobo_win_set_contents (BONOBO_WIN (bonobo_win), sc->hpaned); + gtk_widget_destroy (sc->app); + sc->app = bonobo_win; + + /* Build the menu and toolbar */ + sc->uih = bonobo_ui_handler_new (); + if (!sc->uih) { + g_message ("subscribe_dialog_gui_init(): eeeeek, could not create the UI handler!"); + return; + } + + bonobo_ui_handler_set_app (sc->uih, BONOBO_WIN (sc->app)); + + /* set up the bonobo stuff */ + component = bonobo_ui_compat_get_component (sc->uih); + container = bonobo_ui_compat_get_container (sc->uih); + + bonobo_ui_component_add_verb_list_with_data ( + component, verbs, sc); + + bonobo_ui_container_freeze (container, NULL); + + bonobo_ui_util_set_ui (component, container, + EVOLUTION_DATADIR, + "evolution-subscribe.xml", + "evolution-subscribe"); + + update_pixmaps (container); + + folder_search_widget = make_folder_search_widget (subscribe_search, sc); + gtk_widget_show_all (folder_search_widget); + search_control = bonobo_control_new (folder_search_widget); + + bonobo_ui_container_object_set (container, + "/Toolbar/FolderSearch", + bonobo_object_corba_objref (BONOBO_OBJECT (search_control)), + NULL); + + bonobo_ui_container_thaw (container, NULL); + + + /* set our our contents */ + sc->description = html_new (TRUE); + put_html (GTK_HTML (sc->description), EXAMPLE_DESCR); + + gtk_table_attach ( + GTK_TABLE (sc->table), sc->description->parent->parent, + 0, 1, 0, 1, + GTK_FILL | GTK_EXPAND, + 0, + 0, 0); + + sc->model = e_tree_simple_new (etable_col_count, + etable_duplicate_value, + etable_free_value, + etable_init_value, + etable_value_is_empty, + etable_value_to_string, + etree_icon_at, + etree_value_at, + etree_set_value_at, + etree_is_editable, + sc); + + sc->root = e_tree_model_node_insert (sc->model, NULL, + 0, NULL); + + e_tree_model_root_node_set_visible (sc->model, FALSE); + + + for (i = 0; i < 100; i ++) { + SubscribeData *data = g_new (SubscribeData, 1); + data->subscribed = FALSE; + e_tree_model_node_insert (sc->model, sc->root, + 0, data); + } + + e_table_header = e_table_header_new (); + + toggles[0] = gdk_pixbuf_new_from_xpm_data ((const char **)empty_xpm); + toggles[1] = gdk_pixbuf_new_from_xpm_data ((const char **)mark_xpm); + + cells[2] = e_cell_text_new (E_TABLE_MODEL(sc->model), NULL, GTK_JUSTIFY_LEFT); + cells[1] = e_cell_toggle_new (0, 2, toggles); + cells[0] = e_cell_tree_new (E_TABLE_MODEL(sc->model), + NULL, NULL, + TRUE, cells[2]); + + for (i = 0; i < COL_LAST; i++) { + /* Create the column. */ + ETableCol *ecol; + + if (i == 1) + ecol = e_table_col_new_with_pixbuf (i, toggles[1], + 0, gdk_pixbuf_get_width (toggles[1]), + cells[i], g_str_compare, FALSE); + else + ecol = e_table_col_new (i, headers [i], + 80, 20, + cells[i], + g_str_compare, TRUE); + /* Add it to the header. */ + e_table_header_add_column (e_table_header, ecol, i); + } + + sc->etable = e_table_scrolled_new (e_table_header, E_TABLE_MODEL(sc->model), ETABLE_SPEC); + + gtk_object_set (GTK_OBJECT (E_TABLE_SCROLLED (sc->etable)->table), + "cursor_mode", E_TABLE_CURSOR_LINE, + NULL); + + gtk_table_attach ( + GTK_TABLE (sc->table), sc->etable, + 0, 1, 1, 3, + GTK_FILL | GTK_EXPAND, + GTK_FILL | GTK_EXPAND, + 0, 0); + + gtk_widget_show (sc->description); + gtk_widget_show (sc->etable); + gtk_widget_show (sc->table); + gtk_widget_show (sc->storage_set_view); + gtk_widget_show (sc->hpaned); + + /* FIXME: Session management and stuff? */ + gtk_window_set_default_size ( + GTK_WINDOW (sc->app), + DEFAULT_WIDTH, DEFAULT_HEIGHT); +} + +static void +subscribe_dialog_destroy (GtkObject *object) +{ + SubscribeDialog *subscribe_dialog; + + subscribe_dialog = SUBSCRIBE_DIALOG (object); + + subscribe_dialog_parent_class->destroy (object); +} + +static void +subscribe_dialog_class_init (GtkObjectClass *object_class) +{ + object_class->destroy = subscribe_dialog_destroy; + + subscribe_dialog_parent_class = gtk_type_class (PARENT_TYPE); +} + +static void +subscribe_dialog_init (GtkObject *object) +{ +} + +static void +subscribe_dialog_construct (GtkObject *object, Evolution_Shell shell) +{ + SubscribeDialog *sc = SUBSCRIBE_DIALOG (object); + + /* + * Our instance data + */ + sc->shell = shell; + + subscribe_dialog_gui_init (sc); +} + +GtkWidget * +subscribe_dialog_new (Evolution_Shell shell) +{ + SubscribeDialog *subscribe_dialog; + + subscribe_dialog = gtk_type_new (subscribe_dialog_get_type ()); + + subscribe_dialog_construct (GTK_OBJECT (subscribe_dialog), shell); + + return GTK_WIDGET (subscribe_dialog->app); +} + +E_MAKE_TYPE (subscribe_dialog, "SubscribeDialog", SubscribeDialog, subscribe_dialog_class_init, subscribe_dialog_init, PARENT_TYPE); + diff --git a/mail/subscribe-dialog.h b/mail/subscribe-dialog.h new file mode 100644 index 0000000000..5805c50b5f --- /dev/null +++ b/mail/subscribe-dialog.h @@ -0,0 +1,47 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ + + +#ifndef _SUBSCRIBE_DIALOG_H_ +#define _SUBSCRIBE_DIALOG_H_ + +#include "mail-types.h" +#include +#include +#include +#include +#include "shell/Evolution.h" + + +#define SUBSCRIBE_DIALOG_TYPE (subscribe_dialog_get_type ()) +#define SUBSCRIBE_DIALOG(o) (GTK_CHECK_CAST ((o), SUBSCRIBE_DIALOG_TYPE, SubscribeDialog)) +#define SUBSCRIBE_DIALOG_CLASS(k) (GTK_CHECK_CLASS_CAST((k), SUBSCRIBE_DIALOG_TYPE, SubscribeDialogClass)) +#define IS_SUBSCRIBE_DIALOG(o) (GTK_CHECK_TYPE ((o), SUBSCRIBE_DIALOG_TYPE)) +#define IS_SUBSCRIBE_DIALOG_CLASS(k) (GTK_CHECK_CLASS_TYPE ((k), SUBSCRIBE_DIALOG_TYPE)) + +struct _SubscribeDialog { + GtkObject parent; + + Evolution_Shell shell; + + BonoboUIHandler *uih; + + GtkWidget *app; + + GtkWidget *storage_set_view; + GtkWidget *hpaned; + GtkWidget *table; + GtkWidget *description; + GtkWidget *etable; + ETreeModel *model; + ETreePath *root; +}; + + +typedef struct { + GtkObjectClass parent_class; +} SubscribeDialogClass; + +GtkType subscribe_dialog_get_type (void); +GtkWidget *subscribe_dialog_new (Evolution_Shell shell); + +#endif /* _SUBSCRIBE_DIALOG_H_ */ -- cgit