medit/moo/mooutils/moofiledialog.c

648 lines
20 KiB
C

/*
* mooutils/moofiledialog.c
*
* Copyright (C) 2004-2006 by Yevgen Muntyan <muntyan@math.tamu.edu>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* See COPYING file that comes with this distribution.
*/
#include <gtk/gtk.h>
#include "mooutils/moodialogs.h"
#include "mooutils/mooprefs.h"
#include "mooutils/mooutils-misc.h"
#include "mooutils/moocompat.h"
struct _MooFileDialogPrivate {
gboolean multiple;
char *title;
char *dir;
char *name;
MooFileDialogType type;
GtkWidget *parent;
MooFilterMgr *filter_mgr;
char *filter_mgr_id;
GSList *filenames;
char *filename;
};
G_DEFINE_TYPE (MooFileDialog, moo_file_dialog, G_TYPE_OBJECT)
enum {
PROP_0,
PROP_MULTIPLE,
PROP_TITLE,
PROP_DIR,
PROP_NAME,
PROP_TYPE,
PROP_PARENT,
PROP_FILTER_MGR,
PROP_FILTER_MGR_ID
};
static void
moo_file_dialog_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
MooFileDialog *dialog = MOO_FILE_DIALOG (object);
switch (prop_id)
{
case PROP_MULTIPLE:
dialog->priv->multiple = g_value_get_boolean (value) != 0;
g_object_notify (object, "multiple");
break;
case PROP_TITLE:
g_free (dialog->priv->title);
dialog->priv->title = g_value_dup_string (value);
g_object_notify (object, "title");
break;
case PROP_DIR:
g_free (dialog->priv->dir);
dialog->priv->dir = g_value_dup_string (value);
g_object_notify (object, "dir");
break;
case PROP_NAME:
g_free (dialog->priv->name);
dialog->priv->name = g_value_dup_string (value);
g_object_notify (object, "name");
break;
case PROP_TYPE:
dialog->priv->type = g_value_get_enum (value);
g_object_notify (object, "type");
break;
case PROP_PARENT:
dialog->priv->parent = g_value_get_object (value);
g_object_notify (object, "parent");
break;
case PROP_FILTER_MGR:
if (dialog->priv->filter_mgr)
g_object_unref (dialog->priv->filter_mgr);
dialog->priv->filter_mgr = g_value_get_object (value);
if (dialog->priv->filter_mgr)
g_object_ref (dialog->priv->filter_mgr);
g_object_notify (object, "filter-mgr");
break;
case PROP_FILTER_MGR_ID:
g_free (dialog->priv->filter_mgr_id);
dialog->priv->filter_mgr_id = g_value_dup_string (value);
g_object_notify (object, "filter-mgr-id");
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
moo_file_dialog_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
MooFileDialog *dialog = MOO_FILE_DIALOG (object);
switch (prop_id)
{
case PROP_MULTIPLE:
g_value_set_boolean (value, dialog->priv->multiple != 0);
break;
case PROP_TITLE:
g_value_set_string (value, dialog->priv->title);
break;
case PROP_DIR:
g_value_set_string (value, dialog->priv->dir);
break;
case PROP_NAME:
g_value_set_string (value, dialog->priv->name);
break;
case PROP_TYPE:
g_value_set_enum (value, dialog->priv->type);
break;
case PROP_PARENT:
g_value_set_object (value, dialog->priv->parent);
break;
case PROP_FILTER_MGR:
g_value_set_object (value, dialog->priv->filter_mgr);
break;
case PROP_FILTER_MGR_ID:
g_value_set_string (value, dialog->priv->filter_mgr_id);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
string_slist_free (GSList *list)
{
g_slist_foreach (list, (GFunc) g_free, NULL);
g_slist_free (list);
}
static void
moo_file_dialog_finalize (GObject *object)
{
MooFileDialog *dialog = MOO_FILE_DIALOG (object);
g_free (dialog->priv->dir);
g_free (dialog->priv->title);
g_free (dialog->priv->name);
g_free (dialog->priv->filter_mgr_id);
g_free (dialog->priv->filename);
string_slist_free (dialog->priv->filenames);
if (dialog->priv->filter_mgr)
g_object_unref (dialog->priv->filter_mgr);
g_free (dialog->priv);
G_OBJECT_CLASS(moo_file_dialog_parent_class)->finalize (object);
}
static void
moo_file_dialog_init (MooFileDialog *dialog)
{
dialog->priv = g_new0 (MooFileDialogPrivate, 1);
dialog->priv->type = MOO_DIALOG_FILE_OPEN_EXISTING;
}
static void
moo_file_dialog_class_init (MooFileDialogClass *klass)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
gobject_class->finalize = moo_file_dialog_finalize;
gobject_class->set_property = moo_file_dialog_set_property;
gobject_class->get_property = moo_file_dialog_get_property;
g_object_class_install_property (gobject_class,
PROP_MULTIPLE,
g_param_spec_boolean ("multiple",
"multiple",
"multiple",
FALSE,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_TITLE,
g_param_spec_string ("title",
"title",
"title",
NULL,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_DIR,
g_param_spec_string ("dir",
"dir",
"dir",
NULL,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_NAME,
g_param_spec_string ("name",
"name",
"name",
NULL,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_PARENT,
g_param_spec_object ("parent",
"parent",
"parent",
GTK_TYPE_WIDGET,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_TYPE,
g_param_spec_enum ("type",
"type",
"type",
MOO_TYPE_FILE_DIALOG_TYPE,
MOO_DIALOG_FILE_OPEN_EXISTING,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_FILTER_MGR,
g_param_spec_object ("filter-mgr",
"filter-mgr",
"filter-mgr",
MOO_TYPE_FILTER_MGR,
G_PARAM_READWRITE));
g_object_class_install_property (gobject_class,
PROP_FILTER_MGR_ID,
g_param_spec_string ("filter-mgr-id",
"filter-mgr-id",
"filter-mgr-id",
NULL,
G_PARAM_READWRITE));
}
#if GTK_CHECK_VERSION(2,4,0)
inline static
GtkWidget *file_chooser_dialog_new (const char *title,
GtkFileChooserAction action,
const char *okbtn,
const char *start_dir)
{
GtkWidget *dialog =
gtk_file_chooser_dialog_new (title, NULL, action,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
okbtn, GTK_RESPONSE_OK,
NULL);
#if GTK_CHECK_VERSION(2,6,0)
gtk_dialog_set_alternative_button_order (GTK_DIALOG (dialog),
GTK_RESPONSE_OK,
GTK_RESPONSE_CANCEL,
-1);
#endif /* GTK_CHECK_VERSION(2,6,0) */
if (start_dir)
gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER(dialog),
start_dir);
return dialog;
}
#define file_chooser_set_select_multiple(dialog,multiple) \
gtk_file_chooser_set_select_multiple (GTK_FILE_CHOOSER (dialog), multiple)
#define file_chooser_get_filename(dialog) \
(gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog)))
#define file_chooser_get_filenames(dialog) \
(gtk_file_chooser_get_filenames (GTK_FILE_CHOOSER (dialog)))
#define file_chooser_set_name(dialog, name) \
gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), name)
#else /* !GTK_CHECK_VERSION(2,4,0) */
#define GtkFileChooserAction int
#define GTK_FILE_CHOOSER_ACTION_SAVE 1
#define GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER 2
#define GTK_FILE_CHOOSER_ACTION_OPEN 3
inline static
GtkWidget *file_chooser_dialog_new (const char *title,
G_GNUC_UNUSED GtkFileChooserAction action,
G_GNUC_UNUSED const char *okbtn,
const char *start_dir)
{
GtkWidget *dialog = gtk_file_selection_new (title);
if (start_dir)
{
char *dir = g_strdup_printf ("%s/", start_dir);
gtk_file_selection_set_filename (GTK_FILE_SELECTION (dialog), dir);
g_free (dir);
}
return dialog;
}
#define file_chooser_set_select_multiple(dialog,multiple) \
gtk_file_selection_set_select_multiple (GTK_FILE_SELECTION (dialog), multiple)
#define file_chooser_get_filename(dialog) \
g_strdup (gtk_file_selection_get_filename (GTK_FILE_SELECTION (dialog)))
#define file_chooser_get_filenames(dialog) \
(gtk_file_selection_get_selections (GTK_FILE_SELECTION (dialog)))
#define file_chooser_set_name(dialog, name)
#endif /* !GTK_CHECK_VERSION(2,4,0) */
const char *
moo_file_dialog (GtkWidget *parent,
MooFileDialogType type,
const char *title,
const char *start_dir)
{
static char *filename;
MooFileDialog *dialog;
dialog = moo_file_dialog_new (type, parent, FALSE, title, start_dir, NULL);
g_return_val_if_fail (dialog != NULL, NULL);
moo_file_dialog_run (dialog);
g_free (filename);
filename = g_strdup (moo_file_dialog_get_filename (dialog));
g_object_unref (dialog);
return filename;
}
static GtkWidget *
moo_file_dialog_create_widget (MooFileDialog *dialog)
{
GtkFileChooserAction chooser_action;
GtkWidget *widget = NULL;
switch (dialog->priv->type)
{
case MOO_DIALOG_FILE_OPEN_EXISTING:
case MOO_DIALOG_FILE_OPEN_ANY:
case MOO_DIALOG_DIR_OPEN:
if (dialog->priv->type == MOO_DIALOG_DIR_OPEN)
chooser_action = GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER;
else
chooser_action = GTK_FILE_CHOOSER_ACTION_OPEN;
widget = file_chooser_dialog_new (dialog->priv->title,
chooser_action,
GTK_STOCK_OPEN,
dialog->priv->dir);
file_chooser_set_select_multiple (widget, dialog->priv->multiple);
break;
case MOO_DIALOG_FILE_SAVE:
chooser_action = GTK_FILE_CHOOSER_ACTION_SAVE;
widget = file_chooser_dialog_new (dialog->priv->title,
chooser_action,
GTK_STOCK_SAVE,
dialog->priv->dir);
if (dialog->priv->name)
file_chooser_set_name (widget, dialog->priv->name);
break;
default:
g_critical ("%s: incorrect dialog type specified", G_STRLOC);
return NULL;
}
gtk_dialog_set_default_response (GTK_DIALOG (widget), GTK_RESPONSE_OK);
if (dialog->priv->filter_mgr)
moo_filter_mgr_attach (dialog->priv->filter_mgr,
GTK_FILE_CHOOSER (widget),
dialog->priv->filter_mgr_id);
return widget;
}
static GSList *
string_slist_copy (GSList *list)
{
GSList *copy = NULL;
GSList *l;
for (l = list; l != NULL; l = l->next)
copy = g_slist_prepend (copy, g_strdup (l->data));
return g_slist_reverse (copy);
}
static void
set_filename (MooFileDialog *dialog,
char *filename)
{
g_free (dialog->priv->filename);
dialog->priv->filename = filename;
}
static void
set_filenames (MooFileDialog *dialog,
GSList *filenames)
{
string_slist_free (dialog->priv->filenames);
dialog->priv->filenames = filenames;
}
gboolean
moo_file_dialog_run (MooFileDialog *dialog)
{
char *filename;
GtkWidget *filechooser;
gboolean result = FALSE;
g_return_val_if_fail (MOO_IS_FILE_DIALOG (dialog), FALSE);
set_filename (dialog, NULL);
set_filenames (dialog, NULL);
filechooser = moo_file_dialog_create_widget (dialog);
if (dialog->priv->parent)
moo_position_window (filechooser, dialog->priv->parent,
FALSE, FALSE, 0, 0);
switch (dialog->priv->type)
{
case MOO_DIALOG_FILE_OPEN_EXISTING:
case MOO_DIALOG_FILE_OPEN_ANY:
case MOO_DIALOG_DIR_OPEN:
if (gtk_dialog_run (GTK_DIALOG (filechooser)) == GTK_RESPONSE_OK)
{
set_filename (dialog, file_chooser_get_filename (filechooser));
if (dialog->priv->multiple)
set_filenames (dialog, file_chooser_get_filenames (filechooser));
result = TRUE;
}
goto out;
case MOO_DIALOG_FILE_SAVE:
while (TRUE)
{
if (GTK_RESPONSE_OK == gtk_dialog_run (GTK_DIALOG (filechooser)))
{
filename = file_chooser_get_filename (filechooser);
if (g_file_test (filename, G_FILE_TEST_EXISTS) &&
! g_file_test (filename, G_FILE_TEST_IS_REGULAR))
{
moo_error_dialog (filechooser,
"Choosen file is not a regular file",
NULL);
g_free (filename);
}
else if (g_file_test (filename, G_FILE_TEST_EXISTS) &&
g_file_test (filename, G_FILE_TEST_IS_REGULAR))
{
char *basename = g_path_get_basename (filename);
char *dirname = g_path_get_dirname (filename);
char *display_name = g_filename_display_name (basename);
char *display_dirname = g_filename_display_name (dirname);
gboolean overwrite;
overwrite = moo_overwrite_file_dialog (filechooser, display_name, display_dirname);
g_free (basename);
g_free (dirname);
g_free (display_name);
g_free (display_dirname);
if (overwrite)
{
set_filename (dialog, filename);
result = TRUE;
goto out;
}
}
else
{
set_filename (dialog, filename);
result = TRUE;
goto out;
}
}
else
{
goto out;
}
}
default:
g_critical ("%s: incorrect dialog type specified", G_STRLOC);
}
out:
gtk_widget_destroy (filechooser);
return result;
}
const char *
moo_file_dialog_get_filename (MooFileDialog *dialog)
{
g_return_val_if_fail (MOO_IS_FILE_DIALOG (dialog), NULL);
return dialog->priv->filename;
}
GSList *
moo_file_dialog_get_filenames (MooFileDialog *dialog)
{
g_return_val_if_fail (MOO_IS_FILE_DIALOG (dialog), NULL);
return string_slist_copy (dialog->priv->filenames);
}
const char *
moo_file_dialogp (GtkWidget *parent,
MooFileDialogType type,
const char *title,
const char *prefs_key,
const char *alternate_prefs_key)
{
const char *start = NULL;
const char *filename = NULL;
if (!title) title = "Choose File";
if (prefs_key && moo_prefs_get (prefs_key))
start = moo_prefs_get_string (prefs_key);
if (!start && alternate_prefs_key && moo_prefs_get (alternate_prefs_key))
start = moo_prefs_get_string (alternate_prefs_key);
filename = moo_file_dialog (parent, type, title, start);
if (filename && prefs_key)
{
char *new_start = g_path_get_dirname (filename);
moo_prefs_new_key_string (prefs_key, NULL);
moo_prefs_set_filename (prefs_key, new_start);
g_free (new_start);
}
return filename;
}
GType moo_file_dialog_type_get_type (void)
{
static GType type = 0;
if (!type)
{
static const GEnumValue values[] = {
{ MOO_DIALOG_FILE_OPEN_EXISTING, (char*)"MOO_DIALOG_FILE_OPEN_EXISTING", (char*)"file-open-existing" },
{ MOO_DIALOG_FILE_OPEN_ANY, (char*)"MOO_DIALOG_FILE_OPEN_ANY", (char*)"file-open-any" },
{ MOO_DIALOG_FILE_SAVE, (char*)"MOO_DIALOG_FILE_SAVE", (char*)"file-save" },
{ MOO_DIALOG_DIR_OPEN, (char*)"MOO_DIALOG_DIR_OPEN", (char*)"dir-open" },
{ 0, NULL, NULL }
};
type = g_enum_register_static ("MooFileDialogType", values);
}
return type;
}
MooFileDialog *
moo_file_dialog_new (MooFileDialogType type,
GtkWidget *parent,
gboolean multiple,
const char *title,
const char *start_dir,
const char *start_name)
{
return g_object_new (MOO_TYPE_FILE_DIALOG,
"parent", parent,
"type", type,
"multiple", multiple,
"title", title,
"dir", start_dir,
"name", start_name,
NULL);
}
void
moo_file_dialog_set_filter_mgr (MooFileDialog *dialog,
MooFilterMgr *mgr,
const char *id)
{
g_return_if_fail (MOO_IS_FILE_DIALOG (dialog));
g_return_if_fail (!mgr || MOO_IS_FILTER_MGR (mgr));
g_object_set (dialog, "filter-mgr", mgr, "filter-mgr-id", id, NULL);
}