aboutsummaryrefslogblamecommitdiffstats
path: root/plugins/save-calendar/save-calendar.c
blob: 119009f69ea0c9a90698b20b4f94f844cc0fb859 (plain) (tree)
1
2
3
4
5
6
7
8
9
10




                                                                
  



                                                                    
  
                                                                   
                                                                             
  
  

                                                 
  
                                                        









                                                                      
                    
                    
                       



                                               
                           
                   
 
                           






                              
 

                                                                          
                                                                       
 
           
                                                            
 

                                         

 
           

                                                               

                                       
                         

                                                                 
                                                            

                                                     

                                                        
                                         

                                            


                                                          
         
 

 
           
                                                                
 
                                      
 

                                                             
 

                                       
 
                      

 
           

                                                                                          
                                      


                                                                 
                                                                
                                                                 



                                                     
                               



                                          
                                                         
                                            
                                                         
                                           
                                                         

                                           
                                  
                                                                       
                               
                                               

                                     

                                                                             
                                                                


                                                                  
                                                
                                                     
                                                                   



                                                                             
                                                                   











                                                                                     
 
                                                     
                                                                     
 




                                                                                    
                                                    
 

                                                                                    
                                                                           

                                            
 
                                                                      
                                  
 
                                                             
                                                 
                                                    
 
                                                  
                                                    
 
                                                               
 
                                                                                
 
                                    


                                                                                   
                 
 

                                                                    
 


                                                                             
 
                                   
                                    
                          
 

 




                                                                               
                                                                      









































                                                                                                            


                                                                    
                                                                       




                                                                 
                                                                      
 





                                                                         
/*
 * 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 <http://www.gnu.org/licenses/>
 *
 *
 * Authors:
 *      Rodrigo Moya <rodrigo@novell.com>
 *
 * 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 <config.h>
#endif

#include <gio/gio.h>
#include <gtk/gtk.h>
#include <glib/gi18n.h>
#include <libedataserver/e-source.h>
#include <libedataserverui/e-source-selector.h>
#include <libecal/e-cal.h>
#include <calendar/gui/e-cal-popup.h>
#include <e-util/e-error.h>
#include <string.h>

#include "format-handler.h"

enum {  /* GtkComboBox enum */
    DEST_NAME_COLUMN,
    DEST_HANDLER,
    N_DEST_COLUMNS

};

void org_gnome_save_calendar (EPlugin *ep, ECalPopupTargetSource *target);
void org_gnome_save_tasks (EPlugin *ep, ECalPopupTargetSource *target);
void org_gnome_save_memos (EPlugin *ep, ECalPopupTargetSource *target);

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, combobox);

    gtk_combo_box_get_active_iter (combobox, &iter);

    gtk_tree_model_get (model, &iter,
        DEST_HANDLER, &handler, -1);

    if (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 (EPlugin *ep, ECalPopupTargetSource *target, ECalSourceType type)
{
    FormatHandler *handler = NULL;

    GtkWidget *extra_widget = gtk_vbox_new (FALSE, 0);
    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 ());

    /* The Type GtkComboBox */
    gtk_box_pack_start (GTK_BOX (extra_widget), 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 (G_OBJECT(combo), "changed",
        G_CALLBACK (on_type_combobox_changed), extra_widget);

    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 (GTK_WIDGET(combo));
    gtk_widget_show (extra_widget);

    if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_OK) {
        gchar *tmp = NULL;

        gtk_combo_box_get_active_iter (combo, &iter);
        gtk_tree_model_get (model, &iter,
            DEST_HANDLER, &handler, -1);

           dest_uri = gtk_file_chooser_get_uri
            (GTK_FILE_CHOOSER (dialog));

        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, ep, target, type, dest_uri);
    }

    /* 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) {
        g_clear_error (&err);

        if (e_error_run (parent, E_ERROR_ASK_FILE_EXISTS_OVERWRITE, uri, NULL) == 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;
}

void
org_gnome_save_calendar (EPlugin *ep, ECalPopupTargetSource *target)
{
    ask_destination_and_save (ep, target, E_CAL_SOURCE_TYPE_EVENT);
}

void
org_gnome_save_tasks (EPlugin *ep, ECalPopupTargetSource *target)
{
    ask_destination_and_save (ep, target, E_CAL_SOURCE_TYPE_TODO);
}

void
org_gnome_save_memos (EPlugin *ep, ECalPopupTargetSource *target)
{
    ask_destination_and_save (ep, target, E_CAL_SOURCE_TYPE_JOURNAL);
}