/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* e-shell-view.c * * Copyright (C) 2000, 2001 Ximian, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. * * Authors: * Ettore Perazzoli * Miguel de Icaza * Matt Loper */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "widgets/misc/e-clipped-label.h" #include "widgets/misc/e-bonobo-widget.h" #include "evolution-shell-view.h" #include "e-gray-bar.h" #include "e-shell-constants.h" #include "e-shell-folder-title-bar.h" #include "e-shell-utils.h" #include "e-shell.h" #include "e-shortcuts-view.h" #include "e-storage-set-view.h" #include "e-title-bar.h" #include "e-shell-view.h" #include "e-shell-view-menu.h" static BonoboWindowClass *parent_class = NULL; struct _View { char *uri; GtkWidget *control; EFolder *folder; }; typedef struct _View View; struct _EShellViewPrivate { /* The shell. */ EShell *shell; /* EvolutionShellView Bonobo object for implementing the Evolution::ShellView interface. */ EvolutionShellView *corba_interface; /* The UI handler & container. */ BonoboUIComponent *ui_component; BonoboUIContainer *ui_container; /* Currently displayed URI. */ char *uri; /* delayed selection, used when a path doesn't exist in an EStorage. cleared when we're signaled with "folder_selected" */ char *delayed_selection; /* uri to go to at timeout */ unsigned int set_folder_timeout; char *set_folder_uri; /* Tooltips. */ GtkTooltips *tooltips; /* The widgetry. */ GtkWidget *appbar; GtkWidget *hpaned; GtkWidget *view_vbox; GtkWidget *folder_title_bar; GtkWidget *view_hpaned; GtkWidget *contents; GtkWidget *notebook; GtkWidget *shortcut_frame; GtkWidget *shortcut_bar; GtkWidget *storage_set_title_bar; GtkWidget *storage_set_view; GtkWidget *storage_set_view_box; /* The status bar widgetry. */ GtkWidget *status_bar; GtkWidget *offline_toggle; GtkWidget *offline_toggle_pixmap; GtkWidget *menu_hint_label; GtkWidget *task_bar; /* The pop-up window for the folder-tree (i.e. the one we create when the user clicks on the folder title. */ GtkWidget *folder_bar_popup; /* The views we have already open. */ GHashTable *uri_to_view; /* Position of the handles in the paneds, to be restored when we show elements after hiding them. */ unsigned int hpaned_position; unsigned int view_hpaned_position; /* Whether the shortcut and folder bars are visible or not. */ unsigned int shortcut_bar_shown : 1; unsigned int folder_bar_shown : 1; /* List of sockets we created. */ GList *sockets; }; enum { SHORTCUT_BAR_VISIBILITY_CHANGED, FOLDER_BAR_VISIBILITY_CHANGED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; #define DEFAULT_SHORTCUT_BAR_WIDTH 100 #define DEFAULT_TREE_WIDTH 130 #define MIN_POPUP_TREE_WIDTH 130 #define DEFAULT_WIDTH 705 #define DEFAULT_HEIGHT 550 #define SET_FOLDER_DELAY 250 /* The icons for the offline/online status. */ static GdkPixmap *offline_pixmap = NULL; static GdkBitmap *offline_mask = NULL; static GdkPixmap *online_pixmap = NULL; static GdkBitmap *online_mask = NULL; static void update_for_current_uri (EShellView *shell_view); static void update_offline_toggle_status (EShellView *shell_view); static const char *get_storage_set_path_from_uri (const char *uri); /* Boo. */ static void new_folder_cb (EStorageSet *storage_set, const char *path, void *data); /* View handling. */ static View * view_new (const char *uri, GtkWidget *control) { View *new; new = g_new (View, 1); new->uri = g_strdup (uri); new->control = control; return new; } static void view_destroy (View *view) { g_free (view->uri); g_free (view); } /* Utility functions. */ static GtkWidget * create_label_for_empty_page (void) { GtkWidget *label; label = e_clipped_label_new (_("(No folder displayed)")); gtk_widget_show (label); return label; } /* Initialize the icons. */ static void load_images (void) { GdkPixbuf *pixbuf; pixbuf = gdk_pixbuf_new_from_file (EVOLUTION_IMAGES "/offline.png"); if (pixbuf == NULL) { g_warning ("Cannot load `%s'", EVOLUTION_IMAGES "/offline.png"); } else { gdk_pixbuf_render_pixmap_and_mask (pixbuf, &offline_pixmap, &offline_mask, 128); gdk_pixbuf_unref (pixbuf); } pixbuf = gdk_pixbuf_new_from_file (EVOLUTION_IMAGES "/online.png"); if (pixbuf == NULL) { g_warning ("Cannot load `%s'", EVOLUTION_IMAGES "/online.png"); } else { gdk_pixbuf_render_pixmap_and_mask (pixbuf, &online_pixmap, &online_mask, 128); gdk_pixbuf_unref (pixbuf); } } /* FIXME this is broken. */ static gboolean bonobo_widget_is_dead (BonoboWidget *bonobo_widget) { BonoboControlFrame *control_frame; CORBA_Object corba_object; CORBA_Environment ev; gboolean is_dead; control_frame = bonobo_widget_get_control_frame (bonobo_widget); corba_object = bonobo_control_frame_get_control (control_frame); CORBA_exception_init (&ev); is_dead = CORBA_Object_non_existent (corba_object, &ev); CORBA_exception_free (&ev); return is_dead; } static void cleanup_delayed_selection (EShellView *shell_view) { EShellViewPrivate *priv; priv = shell_view->priv; if (priv->delayed_selection != NULL) { g_free (priv->delayed_selection); priv->delayed_selection = NULL; gtk_signal_disconnect_by_func (GTK_OBJECT (e_shell_get_storage_set (priv->shell)), GTK_SIGNAL_FUNC (new_folder_cb), shell_view); } } static GtkWidget * find_socket (GtkContainer *container) { GList *children, *tmp; children = gtk_container_children (container); while (children) { if (BONOBO_IS_SOCKET (children->data)) return children->data; else if (GTK_IS_CONTAINER (children->data)) { GtkWidget *socket = find_socket (children->data); if (socket) return socket; } tmp = children->next; g_list_free_1 (children); children = tmp; } return NULL; } static void setup_verb_sensitivity_for_folder (EShellView *shell_view, const char *path) { EShellViewPrivate *priv; BonoboUIComponent *ui_component; const char *prop; priv = shell_view->priv; /* Adjust sensitivity for menu options depending on whether the folder selected is a stock folder. */ if (path == NULL) { prop = "0"; } else { EFolder *folder; folder = e_storage_set_get_folder (e_shell_get_storage_set (priv->shell), path); if (folder != NULL && ! e_folder_get_is_stock (folder)) prop = "1"; else prop = "0"; } ui_component = e_shell_view_get_bonobo_ui_component (shell_view); bonobo_ui_component_set_prop (ui_component, "/commands/MoveFolder", "sensitive", prop, NULL); bonobo_ui_component_set_prop (ui_component, "/commands/CopyFolder", "sensitive", prop, NULL); bonobo_ui_component_set_prop (ui_component, "/commands/DeleteFolder", "sensitive", prop, NULL); bonobo_ui_component_set_prop (ui_component, "/commands/RenameFolder", "sensitive", prop, NULL); } /* Callbacks for the EStorageSet. */ static void storage_set_removed_folder_callback (EStorageSet *storage_set, const char *path, void *data) { EShellView *shell_view; EShellViewPrivate *priv; GtkWidget *socket; View *view; int destroy_connection_id; int page_num; char *uri; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; uri = g_strconcat (E_SHELL_URI_PREFIX, path, NULL); view = g_hash_table_lookup (priv->uri_to_view, uri); g_free (uri); if (view == NULL) return; socket = find_socket (GTK_CONTAINER (view->control)); priv->sockets = g_list_remove (priv->sockets, socket); destroy_connection_id = GPOINTER_TO_INT (gtk_object_get_data (GTK_OBJECT (socket), "e_shell_view_destroy_connection_id")); gtk_signal_disconnect (GTK_OBJECT (socket), destroy_connection_id); page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), view->control); if (strncmp (priv->uri, E_SHELL_URI_PREFIX, E_SHELL_URI_PREFIX_LEN) == 0 && strcmp (priv->uri + E_SHELL_URI_PREFIX_LEN, path) == 0) { e_shell_view_display_uri (shell_view, "evolution:/local/Inbox"); } bonobo_control_frame_control_deactivate (BONOBO_CONTROL_FRAME (bonobo_widget_get_control_frame (BONOBO_WIDGET (view->control)))); gtk_widget_destroy (view->control); g_hash_table_remove (priv->uri_to_view, view->uri); view_destroy (view); gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), page_num); } /* Folder bar pop-up handling. */ static void reparent (GtkWidget *widget, GtkContainer *new_container) { gtk_widget_ref (widget); gtk_container_remove (GTK_CONTAINER (widget->parent), widget); gtk_container_add (GTK_CONTAINER (new_container), widget); gtk_widget_unref (widget); } static void reparent_storage_set_view_box_and_destroy_popup (EShellView *shell_view) { EShellViewPrivate *priv; priv = shell_view->priv; if (priv->folder_bar_popup == NULL) return; gtk_widget_ref (priv->storage_set_view_box); gtk_container_remove (GTK_CONTAINER (priv->folder_bar_popup), priv->storage_set_view_box); e_paned_pack1 (E_PANED (priv->view_hpaned), priv->storage_set_view_box, FALSE, FALSE); gtk_widget_unref (priv->storage_set_view_box); gtk_widget_destroy (priv->folder_bar_popup); priv->folder_bar_popup = NULL; /* Re-enable DnD on the StorageSetView (it got disabled when displaying the pop-up). */ e_storage_set_view_set_allow_dnd (E_STORAGE_SET_VIEW (priv->storage_set_view), TRUE); } static void popdown_transient_folder_bar (EShellView *shell_view) { EShellViewPrivate *priv; priv = shell_view->priv; gdk_pointer_ungrab (GDK_CURRENT_TIME); gtk_grab_remove (priv->storage_set_view_box); reparent_storage_set_view_box_and_destroy_popup (shell_view); gtk_widget_hide (priv->storage_set_view_box); e_shell_folder_title_bar_set_toggle_state (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), FALSE); /* Re-enable DnD on the StorageSetView (it got disabled when displaying the pop-up). */ e_storage_set_view_set_allow_dnd (E_STORAGE_SET_VIEW (priv->storage_set_view), TRUE); } static int storage_set_view_box_button_release_event_cb (GtkWidget *widget, GdkEventButton *button_event, void *data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; if (button_event->window == E_PANED (priv->view_hpaned)->handle || button_event->button != 1) return FALSE; popdown_transient_folder_bar (shell_view); return TRUE; } static void popup_storage_set_view_button_clicked (ETitleBar *title_bar, void *data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; g_assert (priv->folder_bar_popup != NULL); reparent_storage_set_view_box_and_destroy_popup (shell_view); e_shell_view_show_folder_bar (shell_view, TRUE); e_shell_folder_title_bar_set_toggle_state (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), FALSE); } static void folder_bar_popup_map_callback (GtkWidget *widget, void *data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; if (gdk_pointer_grab (widget->window, TRUE, (GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | GDK_ENTER_NOTIFY_MASK | GDK_LEAVE_NOTIFY_MASK | GDK_POINTER_MOTION_MASK), NULL, NULL, GDK_CURRENT_TIME) != 0) { g_warning ("e-shell-view.c:folder_bar_popup_map_callback() -- pointer grab failed."); return; } gtk_grab_add (widget); gtk_signal_connect_while_alive (GTK_OBJECT (widget), "button_release_event", GTK_SIGNAL_FUNC (storage_set_view_box_button_release_event_cb), shell_view, GTK_OBJECT (priv->folder_bar_popup)); gtk_signal_connect_while_alive (GTK_OBJECT (priv->storage_set_view), "button_release_event", GTK_SIGNAL_FUNC (storage_set_view_box_button_release_event_cb), shell_view, GTK_OBJECT (priv->folder_bar_popup)); gtk_signal_connect_while_alive (GTK_OBJECT (priv->storage_set_title_bar), "button_clicked", GTK_SIGNAL_FUNC (popup_storage_set_view_button_clicked), shell_view, GTK_OBJECT (priv->folder_bar_popup)); } static void pop_up_folder_bar (EShellView *shell_view) { EShellViewPrivate *priv; int x, y; int orig_x, orig_y; priv = shell_view->priv; g_assert (! priv->folder_bar_shown); e_title_bar_set_button_mode (E_TITLE_BAR (priv->storage_set_title_bar), E_TITLE_BAR_BUTTON_MODE_PIN); priv->folder_bar_popup = gtk_window_new (GTK_WINDOW_POPUP); /* We need to show the storage set view box and do a pointer grab to catch the mouse clicks. But until the box is shown, we cannot grab. So we connect to the "map" signal; `storage_set_view_box_map_cb()' will do the grab. */ gtk_signal_connect (GTK_OBJECT (priv->folder_bar_popup), "map", GTK_SIGNAL_FUNC (folder_bar_popup_map_callback), shell_view); x = priv->folder_title_bar->allocation.x; y = priv->folder_title_bar->allocation.y + priv->folder_title_bar->allocation.height; gdk_window_get_origin (priv->folder_title_bar->window, &orig_x, &orig_y); x += orig_x; y += orig_y + 2; priv->view_hpaned_position = MAX (priv->view_hpaned_position, MIN_POPUP_TREE_WIDTH); gtk_window_set_default_size (GTK_WINDOW (priv->folder_bar_popup), priv->view_hpaned_position, priv->view_hpaned->allocation.height); reparent (priv->storage_set_view_box, GTK_CONTAINER (priv->folder_bar_popup)); gtk_widget_show (priv->storage_set_view_box); gtk_widget_popup (priv->folder_bar_popup, x, y); /* Disable DnD or "interesting" things will happen. */ e_storage_set_view_set_allow_dnd (E_STORAGE_SET_VIEW (priv->storage_set_view), FALSE); } /* Switching views on a tree view click. */ static int set_folder_timeout (gpointer data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; /* set to 0 so we don't remove it in _display_uri() */ priv->set_folder_timeout = 0; e_shell_view_display_uri (shell_view, priv->set_folder_uri); return FALSE; } static int popdown_transient_folder_bar_idle (void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); popdown_transient_folder_bar (shell_view); gtk_object_unref (GTK_OBJECT (shell_view)); return FALSE; } static void switch_on_folder_tree_click (EShellView *shell_view, const char *path) { EShellViewPrivate *priv; char *uri; priv = shell_view->priv; gdk_pointer_ungrab (GDK_CURRENT_TIME); gtk_grab_remove (priv->storage_set_view_box); uri = g_strconcat (E_SHELL_URI_PREFIX, path, NULL); if (priv->uri != NULL && !strcmp (uri, priv->uri)) { g_free (uri); return; } if (priv->set_folder_timeout != 0) gtk_timeout_remove (priv->set_folder_timeout); g_free (priv->set_folder_uri); priv->set_folder_uri = NULL; cleanup_delayed_selection (shell_view); if (priv->folder_bar_popup != NULL) { e_shell_view_display_uri (shell_view, uri); g_free (uri); gtk_object_ref (GTK_OBJECT (shell_view)); gtk_idle_add (popdown_transient_folder_bar_idle, shell_view); return; } priv->set_folder_uri = uri; priv->set_folder_timeout = gtk_timeout_add (SET_FOLDER_DELAY, set_folder_timeout, shell_view); } /* Callbacks. */ /* Callback when a new folder is added. removed when we clear the delayed_selection */ static void new_folder_cb (EStorageSet *storage_set, const char *path, void *data) { EShellView *shell_view; EShellViewPrivate *priv; char *delayed_path; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; delayed_path = strchr (priv->delayed_selection, ':'); if (delayed_path) { delayed_path ++; if (!strcmp(path, delayed_path)) { char *uri; uri = g_strdup (priv->delayed_selection); cleanup_delayed_selection (shell_view); e_shell_view_display_uri (shell_view, uri); g_free (uri); } } } /* Callback called when an icon on the shortcut bar gets clicked. */ static void activate_shortcut_cb (EShortcutsView *shortcut_view, EShortcuts *shortcuts, const char *uri, gboolean in_new_window, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); if (in_new_window) { EShellView *new_view; new_view = e_shell_create_view (e_shell_view_get_shell (shell_view), uri, shell_view); } else { e_shell_view_display_uri (shell_view, uri); } } /* Callback when user chooses "Hide shortcut bar" via a right click */ static void hide_requested_cb (EShortcutsView *shortcut_view, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); e_shell_view_show_shortcut_bar (shell_view, FALSE); } /* Callback called when a folder on the tree view gets clicked. */ static void folder_selected_cb (EStorageSetView *storage_set_view, const char *path, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); setup_verb_sensitivity_for_folder (shell_view, path); switch_on_folder_tree_click (shell_view, path); } /* Callback called when a storage in the tree view is clicked. */ static void storage_selected_cb (EStorageSetView *storage_set_view, const char *name, void *data) { EShellView *shell_view; char *path; shell_view = E_SHELL_VIEW (data); path = g_strconcat (G_DIR_SEPARATOR_S, name, NULL); switch_on_folder_tree_click (shell_view, path); g_free (path); } /* Callbacks for the folder context menu in the folder bar. */ static void folder_context_menu_popping_up_cb (EStorageSetView *storage_set_view, const char *path, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); setup_verb_sensitivity_for_folder (shell_view, path); } static void folder_context_menu_popped_down_cb (EStorageSetView *storage_set_view, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); setup_verb_sensitivity_for_folder (shell_view, e_shell_view_get_current_path (shell_view)); if (shell_view->priv->folder_bar_popup != NULL) popdown_transient_folder_bar (shell_view); } /* Callback called when the button on the tree's title bar is clicked. */ static void storage_set_view_button_clicked_cb (ETitleBar *title_bar, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); if (shell_view->priv->folder_bar_popup == NULL) e_shell_view_show_folder_bar (shell_view, FALSE); } /* Callback called when the title bar button is clicked. */ static void title_bar_toggled_cb (EShellFolderTitleBar *title_bar, gboolean state, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); if (! state) return; if (shell_view->priv->folder_bar_popup == NULL) pop_up_folder_bar (shell_view); } /* Callback called when the offline toggle button is clicked. */ static void offline_toggle_clicked_cb (GtkButton *button, void *data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; switch (e_shell_get_line_status (priv->shell)) { case E_SHELL_LINE_STATUS_ONLINE: e_shell_go_offline (priv->shell, shell_view); break; case E_SHELL_LINE_STATUS_OFFLINE: e_shell_go_online (priv->shell, shell_view); break; default: g_assert_not_reached (); } } /* Widget setup. */ static void setup_storage_set_subwindow (EShellView *shell_view) { EShellViewPrivate *priv; GtkWidget *storage_set_view; GtkWidget *vbox; GtkWidget *scroll_frame; priv = shell_view->priv; storage_set_view = e_storage_set_view_new (e_shell_get_storage_set (priv->shell), priv->ui_container); gtk_signal_connect (GTK_OBJECT (storage_set_view), "folder_selected", GTK_SIGNAL_FUNC (folder_selected_cb), shell_view); gtk_signal_connect (GTK_OBJECT (storage_set_view), "storage_selected", GTK_SIGNAL_FUNC (storage_selected_cb), shell_view); gtk_signal_connect (GTK_OBJECT (storage_set_view), "folder_context_menu_popping_up", GTK_SIGNAL_FUNC (folder_context_menu_popping_up_cb), shell_view); gtk_signal_connect (GTK_OBJECT (storage_set_view), "folder_context_menu_popped_down", GTK_SIGNAL_FUNC (folder_context_menu_popped_down_cb), shell_view); scroll_frame = e_scroll_frame_new (NULL, NULL); e_scroll_frame_set_policy (E_SCROLL_FRAME (scroll_frame), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); e_scroll_frame_set_shadow_type (E_SCROLL_FRAME (scroll_frame), GTK_SHADOW_IN); gtk_container_add (GTK_CONTAINER (scroll_frame), storage_set_view); vbox = gtk_vbox_new (FALSE, 0); priv->storage_set_title_bar = e_title_bar_new (_("Folders")); gtk_box_pack_start (GTK_BOX (vbox), priv->storage_set_title_bar, FALSE, FALSE, 0); gtk_box_pack_start (GTK_BOX (vbox), scroll_frame, TRUE, TRUE, 0); gtk_signal_connect (GTK_OBJECT (priv->storage_set_title_bar), "button_clicked", GTK_SIGNAL_FUNC (storage_set_view_button_clicked_cb), shell_view); gtk_widget_show (storage_set_view); gtk_widget_show (priv->storage_set_title_bar); gtk_widget_show (scroll_frame); priv->storage_set_view_box = vbox; priv->storage_set_view = storage_set_view; /* Notice we don't show the vbox here yet. By default it's hidden. */ } static void setup_offline_toggle (EShellView *shell_view) { EShellViewPrivate *priv; GtkWidget *toggle; GtkWidget *pixmap; priv = shell_view->priv; toggle = gtk_button_new (); GTK_WIDGET_UNSET_FLAGS (toggle, GTK_CAN_FOCUS); gtk_button_set_relief (GTK_BUTTON (toggle), GTK_RELIEF_NONE); gtk_signal_connect (GTK_OBJECT (toggle), "clicked", GTK_SIGNAL_FUNC (offline_toggle_clicked_cb), shell_view); pixmap = gtk_pixmap_new (offline_pixmap, offline_mask); gtk_container_add (GTK_CONTAINER (toggle), pixmap); gtk_widget_show (toggle); gtk_widget_show (pixmap); priv->offline_toggle = toggle; priv->offline_toggle_pixmap = pixmap; update_offline_toggle_status (shell_view); g_assert (priv->status_bar != NULL); gtk_box_pack_start (GTK_BOX (priv->status_bar), priv->offline_toggle, FALSE, TRUE, 0); } static void setup_menu_hint_label (EShellView *shell_view) { EShellViewPrivate *priv; priv = shell_view->priv; priv->menu_hint_label = gtk_label_new (""); gtk_misc_set_alignment (GTK_MISC (priv->menu_hint_label), 0.0, 0.5); gtk_box_pack_start (GTK_BOX (priv->status_bar), priv->menu_hint_label, TRUE, TRUE, 0); } static void setup_task_bar (EShellView *shell_view) { EShellViewPrivate *priv; priv = shell_view->priv; priv->task_bar = e_task_bar_new (); g_assert (priv->status_bar != NULL); gtk_box_pack_start (GTK_BOX (priv->status_bar), priv->task_bar, TRUE, TRUE, 0); gtk_widget_show (priv->task_bar); } static void create_status_bar (EShellView *shell_view) { EShellViewPrivate *priv; priv = shell_view->priv; priv->status_bar = gtk_hbox_new (FALSE, 2); gtk_widget_show (priv->status_bar); setup_offline_toggle (shell_view); setup_menu_hint_label (shell_view); setup_task_bar (shell_view); } /* Menu hints for the status bar. */ static void ui_engine_add_hint_callback (BonoboUIEngine *engine, const char *hint, void *data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; gtk_label_set (GTK_LABEL (priv->menu_hint_label), hint); gtk_widget_show (priv->menu_hint_label); gtk_widget_hide (priv->task_bar); } static void ui_engine_remove_hint_callback (BonoboUIEngine *engine, void *data) { EShellView *shell_view; EShellViewPrivate *priv; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; gtk_widget_hide (priv->menu_hint_label); gtk_widget_show (priv->task_bar); } static void setup_statusbar_hints (EShellView *shell_view) { EShellViewPrivate *priv; BonoboUIEngine *ui_engine; priv = shell_view->priv; g_assert (priv->status_bar != NULL); ui_engine = bonobo_window_get_ui_engine (BONOBO_WINDOW (shell_view)); gtk_signal_connect (GTK_OBJECT (ui_engine), "add_hint", GTK_SIGNAL_FUNC (ui_engine_add_hint_callback), shell_view); gtk_signal_connect (GTK_OBJECT (ui_engine), "remove_hint", GTK_SIGNAL_FUNC (ui_engine_remove_hint_callback), shell_view); } static void setup_widgets (EShellView *shell_view) { EShellViewPrivate *priv; GtkWidget *contents_vbox; GtkWidget *gray_bar; priv = shell_view->priv; /* The shortcut bar. */ priv->shortcut_bar = e_shortcuts_new_view (e_shell_get_shortcuts (priv->shell)); gtk_signal_connect (GTK_OBJECT (priv->shortcut_bar), "activate_shortcut", GTK_SIGNAL_FUNC (activate_shortcut_cb), shell_view); gtk_signal_connect (GTK_OBJECT (priv->shortcut_bar), "hide_requested", GTK_SIGNAL_FUNC (hide_requested_cb), shell_view); priv->shortcut_frame = gtk_frame_new (NULL); gtk_frame_set_shadow_type (GTK_FRAME (priv->shortcut_frame), GTK_SHADOW_IN); /* The storage set view. */ setup_storage_set_subwindow (shell_view); /* The tabless notebook which we used to contain the views. */ priv->notebook = gtk_notebook_new (); gtk_notebook_set_show_border (GTK_NOTEBOOK (priv->notebook), FALSE); gtk_notebook_set_show_tabs (GTK_NOTEBOOK (priv->notebook), FALSE); /* Page for "No URL displayed" message. */ gtk_notebook_append_page (GTK_NOTEBOOK (priv->notebook), create_label_for_empty_page (), NULL); /* Put things into a paned and the paned into the GnomeApp. */ priv->view_vbox = gtk_vbox_new (FALSE, 0); priv->folder_title_bar = e_shell_folder_title_bar_new (); gtk_signal_connect (GTK_OBJECT (priv->folder_title_bar), "title_toggled", GTK_SIGNAL_FUNC (title_bar_toggled_cb), shell_view); priv->view_hpaned = e_hpaned_new (); e_paned_pack1 (E_PANED (priv->view_hpaned), priv->storage_set_view_box, FALSE, FALSE); e_paned_pack2 (E_PANED (priv->view_hpaned), priv->notebook, TRUE, FALSE); gray_bar = e_gray_bar_new (); gtk_container_add (GTK_CONTAINER (gray_bar), priv->folder_title_bar); gtk_box_pack_start (GTK_BOX (priv->view_vbox), gray_bar, FALSE, FALSE, 2); gtk_box_pack_start (GTK_BOX (priv->view_vbox), priv->view_hpaned, TRUE, TRUE, 0); priv->hpaned = e_hpaned_new (); gtk_container_add (GTK_CONTAINER (priv->shortcut_frame), priv->shortcut_bar); e_paned_pack1 (E_PANED (priv->hpaned), priv->shortcut_frame, FALSE, FALSE); e_paned_pack2 (E_PANED (priv->hpaned), priv->view_vbox, TRUE, FALSE); e_paned_set_position (E_PANED (priv->hpaned), DEFAULT_SHORTCUT_BAR_WIDTH); /* The status bar. */ create_status_bar (shell_view); setup_statusbar_hints (shell_view); /* The contents. */ contents_vbox = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (contents_vbox), priv->hpaned, TRUE, TRUE, 0); gtk_box_pack_start (GTK_BOX (contents_vbox), priv->status_bar, FALSE, TRUE, 0); gtk_widget_show (contents_vbox); bonobo_window_set_contents (BONOBO_WINDOW (shell_view), contents_vbox); /* Show stuff. */ gtk_widget_show (priv->shortcut_frame); gtk_widget_show (priv->shortcut_bar); gtk_widget_show (priv->storage_set_view); gtk_widget_show (priv->notebook); gtk_widget_show (priv->hpaned); gtk_widget_show (priv->view_hpaned); gtk_widget_show (priv->view_vbox); gtk_widget_show (priv->folder_title_bar); gtk_widget_show (priv->status_bar); gtk_widget_show (gray_bar); priv->shortcut_bar_shown = TRUE; priv->folder_bar_shown = FALSE; /* FIXME: Session management and stuff? */ gtk_window_set_default_size (GTK_WINDOW (shell_view), DEFAULT_WIDTH, DEFAULT_HEIGHT); } /* GtkObject methods. */ static void hash_foreach_destroy_view (void *name, void *value, void *data) { View *view; view = (View *) value; gtk_widget_destroy (view->control); view_destroy (view); } static void destroy (GtkObject *object) { EShellView *shell_view; EShellViewPrivate *priv; GList *p; shell_view = E_SHELL_VIEW (object); priv = shell_view->priv; cleanup_delayed_selection (shell_view); gtk_object_unref (GTK_OBJECT (priv->tooltips)); if (priv->corba_interface != NULL) bonobo_object_unref (BONOBO_OBJECT (priv->corba_interface)); if (priv->folder_bar_popup != NULL) gtk_widget_destroy (priv->folder_bar_popup); for (p = priv->sockets; p != NULL; p = p->next) { GtkWidget *socket_widget; int destroy_connection_id; socket_widget = GTK_WIDGET (p->data); destroy_connection_id = GPOINTER_TO_INT (gtk_object_get_data (GTK_OBJECT (socket_widget), "e_shell_view_destroy_connection_id")); gtk_signal_disconnect (GTK_OBJECT (socket_widget), destroy_connection_id); } g_hash_table_foreach (priv->uri_to_view, hash_foreach_destroy_view, NULL); g_hash_table_destroy (priv->uri_to_view); bonobo_object_unref (BONOBO_OBJECT (priv->ui_component)); g_free (priv->uri); if (priv->set_folder_timeout != 0) gtk_timeout_remove (priv->set_folder_timeout); g_free (priv->set_folder_uri); g_free (priv); (* GTK_OBJECT_CLASS (parent_class)->destroy) (object); } /* Initialization. */ static void class_init (EShellViewClass *klass) { GtkObjectClass *object_class; object_class = (GtkObjectClass *) klass; object_class->destroy = destroy; parent_class = gtk_type_class (BONOBO_TYPE_WINDOW); signals[SHORTCUT_BAR_VISIBILITY_CHANGED] = gtk_signal_new ("shortcut_bar_visibility_changed", GTK_RUN_FIRST, object_class->type, GTK_SIGNAL_OFFSET (EShellViewClass, shortcut_bar_visibility_changed), gtk_marshal_NONE__INT, GTK_TYPE_NONE, 1, GTK_TYPE_INT); signals[FOLDER_BAR_VISIBILITY_CHANGED] = gtk_signal_new ("folder_bar_visibility_changed", GTK_RUN_FIRST, object_class->type, GTK_SIGNAL_OFFSET (EShellViewClass, folder_bar_visibility_changed), gtk_marshal_NONE__INT, GTK_TYPE_NONE, 1, GTK_TYPE_INT); gtk_object_class_add_signals (object_class, signals, LAST_SIGNAL); load_images (); } static void init (EShellView *shell_view) { EShellViewPrivate *priv; priv = g_new (EShellViewPrivate, 1); priv->shell = NULL; priv->corba_interface = NULL; priv->ui_component = NULL; priv->uri = NULL; priv->delayed_selection = NULL; priv->tooltips = gtk_tooltips_new (); priv->appbar = NULL; priv->hpaned = NULL; priv->view_hpaned = NULL; priv->contents = NULL; priv->notebook = NULL; priv->storage_set_title_bar = NULL; priv->storage_set_view = NULL; priv->storage_set_view_box = NULL; priv->shortcut_bar = NULL; priv->status_bar = NULL; priv->offline_toggle = NULL; priv->offline_toggle_pixmap = NULL; priv->menu_hint_label = NULL; priv->task_bar = NULL; priv->folder_bar_popup = NULL; priv->shortcut_bar_shown = FALSE; priv->folder_bar_shown = FALSE; priv->hpaned_position = 0; priv->view_hpaned_position = 0; priv->uri_to_view = g_hash_table_new (g_str_hash, g_str_equal); priv->sockets = NULL; priv->set_folder_timeout = 0; priv->set_folder_uri = NULL; shell_view->priv = priv; } /* EvolutionShellView interface callbacks. */ static void corba_interface_set_message_cb (EvolutionShellView *shell_view, const char *message, gboolean busy, void *data) { /* Don't do anything here anymore. The interface is going to be deprecated soon. */ } static void corba_interface_unset_message_cb (EvolutionShellView *shell_view, void *data) { /* Don't do anything here anymore. The interface is going to be deprecated soon. */ } static void corba_interface_change_current_view_cb (EvolutionShellView *shell_view, const char *uri, void *data) { EShellView *view; view = E_SHELL_VIEW (data); g_return_if_fail (view != NULL); e_shell_view_display_uri (view, uri); } static void corba_interface_set_title (EvolutionShellView *shell_view, const char *title, void *data) { EShellView *view; view = E_SHELL_VIEW (data); g_return_if_fail (view != NULL); gtk_window_set_title (GTK_WINDOW (view), title); } static void corba_interface_set_folder_bar_label (EvolutionShellView *evolution_shell_view, const char *text, void *data) { EShellView *shell_view; EShellViewPrivate *priv; g_return_if_fail (data != NULL); g_return_if_fail (E_IS_SHELL_VIEW (data)); shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; e_shell_folder_title_bar_set_folder_bar_label (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), text); } static void unmerge_on_error (BonoboObject *object, CORBA_Object cobject, CORBA_Environment *ev) { BonoboWindow *win; win = bonobo_ui_container_get_win (BONOBO_UI_CONTAINER (object)); if (win) bonobo_window_deregister_component_by_ref (win, cobject); } static void updated_folder_cb (EStorageSet *storage_set, const char *path, void *data) { EShellView *shell_view; EShellViewPrivate *priv; const char *view_path; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; view_path = get_storage_set_path_from_uri (priv->uri); if (view_path && strcmp (path, view_path) != 0) return; /* Update the folder title bar and the window title bar */ update_for_current_uri (shell_view); } /* Shell callbacks. */ static void shell_line_status_changed_cb (EShell *shell, EShellLineStatus new_status, void *data) { EShellView *shell_view; shell_view = E_SHELL_VIEW (data); update_offline_toggle_status (shell_view); } static int delete_event_cb (GtkWidget *widget, GdkEventAny *ev, void *data) { return FALSE; } EShellView * e_shell_view_construct (EShellView *shell_view, EShell *shell) { EShellViewPrivate *priv; EShellView *view; g_return_val_if_fail (shell != NULL, NULL); g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL (shell), NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); priv = shell_view->priv; view = E_SHELL_VIEW (bonobo_window_construct (BONOBO_WINDOW (shell_view), "evolution", "Ximian Evolution")); if (!view) { gtk_object_unref (GTK_OBJECT (shell_view)); return NULL; } priv->shell = shell; gtk_signal_connect (GTK_OBJECT (view), "delete_event", GTK_SIGNAL_FUNC (delete_event_cb), NULL); gtk_signal_connect_while_alive (GTK_OBJECT (e_shell_get_storage_set (priv->shell)), "updated_folder", updated_folder_cb, shell_view, GTK_OBJECT (shell_view)); priv->ui_container = bonobo_ui_container_new (); bonobo_ui_container_set_win (priv->ui_container, BONOBO_WINDOW (shell_view)); gtk_signal_connect (GTK_OBJECT (priv->ui_container), "system_exception", GTK_SIGNAL_FUNC (unmerge_on_error), NULL); priv->ui_component = bonobo_ui_component_new ("evolution"); bonobo_ui_component_set_container (priv->ui_component, bonobo_object_corba_objref (BONOBO_OBJECT (priv->ui_container))); bonobo_ui_component_freeze (priv->ui_component, NULL); bonobo_ui_util_set_ui (priv->ui_component, EVOLUTION_DATADIR, "evolution.xml", "evolution"); setup_widgets (shell_view); bonobo_ui_engine_config_set_path (bonobo_window_get_ui_engine (BONOBO_WINDOW (shell_view)), "/evolution/UIConf/kvps"); e_shell_view_menu_setup (shell_view); e_shell_view_show_folder_bar (shell_view, FALSE); bonobo_ui_component_thaw (priv->ui_component, NULL); gtk_signal_connect_while_alive (GTK_OBJECT (shell), "line_status_changed", GTK_SIGNAL_FUNC (shell_line_status_changed_cb), shell_view, GTK_OBJECT (shell_view)); gtk_signal_connect (GTK_OBJECT (e_shell_get_storage_set (shell)), "removed_folder", GTK_SIGNAL_FUNC (storage_set_removed_folder_callback), shell_view); e_shell_user_creatable_items_handler_setup_menus (e_shell_get_user_creatable_items_handler (priv->shell), shell_view); return view; } /* WARNING: Don't use `e_shell_view_new()' to create new views for the shell unless you know what you are doing; this is just the standard GTK+ constructor thing and it won't allow the shell to do the required bookkeeping for the created views. Instead, the right way to create a new view is calling `e_shell_create_view()'. */ EShellView * e_shell_view_new (EShell *shell) { GtkWidget *new; g_return_val_if_fail (shell != NULL, NULL); g_return_val_if_fail (E_IS_SHELL (shell), NULL); new = gtk_type_new (e_shell_view_get_type ()); return e_shell_view_construct (E_SHELL_VIEW (new), shell); } const GNOME_Evolution_ShellView e_shell_view_get_corba_interface (EShellView *shell_view) { EShellViewPrivate *priv; g_return_val_if_fail (shell_view != NULL, CORBA_OBJECT_NIL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), CORBA_OBJECT_NIL); priv = shell_view->priv; return bonobo_object_corba_objref (BONOBO_OBJECT (priv->corba_interface)); } static const char * get_storage_set_path_from_uri (const char *uri) { const char *colon; if (uri == NULL) return NULL; if (g_path_is_absolute (uri)) return NULL; colon = strchr (uri, ':'); if (colon == NULL || colon == uri || colon[1] == '\0') return NULL; if (! g_path_is_absolute (colon + 1)) return NULL; if (g_strncasecmp (uri, E_SHELL_URI_PREFIX, colon - uri) != 0) return NULL; return colon + 1; } static void update_window_icon (EShellView *shell_view, const char *type) { EShellViewPrivate *priv; const char *icon_name; char *icon_path; priv = shell_view->priv; if (type == NULL) { icon_path = NULL; } else { EFolderTypeRegistry *folder_type_registry; folder_type_registry = e_shell_get_folder_type_registry (priv->shell); icon_name = e_folder_type_registry_get_icon_name_for_type (folder_type_registry, type); if (icon_name == NULL) icon_path = NULL; else icon_path = e_shell_get_icon_path (icon_name, TRUE); } if (icon_path == NULL) { gnome_window_icon_set_from_default (GTK_WINDOW (shell_view)); } else { gnome_window_icon_set_from_file (GTK_WINDOW (shell_view), icon_path); g_free (icon_path); } } static void update_folder_title_bar (EShellView *shell_view, const char *title, const char *type) { EShellViewPrivate *priv; EFolderTypeRegistry *folder_type_registry; GdkPixbuf *folder_icon; priv = shell_view->priv; if (type == NULL) { title = NULL; folder_icon = NULL; } else { folder_type_registry = e_shell_get_folder_type_registry (priv->shell); folder_icon = e_folder_type_registry_get_icon_for_type (folder_type_registry, type, TRUE); } if (folder_icon != NULL) e_shell_folder_title_bar_set_icon (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), folder_icon); if (title != NULL) { char *s; s = e_utf8_to_gtk_string (GTK_WIDGET (priv->folder_title_bar), title); e_shell_folder_title_bar_set_title (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), s); g_free (s); } } static void update_for_current_uri (EShellView *shell_view) { EShellViewPrivate *priv; EFolder *folder; const char *path; const char *type; const char *folder_name; char *title; char *utf8_window_title; char *gtk_window_title; int unread_count; priv = shell_view->priv; /* if we update when there is a timeout set, the selection * will jump around against the user's wishes. so we just * return. */ if (priv->set_folder_timeout != 0) return; path = get_storage_set_path_from_uri (priv->uri); folder_name = NULL; type = NULL; unread_count = 0; if (path == NULL) { folder = NULL; } else { folder = e_storage_set_get_folder (e_shell_get_storage_set (priv->shell), path); if (folder != NULL) { folder_name = e_folder_get_name (folder); type = e_folder_get_type_string (folder); unread_count = e_folder_get_unread_count (folder); } else if (path != NULL) { EStorage *storage; storage = e_storage_set_get_storage (e_shell_get_storage_set (priv->shell), path + 1); unread_count = 0; if (storage != NULL) { folder_name = e_storage_get_display_name (storage); type = e_storage_get_toplevel_node_type (storage); } } } if (unread_count > 0) title = g_strdup_printf (_("%s (%d)"), folder_name, unread_count); else if (folder_name == NULL) title = g_strdup (_("(None)")); else title = g_strdup (folder_name); if (SUB_VERSION[0] == '\0') utf8_window_title = g_strdup_printf ("%s - Ximian Evolution %s", title, VERSION); else utf8_window_title = g_strdup_printf ("%s - Ximian Evolution %s [%s]", title, VERSION, SUB_VERSION); gtk_window_title = e_utf8_to_gtk_string (GTK_WIDGET (shell_view), utf8_window_title); gtk_window_set_title (GTK_WINDOW (shell_view), gtk_window_title); update_folder_title_bar (shell_view, title, type); update_window_icon (shell_view, type); g_free (gtk_window_title); g_free (utf8_window_title); g_free (title); gtk_signal_handler_block_by_func (GTK_OBJECT (priv->storage_set_view), GTK_SIGNAL_FUNC (folder_selected_cb), shell_view); if (path != NULL) e_storage_set_view_set_current_folder (E_STORAGE_SET_VIEW (priv->storage_set_view), path); gtk_signal_handler_unblock_by_func (GTK_OBJECT (priv->storage_set_view), GTK_SIGNAL_FUNC (folder_selected_cb), shell_view); } static void update_offline_toggle_status (EShellView *shell_view) { EShellViewPrivate *priv; GdkPixmap *icon_pixmap; GdkBitmap *icon_mask; const char *tooltip; gboolean sensitive; priv = shell_view->priv; switch (e_shell_get_line_status (priv->shell)) { case E_SHELL_LINE_STATUS_ONLINE: icon_pixmap = online_pixmap; icon_mask = online_mask; sensitive = TRUE; tooltip = _("Ximian Evolution is currently online. " "Click on this button to work offline."); break; case E_SHELL_LINE_STATUS_GOING_OFFLINE: icon_pixmap = online_pixmap; icon_mask = online_mask; sensitive = FALSE; tooltip = _("Ximian Evolution is in the process of going offline."); break; case E_SHELL_LINE_STATUS_OFFLINE: icon_pixmap = offline_pixmap; icon_mask = offline_mask; sensitive = TRUE; tooltip = _("Ximian Evolution is currently offline. " "Click on this button to work online."); break; default: g_assert_not_reached (); return; } gtk_pixmap_set (GTK_PIXMAP (priv->offline_toggle_pixmap), icon_pixmap, icon_mask); gtk_widget_set_sensitive (priv->offline_toggle, sensitive); gtk_tooltips_set_tip (priv->tooltips, priv->offline_toggle, tooltip, NULL); } /* This displays the specified page, doing the appropriate Bonobo activation/deactivation magic to make sure things work nicely. FIXME: Crappy way to solve the issue. */ static void set_current_notebook_page (EShellView *shell_view, int page_num) { EShellViewPrivate *priv; GtkNotebook *notebook; GtkWidget *current; BonoboControlFrame *control_frame; int current_page; priv = shell_view->priv; notebook = GTK_NOTEBOOK (priv->notebook); current_page = gtk_notebook_get_current_page (notebook); if (current_page == page_num) return; if (current_page != -1 && current_page != 0) { current = gtk_notebook_get_nth_page (notebook, current_page); control_frame = bonobo_widget_get_control_frame (BONOBO_WIDGET (current)); bonobo_control_frame_set_autoactivate (control_frame, FALSE); bonobo_control_frame_control_deactivate (control_frame); } e_shell_folder_title_bar_set_folder_bar_label (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), ""); gtk_notebook_set_page (notebook, page_num); if (page_num == -1 || page_num == 0) return; current = gtk_notebook_get_nth_page (notebook, page_num); control_frame = bonobo_widget_get_control_frame (BONOBO_WIDGET (current)); bonobo_control_frame_set_autoactivate (control_frame, FALSE); bonobo_control_frame_control_activate (control_frame); } static void setup_corba_interface (EShellView *shell_view, GtkWidget *control) { EShellViewPrivate *priv; BonoboControlFrame *control_frame; EvolutionShellView *corba_interface; g_return_if_fail (control != NULL); priv = shell_view->priv; control_frame = bonobo_widget_get_control_frame (BONOBO_WIDGET (control)); corba_interface = evolution_shell_view_new (); gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "set_message", GTK_SIGNAL_FUNC (corba_interface_set_message_cb), shell_view, GTK_OBJECT (shell_view)); gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "unset_message", GTK_SIGNAL_FUNC (corba_interface_unset_message_cb), shell_view, GTK_OBJECT (shell_view)); gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "change_current_view", GTK_SIGNAL_FUNC (corba_interface_change_current_view_cb), shell_view, GTK_OBJECT (shell_view)); gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "set_title", GTK_SIGNAL_FUNC (corba_interface_set_title), shell_view, GTK_OBJECT (shell_view)); gtk_signal_connect_while_alive (GTK_OBJECT (corba_interface), "set_folder_bar_label", GTK_SIGNAL_FUNC (corba_interface_set_folder_bar_label), shell_view, GTK_OBJECT (shell_view)); bonobo_object_add_interface (BONOBO_OBJECT (control_frame), BONOBO_OBJECT (corba_interface)); bonobo_object_ref (BONOBO_OBJECT (corba_interface)); priv->corba_interface = corba_interface; } /* Socket destruction handling. */ static void socket_destroy_cb (GtkWidget *socket_widget, gpointer data) { EShellView *shell_view; EShellViewPrivate *priv; EFolder *folder; View *view; const char *uri; gboolean viewing_closed_uri; const char *current_uri; const char *path; const char *folder_type; shell_view = E_SHELL_VIEW (data); priv = shell_view->priv; uri = (const char *) gtk_object_get_data (GTK_OBJECT (socket_widget), "e_shell_view_folder_uri"); view = g_hash_table_lookup (priv->uri_to_view, uri); if (view == NULL) { g_warning ("What?! Destroyed socket for non-existing URI? -- %s", uri); return; } priv->sockets = g_list_remove (priv->sockets, socket_widget); gtk_widget_destroy (view->control); g_hash_table_remove (priv->uri_to_view, view->uri); view_destroy (view); path = get_storage_set_path_from_uri (uri); folder = e_storage_set_get_folder (e_shell_get_storage_set (priv->shell), path); if (folder != NULL) { folder_type = e_folder_get_type_string (folder); } else { EStorage *storage; storage = e_storage_set_get_storage (e_shell_get_storage_set (priv->shell), path + 1); if (storage == NULL) folder_type = NULL; else folder_type = e_storage_get_toplevel_node_type (storage); } /* See if we were actively viewing the uri for the socket that's being closed */ current_uri = e_shell_view_get_current_uri (shell_view); if (current_uri == NULL) { viewing_closed_uri = FALSE; } else { if (strcmp (uri, current_uri) == 0) viewing_closed_uri = TRUE; else viewing_closed_uri = FALSE; } if (viewing_closed_uri) e_shell_view_display_uri (shell_view, NULL); e_shell_component_maybe_crashed (priv->shell, uri, folder_type, shell_view); /* We were actively viewing the component that just crashed, so flip to the default URI */ if (viewing_closed_uri) e_shell_view_display_uri (shell_view, E_SHELL_VIEW_DEFAULT_URI); } static const char * get_type_for_storage (EShellView *shell_view, const char *name, const char **physical_uri_return) { EShellViewPrivate *priv; EStorageSet *storage_set; EStorage *storage; priv = shell_view->priv; storage_set = e_shell_get_storage_set (priv->shell); storage = e_storage_set_get_storage (storage_set, name); if (!storage) return NULL; *physical_uri_return = e_storage_get_toplevel_node_uri (storage); return e_storage_get_toplevel_node_type (storage); } static const char * get_type_for_folder (EShellView *shell_view, const char *path, const char **physical_uri_return) { EShellViewPrivate *priv; EStorageSet *storage_set; EFolder *folder; priv = shell_view->priv; storage_set = e_shell_get_storage_set (priv->shell); folder = e_storage_set_get_folder (storage_set, path); if (!folder) return NULL; if (physical_uri_return != NULL) *physical_uri_return = e_folder_get_physical_uri (folder); return e_folder_get_type_string (folder); } /* Create a new view for @uri with @control. It assumes a view for @uri does not exist yet. */ static View * get_view_for_uri (EShellView *shell_view, const char *uri) { EShellViewPrivate *priv; CORBA_Environment ev; EvolutionShellComponentClient *handler_client; EFolderTypeRegistry *folder_type_registry; GNOME_Evolution_ShellComponent handler; Bonobo_UIContainer container; GtkWidget *control; GtkWidget *socket; Bonobo_Control corba_control; const char *path; const char *slash; const char *physical_uri; const char *folder_type; int destroy_connection_id; priv = shell_view->priv; path = strchr (uri, ':'); if (path == NULL) return NULL; path++; if (*path == '\0') return NULL; /* FIXME: This code needs to be made more robust. */ slash = strchr (path + 1, G_DIR_SEPARATOR); if (slash == NULL || slash[1] == '\0') folder_type = get_type_for_storage (shell_view, path + 1, &physical_uri); else folder_type = get_type_for_folder (shell_view, path, &physical_uri); if (folder_type == NULL) return NULL; folder_type_registry = e_shell_get_folder_type_registry (e_shell_view_get_shell (shell_view)); handler_client = e_folder_type_registry_get_handler_for_type (folder_type_registry, folder_type); if (handler_client == CORBA_OBJECT_NIL) return NULL; handler = bonobo_object_corba_objref (BONOBO_OBJECT (handler_client)); CORBA_exception_init (&ev); corba_control = GNOME_Evolution_ShellComponent_createView (handler, physical_uri, folder_type, &ev); if (ev._major != CORBA_NO_EXCEPTION) { CORBA_exception_free (&ev); return NULL; } CORBA_exception_free (&ev); if (corba_control == CORBA_OBJECT_NIL) return NULL; container = bonobo_ui_component_get_container (priv->ui_component); control = e_bonobo_widget_new_control_from_objref (corba_control, container); socket = find_socket (GTK_CONTAINER (control)); destroy_connection_id = gtk_signal_connect (GTK_OBJECT (socket), "destroy", GTK_SIGNAL_FUNC (socket_destroy_cb), shell_view); gtk_object_set_data (GTK_OBJECT (socket), "e_shell_view_destroy_connection_id", GINT_TO_POINTER (destroy_connection_id)); gtk_object_set_data_full (GTK_OBJECT (socket), "e_shell_view_folder_uri", g_strdup (uri), g_free); priv->sockets = g_list_prepend (priv->sockets, socket); setup_corba_interface (shell_view, control); return view_new (uri, control); } static gboolean show_existing_view (EShellView *shell_view, const char *uri, View *view) { EShellViewPrivate *priv; int notebook_page; priv = shell_view->priv; notebook_page = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), view->control); g_assert (notebook_page != -1); /* A BonoboWidget can be a "zombie" in the sense that its actual control is dead; if it's zombie, we have to recreate it. */ if (bonobo_widget_is_dead (BONOBO_WIDGET (view->control))) return FALSE; g_free (priv->uri); priv->uri = g_strdup (uri); set_current_notebook_page (shell_view, notebook_page); return TRUE; } static gboolean create_new_view_for_uri (EShellView *shell_view, const char *uri) { View *view; EShellViewPrivate *priv; int page_num; priv = shell_view->priv; view = get_view_for_uri (shell_view, uri); if (view == NULL) return FALSE; g_free (priv->uri); priv->uri = g_strdup (uri); gtk_widget_show (view->control); gtk_notebook_append_page (GTK_NOTEBOOK (priv->notebook), view->control, NULL); page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), view->control); g_assert (page_num != -1); set_current_notebook_page (shell_view, page_num); g_hash_table_insert (priv->uri_to_view, view->uri, view); return TRUE; } gboolean e_shell_view_display_uri (EShellView *shell_view, const char *uri) { EShellViewPrivate *priv; View *view; gboolean retval; g_return_val_if_fail (shell_view != NULL, FALSE); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE); priv = shell_view->priv; bonobo_window_freeze (BONOBO_WINDOW (shell_view)); if (uri == NULL) { gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), 0); gtk_notebook_prepend_page (GTK_NOTEBOOK (priv->notebook), create_label_for_empty_page (), NULL); set_current_notebook_page (shell_view, 0); g_free (priv->uri); priv->uri = NULL; retval = TRUE; goto end; } if (strncmp (uri, E_SHELL_URI_PREFIX, E_SHELL_URI_PREFIX_LEN) != 0) { retval = FALSE; goto end; } view = g_hash_table_lookup (priv->uri_to_view, uri); if (view != NULL) { show_existing_view (shell_view, uri, view); } else if (! create_new_view_for_uri (shell_view, uri)) { cleanup_delayed_selection (shell_view); priv->delayed_selection = g_strdup (uri); e_gtk_signal_connect_full_while_alive (GTK_OBJECT (e_shell_get_storage_set (priv->shell)), "new_folder", GTK_SIGNAL_FUNC (new_folder_cb), NULL, shell_view, NULL, NULL, TRUE, GTK_OBJECT (shell_view)); retval = FALSE; goto end; } retval = TRUE; end: g_free (priv->set_folder_uri); priv->set_folder_uri = NULL; if (priv->set_folder_timeout != 0) { gtk_timeout_remove (priv->set_folder_timeout); priv->set_folder_timeout = 0; } update_for_current_uri (shell_view); bonobo_window_thaw (BONOBO_WINDOW (shell_view)); return retval; } void e_shell_view_show_shortcut_bar (EShellView *shell_view, gboolean show) { EShellViewPrivate *priv; g_return_if_fail (shell_view != NULL); g_return_if_fail (E_IS_SHELL_VIEW (shell_view)); priv = shell_view->priv; if (!! show == priv->shortcut_bar_shown) return; if (show) { if (! GTK_WIDGET_VISIBLE (priv->shortcut_frame)) { gtk_widget_show (priv->shortcut_frame); e_paned_set_position (E_PANED (priv->hpaned), priv->hpaned_position); } } else { if (GTK_WIDGET_VISIBLE (priv->shortcut_frame)) { /* FIXME this is a private field! */ priv->hpaned_position = E_PANED (priv->hpaned)->child1_size; gtk_widget_hide (priv->shortcut_frame); } e_paned_set_position (E_PANED (priv->hpaned), 0); } priv->shortcut_bar_shown = !! show; gtk_signal_emit (GTK_OBJECT (shell_view), signals[SHORTCUT_BAR_VISIBILITY_CHANGED], priv->shortcut_bar_shown); } void e_shell_view_show_folder_bar (EShellView *shell_view, gboolean show) { EShellViewPrivate *priv; g_return_if_fail (shell_view != NULL); g_return_if_fail (E_IS_SHELL_VIEW (shell_view)); priv = shell_view->priv; if (!! show == priv->folder_bar_shown) return; if (show) { gtk_widget_show (priv->storage_set_view_box); e_paned_set_position (E_PANED (priv->view_hpaned), priv->view_hpaned_position); e_title_bar_set_button_mode (E_TITLE_BAR (priv->storage_set_title_bar), E_TITLE_BAR_BUTTON_MODE_CLOSE); e_shell_folder_title_bar_set_clickable (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), FALSE); } else { if (GTK_WIDGET_VISIBLE (priv->storage_set_view_box)) { /* FIXME this is a private field! */ priv->view_hpaned_position = E_PANED (priv->view_hpaned)->child1_size; gtk_widget_hide (priv->storage_set_view_box); } e_paned_set_position (E_PANED (priv->view_hpaned), 0); e_title_bar_set_button_mode (E_TITLE_BAR (priv->storage_set_title_bar), E_TITLE_BAR_BUTTON_MODE_PIN); e_shell_folder_title_bar_set_clickable (E_SHELL_FOLDER_TITLE_BAR (priv->folder_title_bar), TRUE); } priv->folder_bar_shown = !! show; gtk_signal_emit (GTK_OBJECT (shell_view), signals[FOLDER_BAR_VISIBILITY_CHANGED], priv->folder_bar_shown); } gboolean e_shell_view_shortcut_bar_shown (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, FALSE); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE); return shell_view->priv->shortcut_bar_shown; } gboolean e_shell_view_folder_bar_shown (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, FALSE); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE); return shell_view->priv->folder_bar_shown; } ETaskBar * e_shell_view_get_task_bar (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); return E_TASK_BAR (shell_view->priv->task_bar); } EShell * e_shell_view_get_shell (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); return shell_view->priv->shell; } BonoboUIComponent * e_shell_view_get_bonobo_ui_component (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); return shell_view->priv->ui_component; } BonoboUIContainer * e_shell_view_get_bonobo_ui_container (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); return shell_view->priv->ui_container; } GtkWidget * e_shell_view_get_appbar (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); return shell_view->priv->appbar; } /** * e_shell_view_get_current_uri: * @shell_view: A pointer to an EShellView object * * Get the URI currently displayed by this shell view. * * Return value: **/ const char * e_shell_view_get_current_uri (EShellView *shell_view) { g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); return shell_view->priv->uri; } /** * e_shell_view_get_current_path: * @shell_view: A pointer to an EShellView object * * Get the path of the current displayed folder. * * Return value: **/ const char * e_shell_view_get_current_path (EShellView *shell_view) { const char *current_uri; const char *current_path; current_uri = e_shell_view_get_current_uri (shell_view); if (current_uri == NULL) return NULL; if (strncmp (current_uri, E_SHELL_URI_PREFIX, E_SHELL_URI_PREFIX_LEN) == 0) current_path = current_uri + E_SHELL_URI_PREFIX_LEN; else current_path = NULL; return current_path; } const char * e_shell_view_get_current_physical_uri (EShellView *shell_view) { const char *current_path; const char *physical_uri; g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); current_path = e_shell_view_get_current_path (shell_view); if (current_path == NULL) return NULL; if (get_type_for_folder (shell_view, current_path, &physical_uri) == NULL) return NULL; else return physical_uri; } const char * e_shell_view_get_current_folder_type (EShellView *shell_view) { const char *current_path; g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); current_path = e_shell_view_get_current_path (shell_view); if (current_path == NULL) return NULL; return get_type_for_folder (shell_view, current_path, NULL); } /** * e_shell_view_save_settings: * @shell_view: * @prefix: * * Save settings for @shell_view at the specified gnome config @prefix * * Return value: TRUE if successful, FALSE if not. **/ gboolean e_shell_view_save_settings (EShellView *shell_view, int view_num) { Bonobo_ConfigDatabase db; EShellViewPrivate *priv; EShortcutBar *shortcut_bar; const char *uri; char *prefix, *key; char *filename; int num_groups; int group; struct stat temp; g_return_val_if_fail (shell_view != NULL, FALSE); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE); priv = shell_view->priv; shortcut_bar = E_SHORTCUT_BAR (priv->shortcut_bar); db = e_shell_get_config_db (priv->shell); g_return_val_if_fail (db != CORBA_OBJECT_NIL, FALSE); prefix = g_strdup_printf ("/Shell/Views/%d/", view_num); key = g_strconcat (prefix, "Width", NULL); bonobo_config_set_long (db, key, GTK_WIDGET (shell_view)->allocation.width, NULL); g_free (key); key = g_strconcat (prefix, "Height", NULL); bonobo_config_set_long (db, key, GTK_WIDGET (shell_view)->allocation.height, NULL); g_free (key); key = g_strconcat (prefix, "CurrentShortcutsGroupNum", NULL); bonobo_config_set_long (db, key, e_shell_view_get_current_shortcuts_group_num (shell_view), NULL); g_free (key); key = g_strconcat (prefix, "FolderBarShown", NULL); bonobo_config_set_long (db, key, e_shell_view_folder_bar_shown (shell_view), NULL); g_free (key); key = g_strconcat (prefix, "ShortcutBarShown", NULL); bonobo_config_set_long (db, key, e_shell_view_shortcut_bar_shown (shell_view), NULL); g_free (key); key = g_strconcat (prefix, "HPanedPosition", NULL); if (GTK_WIDGET_VISIBLE (priv->shortcut_frame)) bonobo_config_set_long (db, key, E_PANED (priv->hpaned)->child1_size, NULL); else bonobo_config_set_long (db, key, priv->hpaned_position, NULL); g_free (key); key = g_strconcat (prefix, "ViewHPanedPosition", NULL); if (GTK_WIDGET_VISIBLE (priv->storage_set_view_box)) bonobo_config_set_long (db, key, E_PANED (priv->view_hpaned)->child1_size, NULL); else bonobo_config_set_long (db, key, priv->view_hpaned_position, NULL); g_free (key); key = g_strconcat (prefix, "DisplayedURI", NULL); uri = e_shell_view_get_current_uri (shell_view); if (uri != NULL) bonobo_config_set_string (db, key, uri, NULL); else bonobo_config_set_string (db, key, E_SHELL_VIEW_DEFAULT_URI, NULL); g_free (key); num_groups = e_shortcut_model_get_num_groups (shortcut_bar->model); for (group = 0; group < num_groups; group++) { key = g_strdup_printf ("%sShortcutBarGroup%dIconMode", prefix, group); bonobo_config_set_long (db, key, e_shortcut_bar_get_view_type (shortcut_bar, group), NULL); g_free (key); } g_free (prefix); /* If ~/evolution/config/ doesn't exist yet, make it */ filename = g_strdup_printf ("%s/config/", e_shell_get_local_directory (priv->shell)); if (stat (filename, &temp) != 0) mkdir (filename, S_IRWXU); g_free (filename); /* Save the expanded state for this ShellView's StorageSetView */ filename = g_strdup_printf ("%s/config/storage-set-view-expanded:view_%d", e_shell_get_local_directory (priv->shell), view_num); e_tree_save_expanded_state (E_TREE (priv->storage_set_view), filename); g_free (filename); return TRUE; } /** * e_shell_view_load_settings: * @shell_view: * @prefix: * * Load settings for @shell_view at the specified gnome config @prefix * * Return value: **/ gboolean e_shell_view_load_settings (EShellView *shell_view, int view_num) { Bonobo_ConfigDatabase db; EShellViewPrivate *priv; EShortcutBar *shortcut_bar; int num_groups, group, val; long width, height; char *stringval, *prefix, *filename, *key; g_return_val_if_fail (shell_view != NULL, FALSE); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), FALSE); priv = shell_view->priv; shortcut_bar = E_SHORTCUT_BAR (priv->shortcut_bar); db = e_shell_get_config_db (priv->shell); g_return_val_if_fail (db != CORBA_OBJECT_NIL, FALSE); prefix = g_strdup_printf ("/Shell/Views/%d/", view_num); key = g_strconcat (prefix, "Width", NULL); width = bonobo_config_get_long (db, key, NULL); g_free (key); key = g_strconcat (prefix, "Height", NULL); height = bonobo_config_get_long (db, key, NULL); g_free (key); gtk_window_set_default_size (GTK_WINDOW (shell_view), width, height); key = g_strconcat (prefix, "CurrentShortcutsGroupNum", NULL); val = bonobo_config_get_long (db, key, NULL); e_shell_view_set_current_shortcuts_group_num (shell_view, val); g_free (key); key = g_strconcat (prefix, "FolderBarShown", NULL); val = bonobo_config_get_long (db, key, NULL); e_shell_view_show_folder_bar (shell_view, val); g_free (key); key = g_strconcat (prefix, "ShortcutBarShown", NULL); val = bonobo_config_get_long (db, key, NULL); e_shell_view_show_shortcut_bar (shell_view, val); g_free (key); key = g_strconcat (prefix, "HPanedPosition", NULL); val = bonobo_config_get_long (db, key, NULL); if (priv->shortcut_bar_shown) e_paned_set_position (E_PANED (priv->hpaned), val); priv->hpaned_position = val; g_free (key); key = g_strconcat (prefix, "ViewHPanedPosition", NULL); val = bonobo_config_get_long (db, key, NULL); if (priv->folder_bar_shown) e_paned_set_position (E_PANED (priv->view_hpaned), val); priv->view_hpaned_position = val; g_free (key); key = g_strconcat (prefix, "DisplayedURI", NULL); stringval = bonobo_config_get_string (db, key, NULL); if (! e_shell_view_display_uri (shell_view, stringval)) e_shell_view_display_uri (shell_view, E_SHELL_VIEW_DEFAULT_URI); g_free (stringval); g_free (key); num_groups = e_shortcut_model_get_num_groups (shortcut_bar->model); for (group = 0; group < num_groups; group++) { int iconmode; key = g_strdup_printf ("%sShortcutBarGroup%dIconMode", prefix, group); iconmode = bonobo_config_get_long (db, key, NULL); g_free (key); e_shortcut_bar_set_view_type (shortcut_bar, group, iconmode); } g_free (prefix); /* Load the expanded state for the ShellView's StorageSetView */ filename = g_strdup_printf ("%s/config/storage-set-view-expanded:view_%d", e_shell_get_local_directory (priv->shell), view_num); e_tree_load_expanded_state (E_TREE (priv->storage_set_view), filename); g_free (filename); return TRUE; } /* FIXME: This function could become static */ void e_shell_view_set_current_shortcuts_group_num (EShellView *shell_view, int group_num) { EShellViewPrivate *priv; EShortcutsView *shortcuts_view; g_return_if_fail (shell_view != NULL); g_return_if_fail (E_IS_SHELL_VIEW (shell_view)); priv = shell_view->priv; shortcuts_view = E_SHORTCUTS_VIEW (priv->shortcut_bar); e_group_bar_set_current_group_num (E_GROUP_BAR (E_SHORTCUT_BAR (shortcuts_view)), group_num, FALSE); } int e_shell_view_get_current_shortcuts_group_num (EShellView *shell_view) { EShellViewPrivate *priv; EShortcutsView *shortcuts_view; int group; g_return_val_if_fail (shell_view != NULL, -1); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), -1); priv = shell_view->priv; shortcuts_view = E_SHORTCUTS_VIEW (priv->shortcut_bar); group = e_group_bar_get_current_group_num (E_GROUP_BAR (E_SHORTCUT_BAR (shortcuts_view))); return group; } const char * e_shell_view_get_folder_bar_right_click_path (EShellView *shell_view) { EShellViewPrivate *priv; g_return_val_if_fail (shell_view != NULL, NULL); g_return_val_if_fail (E_IS_SHELL_VIEW (shell_view), NULL); priv = shell_view->priv; return e_storage_set_view_get_right_click_path (E_STORAGE_SET_VIEW (priv->storage_set_view)); } E_MAKE_TYPE (e_shell_view, "EShellView", EShellView, class_init, init, BONOBO_TYPE_WINDOW)