2008-04-17 Milan Crha <mcrha@redhat.com> ** Fix for bug #526741 (gnome-vfs to gio/gvfs port) * configure.in: Drop dependency on gnome-vfs, depend on gio instead. * addressbook/gui/contact-editor/e-contact-editor.c: Do not include gnome-vfs. * calendar/gui/e-meeting-store.c: (start_async_read), (freebusy_async), (async_read): Read data with gio, not with gnome-vfs. * calendar/gui/e-meeting-store.c: (async_close): Function removed. * calendar/gui/migration.c: * calendar/gui/e-memos.c: * calendar/gui/alarm-notify/notify-main.c: * calendar/gui/e-tasks.c: * calendar/gui/main.c: * calendar/gui/e-meeting-list-view.c: Do not include gnome-vfs. * calendar/gui/dialogs/comp-editor.c: (set_attachment_list): * calendar/gui/dialogs/comp-editor.c: (attachment_guess_mime_type): Function removed, use e-util function instead. * calendar/gui/e-cal-popup.c: (temp_save_part): Reflect changes in CamelStreamVFS. * calendar/gui/e-cal-popup.c: (ecalp_apps_open_in), (ecalp_apps_popup_free), (ecalp_standard_menu_factory): Use gio to fill popup and launch app. * calendar/gui/e-cal-component-preview.h: (e_cal_comp_preview_url_requested_cb): * calendar/gui/e-cal-component-preview.c: (e_cal_comp_preview_url_requested_cb): New helper function to manage file:// urls for GtkHTML component. * calendar/gui/e-cal-component-preview.c: (e_cal_component_preview_init): * calendar/gui/e-cal-component-memo-preview.c: (e_cal_comp_preview_url_requested_cb): Use above helper function. * calendar/gui/e-cal-component-memo-preview.c: (url_requested_cb): * calendar/gui/e-cal-component-preview.c: (url_requested_cb): Function removed. * composer/e-msg-composer.h: (e_msg_composer_guess_mime_type): * composer/e-msg-composer.c: (e_msg_composer_guess_mime_type), (handle_uri), (e_msg_composer_add_inline_image_from_file): Use shared e_util_guess_mime_type instead of its own. * e-util/e-gui-utils.c: Do not include gnome-vfs. * e-util/e-util.c: (e_write_file_uri), (e_file_get_save_path): * e-util/e-dialog-utils.c: (e_file_can_save), (e_file_check_local): Use gio instead of gnome-vfs. * e-util/e-util.h: * e-util/e-util.c: (e_util_guess_mime_type), (e_util_filename_to_uri), (e_util_uri_to_filename), (e_util_read_file): New helper functions. * mail/Makefile.am: * mail/em-vfs-stream.h: * mail/em-vfs-stream.c: Removed from project, not used anywhere. * mail/em-menu.c: * mail/em-config.c: Do not include gnome-vfs. * mail/mail-ops.c: (save_messages_exec), (save_part_exec): Changes related to Data Server change of CamelStreamVFS. * mail/em-format.c: (em_format_describe_part): * mail/em-format-html-display.c: (efhd_bonobo_object), (efhd_find_handler), (efhd_use_component), (efhd_bonobo_unknown), (efhd_check_server_prop), (efhd_format_attachment): Drop use of deprecated gnome-vfs functions. * mail/em-utils.c: (em_utils_snoop_type): Use gio to guess the mime_type. * mail/em-popup.c: (emp_apps_open_in), (emp_apps_popup_free), (emp_standard_menu_factory): Use gio to fill popup and launch app. * plugins/import-ics-attachments/icsimporter.c: Do not include gnome-vfs. * plugins/ipod-sync/ical-format.c: (do_save_calendar_ical): * plugins/ipod-sync/ipod-sync.c: (destination_save_addressbook): Write to gio GOutputStream instead of gnome-vfs handle. * plugins/ipod-sync/format-handler.h: (open_for_writing): * plugins/ipod-sync/ipod-sync.c: (open_for_writing): New helper function. * plugins/publish-calendar/publish-format-fb.h: (publish_calendar_as_fb): * plugins/publish-calendar/publish-format-fb.c: (write_calendar), (publish_calendar_as_fb): * plugins/publish-calendar/publish-format-ical.h: (publish_calendar_as_ical): * plugins/publish-calendar/publish-format-ical.c: (write_calendar), (publish_calendar_as_ical): Use gio GOutputStream instead of gnome-vfs handle. * plugins/publish-calendar/url-editor-dialog.c: (create_uri): Use glib function to escape URI. * plugins/publish-calendar/publish-location.c: (migrateURI): Use EUri to parse URI. * plugins/publish-calendar/publish-calendar.c: (publish_online), (unmount_done_cb), (struct mnt_struct), (mount_ready_cb), (ask_password), (ask_question), (mount_first), (publish): Use gio instead of gnome-vfs for opening (remote) files. * plugins/save-calendar/ical-format.c: (do_save_calendar_ical): * plugins/save-calendar/csv-format.c: (do_save_calendar_csv): * plugins/save-calendar/rdf-format.c: (do_save_calendar_rdf): Use gio GOutpuStream instead of gnome-vfs handle. * plugins/save-calendar/format-handler.h: (open_for_writing): * plugins/save-calendar/save-calendar.c: (open_for_writing): New helper function. * shell/e-shell-window-commands.c: (command_quick_reference): Use gio instead of gnome-vfs. * widgets/misc/e-attachment.h: (struct _EAttachment): * widgets/misc/e-attachment.c: (finalise), (init), (attachment_guess_mime_type), (e_attachment_new), (struct DownloadInfo), (download_info_free), (data_ready_cb), (download_to_local_path), (e_attachment_new_remote_file): Use gio instead of gnome-vfs to download remote files. * widgets/misc/e-image-chooser.c: (image_drag_data_received_cb): Use new util function to read file. * widgets/misc/e-attachment-bar.c: (size_to_string): Stolen from gnome-vfs. svn path=/trunk/; revision=35378
447 lines
11 KiB
C
447 lines
11 KiB
C
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
|
||
/* e-dialog-utils.h
|
||
*
|
||
* Copyright (C) 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., 51 Franklin Street, Fifth Floor,
|
||
* Boston, MA 02110-1301, USA.
|
||
*
|
||
* Authors:
|
||
* Michael Meeks <michael@ximian.com>
|
||
* Ettore Perazzoli <ettore@ximian.com>
|
||
*/
|
||
|
||
#ifdef HAVE_CONFIG_H
|
||
#include <config.h>
|
||
#endif
|
||
|
||
#include "e-dialog-utils.h"
|
||
|
||
#include <unistd.h>
|
||
#include <glib.h>
|
||
#include <glib/gstdio.h>
|
||
|
||
#include <gdkconfig.h>
|
||
#ifdef GDK_WINDOWING_X11
|
||
#include <gdk/gdkx.h>
|
||
#endif
|
||
|
||
#include <gtk/gtkmain.h>
|
||
#include <gtk/gtkplug.h>
|
||
#include <gtk/gtkversion.h>
|
||
|
||
#include <gtk/gtkfilechooser.h>
|
||
#include <gtk/gtkfilechooserdialog.h>
|
||
#include <gtk/gtkstock.h>
|
||
|
||
#include <gconf/gconf-client.h>
|
||
#include <glib/gi18n.h>
|
||
#include <libgnome/gnome-util.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 char *format, ...)
|
||
{
|
||
GtkWidget *dialog;
|
||
va_list args;
|
||
char *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)
|
||
e_dialog_set_transient_for (GTK_WINDOW (dialog), parent);
|
||
gtk_dialog_run (GTK_DIALOG (dialog));
|
||
gtk_widget_destroy (dialog);
|
||
}
|
||
|
||
/**
|
||
* e_notice_with_xid:
|
||
* @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
|
||
*
|
||
* Like e_notice(), but takes a GdkNativeWindow for the parent
|
||
* window argument.
|
||
**/
|
||
void
|
||
e_notice_with_xid (GdkNativeWindow parent, GtkMessageType type, const char *format, ...)
|
||
{
|
||
GtkWidget *dialog;
|
||
va_list args;
|
||
char *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);
|
||
gtk_dialog_set_has_separator (GTK_DIALOG (dialog), FALSE);
|
||
va_end (args);
|
||
g_free (str);
|
||
|
||
if (parent)
|
||
e_dialog_set_transient_for_xid (GTK_WINDOW (dialog), parent);
|
||
gtk_dialog_run (GTK_DIALOG (dialog));
|
||
gtk_widget_destroy (dialog);
|
||
}
|
||
|
||
|
||
#ifdef GDK_WINDOWING_X11
|
||
/* Tests whether or not an X Window is being managed by the
|
||
* window manager.
|
||
*/
|
||
static gboolean
|
||
window_is_wm_toplevel (Display *display, Window window)
|
||
{
|
||
static Atom WM_STATE = None;
|
||
unsigned long nitems, after;
|
||
unsigned char *data = NULL;
|
||
Atom type = None;
|
||
int format;
|
||
|
||
if (!WM_STATE)
|
||
WM_STATE = XInternAtom (display, "WM_STATE", False);
|
||
|
||
if (XGetWindowProperty (display, window, WM_STATE, 0, 0, False,
|
||
AnyPropertyType, &type, &format,
|
||
&nitems, &after, &data) == Success) {
|
||
if (data)
|
||
XFree((char*)data);
|
||
if (type)
|
||
return TRUE;
|
||
}
|
||
return FALSE;
|
||
}
|
||
|
||
#endif
|
||
|
||
/**
|
||
* e_dialog_set_transient_for:
|
||
* @dialog: a dialog window
|
||
* @parent_widget: the parent for @dialog
|
||
*
|
||
* This sets the parent for @dialog to be @parent_widget. Unlike
|
||
* gtk_window_set_transient_for(), this doesn't need @parent_widget to
|
||
* be the actual toplevel, and also works if @parent_widget is
|
||
* embedded as a Bonobo control by an out-of-process container.
|
||
* @parent_widget must already be realized before calling this
|
||
* function, but @dialog does not need to be.
|
||
**/
|
||
void
|
||
e_dialog_set_transient_for (GtkWindow *dialog,
|
||
GtkWidget *parent_widget)
|
||
{
|
||
GtkWidget *toplevel;
|
||
#ifdef GDK_WINDOWING_X11
|
||
Window parent, root_ret, *children;
|
||
unsigned int numchildren;
|
||
Display *display;
|
||
Status status;
|
||
#endif
|
||
g_return_if_fail (GTK_IS_WINDOW (dialog));
|
||
g_return_if_fail (GTK_IS_WIDGET (parent_widget));
|
||
|
||
toplevel = gtk_widget_get_toplevel (parent_widget);
|
||
if (toplevel == NULL)
|
||
return;
|
||
|
||
if (!GTK_IS_PLUG (toplevel)) {
|
||
gtk_window_set_transient_for (GTK_WINDOW (dialog),
|
||
GTK_WINDOW (toplevel));
|
||
return;
|
||
}
|
||
#ifdef GDK_WINDOWING_X11
|
||
/* Find the top-level windowmanager-managed X Window */
|
||
display = GDK_WINDOW_XDISPLAY (parent_widget->window);
|
||
parent = GDK_WINDOW_XID (parent_widget->window);
|
||
|
||
while (parent && !window_is_wm_toplevel (display, parent)) {
|
||
status = XQueryTree (display, parent, &root_ret,
|
||
&parent, &children, &numchildren);
|
||
if (status != 0)
|
||
XFree (children);
|
||
}
|
||
|
||
e_dialog_set_transient_for_xid (dialog, parent);
|
||
#endif
|
||
#ifdef GDK_WINDOWING_WIN32
|
||
g_warning ("Not implemented on Win32: e_dialog_set_transient_for() for plug windows");
|
||
#endif
|
||
}
|
||
|
||
static void
|
||
dialog_realized (GtkWindow *dialog, gpointer xid)
|
||
{
|
||
e_dialog_set_transient_for_xid (dialog, (GdkNativeWindow)GPOINTER_TO_INT(xid));
|
||
}
|
||
|
||
/**
|
||
* e_dialog_set_transient_for_xid:
|
||
* @dialog: a dialog window
|
||
* @xid: the X Window parent
|
||
*
|
||
* Like e_dialog_set_transient_for(), but use an XID to specify the
|
||
* parent window.
|
||
**/
|
||
void
|
||
e_dialog_set_transient_for_xid (GtkWindow *dialog,
|
||
GdkNativeWindow xid)
|
||
{
|
||
#ifdef GDK_MULTIHEAD_SAFE
|
||
GdkDisplay *display;
|
||
#endif
|
||
GdkWindow *parent;
|
||
|
||
g_return_if_fail (GTK_IS_WINDOW (dialog));
|
||
|
||
if (!GTK_WIDGET_REALIZED (dialog)) {
|
||
g_signal_connect (dialog, "realize",
|
||
G_CALLBACK (dialog_realized),
|
||
GINT_TO_POINTER(xid));
|
||
return;
|
||
}
|
||
|
||
#ifdef GDK_MULTIHEAD_SAFE
|
||
display = gdk_drawable_get_display (GDK_DRAWABLE (GTK_WIDGET (dialog)->window));
|
||
parent = gdk_window_lookup_for_display (display, xid);
|
||
if (!parent)
|
||
parent = gdk_window_foreign_new_for_display (display, xid);
|
||
#else
|
||
parent = gdk_window_lookup (xid);
|
||
if (!parent)
|
||
parent = gdk_window_foreign_new (xid);
|
||
#endif
|
||
g_return_if_fail (parent != NULL);
|
||
|
||
gdk_window_set_transient_for (GTK_WIDGET (dialog)->window, parent);
|
||
}
|
||
|
||
|
||
|
||
static void
|
||
save_ok (GtkWidget *widget, gpointer data)
|
||
{
|
||
GtkWidget *fs;
|
||
char **filename = data;
|
||
char *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);
|
||
}
|
||
|
||
char *
|
||
e_file_dialog_save (const char *title, const char *fname)
|
||
{
|
||
GtkWidget *selection;
|
||
char *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;
|
||
}
|
||
|
||
static void
|
||
save_folder_ok (GtkWidget *widget, gpointer data)
|
||
{
|
||
GtkWidget *fs;
|
||
char **filename = data;
|
||
char *uri;
|
||
|
||
fs = gtk_widget_get_toplevel (widget);
|
||
uri = gtk_file_chooser_get_current_folder_uri (GTK_FILE_CHOOSER (fs));
|
||
|
||
e_file_update_save_path(uri, FALSE);
|
||
*filename = uri;
|
||
|
||
gtk_main_quit ();
|
||
}
|
||
|
||
static void
|
||
folderchooser_response (GtkWidget *fc, gint response_id, gpointer data)
|
||
{
|
||
if (response_id == GTK_RESPONSE_OK)
|
||
save_folder_ok (fc, data);
|
||
else
|
||
gtk_widget_destroy (fc);
|
||
}
|
||
|
||
char *
|
||
e_file_dialog_save_folder (const char *title)
|
||
{
|
||
GtkWidget *selection;
|
||
char *filename = NULL;
|
||
|
||
selection = e_file_get_save_filesel(NULL, title, NULL, GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER);
|
||
g_signal_connect (G_OBJECT (selection), "response", G_CALLBACK (folderchooser_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 (GtkWidget *parent, const char *title, const char *name, GtkFileChooserAction action)
|
||
{
|
||
GtkWidget *filesel;
|
||
char *uri;
|
||
|
||
filesel = gtk_file_chooser_dialog_new (title,
|
||
NULL,
|
||
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);
|
||
|
||
if (parent)
|
||
e_dialog_set_transient_for((GtkWindow *)filesel, parent);
|
||
|
||
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 char *uri)
|
||
{
|
||
struct stat st;
|
||
char *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 char *name)
|
||
{
|
||
char *uri;
|
||
|
||
uri = e_util_uri_to_filename (name);
|
||
if (uri) {
|
||
g_free(uri);
|
||
return TRUE;
|
||
}
|
||
|
||
return FALSE;
|
||
}
|