/* * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) version 3. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with the program; if not, see * * * Authors: * Rodrigo Moya * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * */ /* This is prototype code only, this may, or may not, use undocumented * unstable or private internal function calls. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include "format-handler.h" /* Plugin entry points */ gboolean calendar_save_as_init (GtkUIManager *ui_manager, EShellView *shell_view); gboolean memo_list_save_as_init (GtkUIManager *ui_manager, EShellView *shell_view); gboolean task_list_save_as_init (GtkUIManager *ui_manager, EShellView *shell_view); gint e_plugin_lib_enable (EPlugin *ep, gint enable); gint e_plugin_lib_enable (EPlugin *ep, gint enable) { return 0; } enum { /* GtkComboBox enum */ DEST_NAME_COLUMN, DEST_HANDLER, N_DEST_COLUMNS }; static void extra_widget_foreach_hide (GtkWidget *widget, gpointer data) { if (widget != data) gtk_widget_hide (widget); } static void on_type_combobox_changed (GtkComboBox *combobox, gpointer data) { FormatHandler *handler = NULL; GtkWidget *extra_widget = data; GtkTreeIter iter; GtkTreeModel *model = gtk_combo_box_get_model (combobox); gtk_container_foreach ( GTK_CONTAINER (extra_widget), extra_widget_foreach_hide, g_object_get_data (G_OBJECT (combobox), "format-box")); if (!gtk_combo_box_get_active_iter (combobox, &iter)) return; gtk_tree_model_get ( model, &iter, DEST_HANDLER, &handler, -1); if (handler && handler->options_widget) { gtk_widget_show (handler->options_widget); } } static void format_handlers_foreach_free (gpointer data, gpointer user_data) { FormatHandler *handler = data; if (handler->options_widget) gtk_widget_destroy (handler->options_widget); if (handler->data) g_free (handler->data); g_free (data); } static void ask_destination_and_save (ESourceSelector *selector, ECalClientSourceType type) { FormatHandler *handler = NULL; GtkWidget *extra_widget = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); GtkWidget *hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); GtkLabel *label = GTK_LABEL (gtk_label_new_with_mnemonic (_("_Format:"))); GtkComboBox *combo = GTK_COMBO_BOX (gtk_combo_box_new ()); GtkTreeModel *model = GTK_TREE_MODEL (gtk_list_store_new (N_DEST_COLUMNS, G_TYPE_STRING, G_TYPE_POINTER)); GtkCellRenderer *renderer = NULL; GtkListStore *store = GTK_LIST_STORE (model); GtkTreeIter iter; GtkWidget *dialog = NULL; gchar *dest_uri = NULL; GList *format_handlers = NULL; /* The available formathandlers */ format_handlers = g_list_append (format_handlers, ical_format_handler_new ()); format_handlers = g_list_append (format_handlers, csv_format_handler_new ()); format_handlers = g_list_append (format_handlers, rdf_format_handler_new ()); gtk_box_pack_start (GTK_BOX (extra_widget), GTK_WIDGET (hbox), FALSE, FALSE, 0); gtk_label_set_mnemonic_widget (label, GTK_WIDGET (combo)); gtk_box_pack_start (GTK_BOX (hbox), GTK_WIDGET (label), FALSE, FALSE, 0); /* The Type GtkComboBox */ gtk_box_pack_start (GTK_BOX (hbox), GTK_WIDGET (combo), TRUE, TRUE, 0); gtk_combo_box_set_model (combo, model); gtk_list_store_clear (store); renderer = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), renderer, TRUE); gtk_cell_layout_set_attributes ( GTK_CELL_LAYOUT (combo), renderer, "text", DEST_NAME_COLUMN, NULL); while (format_handlers) { handler = format_handlers->data; gtk_list_store_append (store, &iter); gtk_list_store_set ( store, &iter, DEST_NAME_COLUMN, handler->combo_label, -1); gtk_list_store_set (store, &iter, DEST_HANDLER, handler, -1); if (handler->options_widget) { gtk_box_pack_start ( GTK_BOX (extra_widget), GTK_WIDGET (handler->options_widget), TRUE, TRUE, 0); gtk_widget_hide (handler->options_widget); } if (handler->isdefault) { gtk_combo_box_set_active_iter (combo, &iter); if (handler->options_widget) gtk_widget_show (handler->options_widget); } format_handlers = g_list_next (format_handlers); } g_signal_connect ( combo, "changed", G_CALLBACK (on_type_combobox_changed), extra_widget); g_object_set_data (G_OBJECT (combo), "format-box", hbox); dialog = gtk_file_chooser_dialog_new ( _("Select destination file"), NULL, GTK_FILE_CHOOSER_ACTION_SAVE, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_SAVE_AS, GTK_RESPONSE_OK, NULL); gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_OK); gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER (dialog), extra_widget); gtk_file_chooser_set_local_only (GTK_FILE_CHOOSER (dialog), FALSE); gtk_widget_show (hbox); gtk_widget_show (GTK_WIDGET (label)); gtk_widget_show (GTK_WIDGET (combo)); gtk_widget_show (extra_widget); if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_OK) { gchar *tmp = NULL; if (gtk_combo_box_get_active_iter (combo, &iter)) gtk_tree_model_get ( model, &iter, DEST_HANDLER, &handler, -1); else handler = NULL; dest_uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (dialog)); if (handler) { tmp = strstr (dest_uri, handler->filename_ext); if (!(tmp && *(tmp + strlen (handler->filename_ext)) == '\0')) { gchar *temp; temp = g_strconcat (dest_uri, handler->filename_ext, NULL); g_free (dest_uri); dest_uri = temp; } handler->save (handler, selector, type, dest_uri); } else { g_warn_if_reached (); } } /* Free the handlers */ g_list_foreach (format_handlers, format_handlers_foreach_free, NULL); g_list_free (format_handlers); /* Now we can destroy it */ gtk_widget_destroy (dialog); g_free (dest_uri); } /* Returns output stream for the uri, or NULL on any error. * When done with the stream, just g_output_stream_close and g_object_unref it. * It will ask for overwrite if file already exists. */ GOutputStream * open_for_writing (GtkWindow *parent, const gchar *uri, GError **error) { GFile *file; GFileOutputStream *fostream; GError *err = NULL; g_return_val_if_fail (uri != NULL, NULL); file = g_file_new_for_uri (uri); g_return_val_if_fail (file != NULL, NULL); fostream = g_file_create (file, G_FILE_CREATE_NONE, NULL, &err); if (err && err->code == G_IO_ERROR_EXISTS) { gint response; g_clear_error (&err); response = e_alert_run_dialog_for_args ( parent, E_ALERT_ASK_FILE_EXISTS_OVERWRITE, uri, NULL); if (response == GTK_RESPONSE_OK) { fostream = g_file_replace ( file, NULL, FALSE, G_FILE_CREATE_NONE, NULL, &err); if (err && fostream) { g_object_unref (fostream); fostream = NULL; } } else if (fostream) { g_object_unref (fostream); fostream = NULL; } } g_object_unref (file); if (error && err) *error = err; else if (err) g_error_free (err); if (fostream) return G_OUTPUT_STREAM (fostream); return NULL; } static void save_general (EShellView *shell_view, ECalClientSourceType type) { EShellSidebar *shell_sidebar; ESourceSelector *selector = NULL; shell_sidebar = e_shell_view_get_shell_sidebar (shell_view); g_object_get (shell_sidebar, "selector", &selector, NULL); g_return_if_fail (selector != NULL); ask_destination_and_save (selector, type); g_object_unref (selector); } static void action_calendar_save_as_cb (GtkAction *action, EShellView *shell_view) { save_general (shell_view, E_CAL_CLIENT_SOURCE_TYPE_EVENTS); } static void action_memo_list_save_as_cb (GtkAction *action, EShellView *shell_view) { save_general (shell_view, E_CAL_CLIENT_SOURCE_TYPE_MEMOS); } static void action_task_list_save_as_cb (GtkAction *action, EShellView *shell_view) { save_general (shell_view, E_CAL_CLIENT_SOURCE_TYPE_TASKS); } gboolean calendar_save_as_init (GtkUIManager *ui_manager, EShellView *shell_view) { EShellWindow *shell_window; GtkActionGroup *action_group; GtkAction *action; const gchar *tooltip; const gchar *stock_id; const gchar *name; shell_window = e_shell_view_get_shell_window (shell_view); name = "calendar-save-as"; tooltip = _("Save the selected calendar to disk"); stock_id = GTK_STOCK_SAVE_AS; action = gtk_action_new (name, NULL, tooltip, stock_id); name = "lockdown-save-to-disk"; action_group = e_shell_window_get_action_group (shell_window, name); gtk_action_group_add_action (action_group, action); g_signal_connect ( action, "activate", G_CALLBACK (action_calendar_save_as_cb), shell_view); g_object_unref (action); return TRUE; } gboolean memo_list_save_as_init (GtkUIManager *ui_manager, EShellView *shell_view) { EShellWindow *shell_window; GtkActionGroup *action_group; GtkAction *action; const gchar *tooltip; const gchar *stock_id; const gchar *name; shell_window = e_shell_view_get_shell_window (shell_view); name = "memo-list-save-as"; tooltip = _("Save the selected memo list to disk"); stock_id = GTK_STOCK_SAVE_AS; action = gtk_action_new (name, NULL, tooltip, stock_id); name = "lockdown-save-to-disk"; action_group = e_shell_window_get_action_group (shell_window, name); gtk_action_group_add_action (action_group, action); g_signal_connect ( action, "activate", G_CALLBACK (action_memo_list_save_as_cb), shell_view); g_object_unref (action); return TRUE; } gboolean task_list_save_as_init (GtkUIManager *ui_manager, EShellView *shell_view) { EShellWindow *shell_window; GtkActionGroup *action_group; GtkAction *action; const gchar *tooltip; const gchar *stock_id; const gchar *name; shell_window = e_shell_view_get_shell_window (shell_view); name = "task-list-save-as"; tooltip = _("Save the selected task list to disk"); stock_id = GTK_STOCK_SAVE_AS; action = gtk_action_new (name, NULL, tooltip, stock_id); name = "lockdown-save-to-disk"; action_group = e_shell_window_get_action_group (shell_window, name); gtk_action_group_add_action (action_group, action); g_signal_connect ( action, "activate", G_CALLBACK (action_task_list_save_as_cb), shell_view); g_object_unref (action); return TRUE; }