aboutsummaryrefslogblamecommitdiffstats
path: root/e-util/e-dialog-utils.c
blob: 9e434cde675975a5f1e3a249eb84ed85fd4797f5 (plain) (tree)
1
2
3
4
5
6
7
8
9
  
                                                                


                                                               



                                                                    


                                                                   
                                                                             
  

           
                                                  
                                            


                                                        

   



                    

                           
                  
                   

                        
                        
                     
      
 
                               
                       
 


                           











                                                                      
                                                                         


                          
                   










                                                                        
 
                   
                                                                           



                                             
           


                                          

                                
 
                                              
                                                               
 

                                                                                                             
                                
         
 


                         


                                                                     
                                           



                                        
 

                                                           
 
                             
                               
 
                                                                                              
 
                                                                                                          
 

                                                 
                    
 
                                                    
 


                        



                           
                                                                       





                                                                                
                                                                                                               

                           
                   
 






                                                                        


                                                                                
                                      
 

                                                 
 
                            

                                                          
 













                                                                               
                                                    

                       
                    








                                                                   
                                            

                             
 





















                                                                                                                          
                                      
 
                   
 
                                            






                            
/*
 * 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:
 *      Michael Meeks <michael@ximian.com>
 *  Ettore Perazzoli <ettore@ximian.com>
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include "e-dialog-utils.h"

#include <errno.h>
#include <unistd.h>
#include <glib/gstdio.h>

#ifdef GDK_WINDOWING_X11
#include <gdk/gdkx.h>
#endif

#include <gconf/gconf-client.h>
#include <glib/gi18n.h>

#include "e-util/e-util.h"
#include "e-util/e-error.h"


/**
 * e_notice:
 * @parent: the dialog's parent window, or %NULL
 * @type: the type of dialog (%GTK_MESSAGE_INFO, %GTK_MESSAGE_WARNING,
 * or %GTK_MESSAGE_ERROR)
 * @format: printf-style format string, followed by arguments
 *
 * Convenience function to show a dialog with a message and an "OK"
 * button.
 **/
void
e_notice (gpointer parent, GtkMessageType type, const gchar *format, ...)
{
    GtkWidget *dialog;
    va_list args;
    gchar *str;

    va_start (args, format);
    str = g_strdup_vprintf (format, args);
    dialog = gtk_message_dialog_new (NULL,
                     GTK_DIALOG_DESTROY_WITH_PARENT,
                     type,
                     GTK_BUTTONS_OK,
                     "%s",
                     str);
    va_end (args);
    g_free (str);

    if (parent)
        gtk_window_set_transient_for (GTK_WINDOW (dialog), parent);
    gtk_dialog_run (GTK_DIALOG (dialog));
    gtk_widget_destroy (dialog);
}

static void
save_ok (GtkWidget *widget, gpointer data)
{
    GtkWidget *fs;
    gchar **filename = data;
    gchar *uri;

    fs = gtk_widget_get_toplevel (widget);
    uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (fs));

    if (e_file_can_save((GtkWindow *)widget, uri)) {
        e_file_update_save_path(gtk_file_chooser_get_current_folder_uri(GTK_FILE_CHOOSER(fs)), TRUE);
        *filename = uri;
    }

    gtk_main_quit ();
}

static void
filechooser_response (GtkWidget *fc, gint response_id, gpointer data)
{
    if (response_id == GTK_RESPONSE_OK)
        save_ok (fc, data);
    else
        gtk_widget_destroy (fc);
}

gchar *
e_file_dialog_save (const gchar *title, const gchar *fname)
{
    GtkWidget *selection;
    gchar *filename = NULL;

    selection = e_file_get_save_filesel(NULL, title, fname, GTK_FILE_CHOOSER_ACTION_SAVE);

    g_signal_connect (G_OBJECT (selection), "response", G_CALLBACK (filechooser_response), &filename);

    gtk_widget_show (GTK_WIDGET (selection));
    gtk_grab_add (GTK_WIDGET (selection));
    gtk_main ();

    gtk_widget_destroy (GTK_WIDGET (selection));

    return filename;
}

/**
 * e_file_get_save_filesel:
 * @parent: parent window
 * @title: dialog title
 * @name: filename; already in a proper form (suitable for file system)
 * @action: action for dialog
 *
 * Creates a save dialog, using the saved directory from gconf.   The dialog has
 * no signals connected and is not shown.
 **/
GtkWidget *
e_file_get_save_filesel (GtkWindow *parent, const gchar *title, const gchar *name, GtkFileChooserAction action)
{
    GtkWidget *filesel;
    gchar *uri;

    g_return_val_if_fail (GTK_IS_WINDOW (parent), NULL);

    filesel = gtk_file_chooser_dialog_new (
        title, parent, action,
        GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
        (action == GTK_FILE_CHOOSER_ACTION_OPEN) ?
        GTK_STOCK_OPEN : GTK_STOCK_SAVE, GTK_RESPONSE_OK, NULL);
    gtk_dialog_set_default_response (GTK_DIALOG (filesel), GTK_RESPONSE_OK);
    gtk_file_chooser_set_local_only (GTK_FILE_CHOOSER (filesel), FALSE);

    uri = e_file_get_save_path ();

    gtk_file_chooser_set_current_folder_uri (
        GTK_FILE_CHOOSER (filesel), uri);

    if (name && name[0])
        gtk_file_chooser_set_current_name (
            GTK_FILE_CHOOSER (filesel), name);

    g_free (uri);

    return filesel;
}

/**
 * e_file_can_save:
 *
 * Return TRUE if the URI can be saved to, FALSE otherwise.  It checks local
 * files to see if they're regular and can be accessed.  If the file exists and
 * is writable, it pops up a dialog asking the user if they want to overwrite
 * it.  Returns the users choice.
 **/
gboolean
e_file_can_save(GtkWindow *parent, const gchar *uri)
{
    struct stat st;
    gchar *path;
    gboolean res;

    if (!uri || uri[0] == 0)
        return FALSE;

    /* Assume remote files are writable; too costly to check */
    if (!e_file_check_local(uri))
        return TRUE;

    path = e_util_uri_to_filename (uri);
    if (!path)
        return FALSE;

    /* make sure we can actually save to it... */
    if (g_stat (path, &st) != -1 && !S_ISREG (st.st_mode)) {
        g_free(path);
        return FALSE;
    }

    res = TRUE;
    if (g_access (path, F_OK) == 0) {
        if (g_access (path, W_OK) != 0) { e_error_run(parent, "mail:no-save-path", path, g_strerror(errno), NULL);
            g_free(path);
            return FALSE;
        }

        res = e_error_run(parent, E_ERROR_ASK_FILE_EXISTS_OVERWRITE, path, NULL) == GTK_RESPONSE_OK;

    }

    g_free(path);
    return res;
}

gboolean
e_file_check_local (const gchar *name)
{
    gchar *uri;

    uri = e_util_uri_to_filename (name);
    if (uri) {
        g_free(uri);
        return TRUE;
    }

    return FALSE;
}