531 lines
16 KiB
C
531 lines
16 KiB
C
/* LIBPIKA - The PIKA Library
|
|
* Copyright (C) 1995-1997 Peter Mattis and Spencer Kimball
|
|
*
|
|
* pikafileentry.c
|
|
* Copyright (C) 1999-2004 Michael Natterer <mitch@gimp.org>
|
|
*
|
|
* This library 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 3 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libpikabase/pikabase.h"
|
|
|
|
#include "pikawidgetstypes.h"
|
|
|
|
#include "pikadialog.h"
|
|
|
|
#undef PIKA_DISABLE_DEPRECATED
|
|
#include "pikafileentry.h"
|
|
|
|
#include "pikahelpui.h"
|
|
#include "pikaicons.h"
|
|
|
|
#include "libpika/libpika-intl.h"
|
|
|
|
|
|
/**
|
|
* SECTION: pikafileentry
|
|
* @title: PikaFileEntry
|
|
* @short_description: Widget for entering a filename.
|
|
* @see_also: #PikaPathEditor
|
|
*
|
|
* This widget is used to enter filenames or directories.
|
|
*
|
|
* There is a #GtkEntry for entering the filename manually and a "..."
|
|
* button which will pop up a #GtkFileChooserDialog.
|
|
*
|
|
* You can restrict the #PikaFileEntry to directories. In this
|
|
* case the filename listbox of the #GtkFileChooser dialog will be
|
|
* set to directory mode.
|
|
*
|
|
* If you specify @check_valid as %TRUE in pika_file_entry_new() the
|
|
* entered filename will be checked for validity and a pixmap will be
|
|
* shown which indicates if the file exists or not.
|
|
*
|
|
* Whenever the user changes the filename, the "filename_changed"
|
|
* signal will be emitted.
|
|
**/
|
|
|
|
|
|
enum
|
|
{
|
|
FILENAME_CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
struct _PikaFileEntryPrivate
|
|
{
|
|
GtkWidget *file_exists;
|
|
GtkWidget *entry;
|
|
GtkWidget *browse_button;
|
|
|
|
GtkWidget *file_dialog;
|
|
|
|
gchar *title;
|
|
gboolean dir_only;
|
|
gboolean check_valid;
|
|
};
|
|
|
|
|
|
static void pika_file_entry_dispose (GObject *object);
|
|
|
|
static void pika_file_entry_entry_changed (GtkWidget *widget,
|
|
GtkWidget *button);
|
|
static void pika_file_entry_entry_activate (GtkWidget *widget,
|
|
PikaFileEntry *entry);
|
|
static gint pika_file_entry_entry_focus_out (GtkWidget *widget,
|
|
GdkEvent *event,
|
|
PikaFileEntry *entry);
|
|
static void pika_file_entry_file_manager_clicked (GtkWidget *widget,
|
|
PikaFileEntry *entry);
|
|
static void pika_file_entry_browse_clicked (GtkWidget *widget,
|
|
PikaFileEntry *entry);
|
|
static void pika_file_entry_check_filename (PikaFileEntry *entry);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (PikaFileEntry, pika_file_entry, GTK_TYPE_BOX)
|
|
|
|
#define parent_class pika_file_entry_parent_class
|
|
|
|
static guint pika_file_entry_signals[LAST_SIGNAL] = { 0 };
|
|
|
|
|
|
static void
|
|
pika_file_entry_class_init (PikaFileEntryClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
/**
|
|
* PikaFileEntry::filename-changed:
|
|
*
|
|
* This signal is emitted whenever the user changes the filename.
|
|
**/
|
|
pika_file_entry_signals[FILENAME_CHANGED] =
|
|
g_signal_new ("filename-changed",
|
|
G_TYPE_FROM_CLASS (klass),
|
|
G_SIGNAL_RUN_FIRST,
|
|
G_STRUCT_OFFSET (PikaFileEntryClass, filename_changed),
|
|
NULL, NULL, NULL,
|
|
G_TYPE_NONE, 0);
|
|
|
|
object_class->dispose = pika_file_entry_dispose;
|
|
|
|
klass->filename_changed = NULL;
|
|
}
|
|
|
|
static void
|
|
pika_file_entry_init (PikaFileEntry *entry)
|
|
{
|
|
GtkWidget *image;
|
|
GtkWidget *button;
|
|
|
|
entry->priv = pika_file_entry_get_instance_private (entry);
|
|
entry->priv->title = NULL;
|
|
entry->priv->file_dialog = NULL;
|
|
entry->priv->check_valid = FALSE;
|
|
entry->priv->file_exists = NULL;
|
|
|
|
gtk_orientable_set_orientation (GTK_ORIENTABLE (entry),
|
|
GTK_ORIENTATION_HORIZONTAL);
|
|
|
|
gtk_box_set_spacing (GTK_BOX (entry), 4);
|
|
gtk_box_set_homogeneous (GTK_BOX (entry), FALSE);
|
|
|
|
button = gtk_button_new ();
|
|
gtk_box_pack_end (GTK_BOX (entry), button, FALSE, FALSE, 0);
|
|
gtk_widget_show (button);
|
|
|
|
gtk_widget_set_sensitive (button, FALSE);
|
|
|
|
image = gtk_image_new_from_icon_name (PIKA_ICON_FILE_MANAGER,
|
|
GTK_ICON_SIZE_BUTTON);
|
|
gtk_container_add (GTK_CONTAINER (button), image);
|
|
gtk_widget_show (image);
|
|
|
|
g_signal_connect (button, "clicked",
|
|
G_CALLBACK (pika_file_entry_file_manager_clicked),
|
|
entry);
|
|
|
|
pika_help_set_help_data (button,
|
|
_("Show file location in the file manager"),
|
|
NULL);
|
|
|
|
entry->priv->browse_button = gtk_button_new ();
|
|
gtk_box_pack_end (GTK_BOX (entry), entry->priv->browse_button, FALSE, FALSE, 0);
|
|
gtk_widget_show (entry->priv->browse_button);
|
|
|
|
image = gtk_image_new_from_icon_name (PIKA_ICON_DOCUMENT_OPEN,
|
|
GTK_ICON_SIZE_BUTTON);
|
|
gtk_container_add (GTK_CONTAINER (entry->priv->browse_button), image);
|
|
gtk_widget_show (image);
|
|
|
|
g_signal_connect (entry->priv->browse_button, "clicked",
|
|
G_CALLBACK (pika_file_entry_browse_clicked),
|
|
entry);
|
|
|
|
entry->priv->entry = gtk_entry_new ();
|
|
gtk_box_pack_end (GTK_BOX (entry), entry->priv->entry, TRUE, TRUE, 0);
|
|
gtk_widget_show (entry->priv->entry);
|
|
|
|
g_signal_connect (entry->priv->entry, "changed",
|
|
G_CALLBACK (pika_file_entry_entry_changed),
|
|
button);
|
|
g_signal_connect (entry->priv->entry, "activate",
|
|
G_CALLBACK (pika_file_entry_entry_activate),
|
|
entry);
|
|
g_signal_connect (entry->priv->entry, "focus-out-event",
|
|
G_CALLBACK (pika_file_entry_entry_focus_out),
|
|
entry);
|
|
}
|
|
|
|
static void
|
|
pika_file_entry_dispose (GObject *object)
|
|
{
|
|
PikaFileEntry *entry = PIKA_FILE_ENTRY (object);
|
|
|
|
g_clear_pointer (&entry->priv->file_dialog, gtk_widget_destroy);
|
|
|
|
g_clear_pointer (&entry->priv->title, g_free);
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
/**
|
|
* pika_file_entry_new:
|
|
* @title: The title of the #PikaFileEntry dialog.
|
|
* @filename: The initial filename.
|
|
* @dir_only: %TRUE if the file entry should accept directories only.
|
|
* @check_valid: %TRUE if the widget should check if the entered file
|
|
* really exists.
|
|
*
|
|
* You should use #GtkFileChooserButton instead.
|
|
*
|
|
* Returns: A pointer to the new #PikaFileEntry widget.
|
|
**/
|
|
GtkWidget *
|
|
pika_file_entry_new (const gchar *title,
|
|
const gchar *filename,
|
|
gboolean dir_only,
|
|
gboolean check_valid)
|
|
{
|
|
PikaFileEntry *entry;
|
|
|
|
entry = g_object_new (PIKA_TYPE_FILE_ENTRY, NULL);
|
|
|
|
entry->priv->title = g_strdup (title);
|
|
entry->priv->dir_only = dir_only;
|
|
entry->priv->check_valid = check_valid;
|
|
|
|
pika_help_set_help_data (entry->priv->browse_button,
|
|
entry->priv->dir_only ?
|
|
_("Open a file selector to browse your folders") :
|
|
_("Open a file selector to browse your files"),
|
|
NULL);
|
|
|
|
if (check_valid)
|
|
{
|
|
entry->priv->file_exists = gtk_image_new_from_icon_name ("gtk-no",
|
|
GTK_ICON_SIZE_BUTTON);
|
|
gtk_box_pack_start (GTK_BOX (entry), entry->priv->file_exists, FALSE, FALSE, 0);
|
|
gtk_widget_show (entry->priv->file_exists);
|
|
|
|
pika_help_set_help_data (entry->priv->file_exists,
|
|
entry->priv->dir_only ?
|
|
_("Indicates whether or not the folder exists") :
|
|
_("Indicates whether or not the file exists"),
|
|
NULL);
|
|
}
|
|
|
|
pika_file_entry_set_filename (entry, filename);
|
|
|
|
return GTK_WIDGET (entry);
|
|
}
|
|
|
|
/**
|
|
* pika_file_entry_get_filename:
|
|
* @entry: The file entry you want to know the filename from.
|
|
*
|
|
* Note that you have to g_free() the returned string.
|
|
*
|
|
* Returns: The file or directory the user has entered.
|
|
**/
|
|
gchar *
|
|
pika_file_entry_get_filename (PikaFileEntry *entry)
|
|
{
|
|
gchar *utf8;
|
|
gchar *filename;
|
|
|
|
g_return_val_if_fail (PIKA_IS_FILE_ENTRY (entry), NULL);
|
|
|
|
utf8 = gtk_editable_get_chars (GTK_EDITABLE (entry->priv->entry), 0, -1);
|
|
|
|
filename = g_filename_from_utf8 (utf8, -1, NULL, NULL, NULL);
|
|
|
|
g_free (utf8);
|
|
|
|
return filename;
|
|
}
|
|
|
|
/**
|
|
* pika_file_entry_set_filename:
|
|
* @entry: The file entry you want to set the filename for.
|
|
* @filename: The new filename.
|
|
*
|
|
* If you specified @check_valid as %TRUE in pika_file_entry_new()
|
|
* the #PikaFileEntry will immediately check the validity of the file
|
|
* name.
|
|
**/
|
|
void
|
|
pika_file_entry_set_filename (PikaFileEntry *entry,
|
|
const gchar *filename)
|
|
{
|
|
gchar *utf8;
|
|
|
|
g_return_if_fail (PIKA_IS_FILE_ENTRY (entry));
|
|
|
|
if (filename)
|
|
utf8 = g_filename_to_utf8 (filename, -1, NULL, NULL, NULL);
|
|
else
|
|
utf8 = g_strdup ("");
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (entry->priv->entry), utf8);
|
|
g_free (utf8);
|
|
|
|
/* update everything
|
|
*/
|
|
pika_file_entry_entry_activate (entry->priv->entry, entry);
|
|
}
|
|
|
|
/**
|
|
* pika_file_entry_get_entry:
|
|
* @entry: The #PikaFileEntry.
|
|
*
|
|
* Returns: (transfer none): the #GtkEntry internally used by the
|
|
* widget. The object belongs to @entry and should not be
|
|
* freed.
|
|
**/
|
|
GtkWidget *
|
|
pika_file_entry_get_entry (PikaFileEntry *entry)
|
|
{
|
|
return entry->priv->entry;
|
|
}
|
|
|
|
/* Private Functions */
|
|
|
|
static void
|
|
pika_file_entry_entry_changed (GtkWidget *widget,
|
|
GtkWidget *button)
|
|
{
|
|
const gchar *text = gtk_entry_get_text (GTK_ENTRY (widget));
|
|
|
|
if (text && strlen (text))
|
|
gtk_widget_set_sensitive (button, TRUE);
|
|
else
|
|
gtk_widget_set_sensitive (button, FALSE);
|
|
}
|
|
|
|
static void
|
|
pika_file_entry_entry_activate (GtkWidget *widget,
|
|
PikaFileEntry *entry)
|
|
{
|
|
gchar *utf8;
|
|
gchar *filename;
|
|
gint len;
|
|
|
|
/* filenames still need more sanity checking
|
|
* (erase double G_DIR_SEPARATORS, ...)
|
|
*/
|
|
utf8 = gtk_editable_get_chars (GTK_EDITABLE (widget), 0, -1);
|
|
utf8 = g_strstrip (utf8);
|
|
|
|
while (((len = strlen (utf8)) > 1) &&
|
|
(utf8[len - 1] == G_DIR_SEPARATOR))
|
|
utf8[len - 1] = '\0';
|
|
|
|
filename = g_filename_from_utf8 (utf8, -1, NULL, NULL, NULL);
|
|
|
|
g_signal_handlers_block_by_func (entry->priv->entry,
|
|
pika_file_entry_entry_activate,
|
|
entry);
|
|
gtk_entry_set_text (GTK_ENTRY (entry->priv->entry), utf8);
|
|
g_signal_handlers_unblock_by_func (entry->priv->entry,
|
|
pika_file_entry_entry_activate,
|
|
entry);
|
|
|
|
if (entry->priv->file_dialog)
|
|
gtk_file_chooser_set_filename (GTK_FILE_CHOOSER (entry->priv->file_dialog),
|
|
filename);
|
|
|
|
g_free (filename);
|
|
g_free (utf8);
|
|
|
|
pika_file_entry_check_filename (entry);
|
|
|
|
gtk_editable_set_position (GTK_EDITABLE (entry->priv->entry), -1);
|
|
|
|
g_signal_emit (entry, pika_file_entry_signals[FILENAME_CHANGED], 0);
|
|
}
|
|
|
|
static gboolean
|
|
pika_file_entry_entry_focus_out (GtkWidget *widget,
|
|
GdkEvent *event,
|
|
PikaFileEntry *entry)
|
|
{
|
|
pika_file_entry_entry_activate (widget, entry);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* local callback of pika_file_entry_browse_clicked() */
|
|
static void
|
|
pika_file_entry_chooser_response (GtkWidget *dialog,
|
|
gint response_id,
|
|
PikaFileEntry *entry)
|
|
{
|
|
if (response_id == GTK_RESPONSE_OK)
|
|
{
|
|
gchar *filename;
|
|
|
|
filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
|
|
pika_file_entry_set_filename (entry, filename);
|
|
g_free (filename);
|
|
}
|
|
|
|
gtk_widget_hide (dialog);
|
|
}
|
|
|
|
static void
|
|
pika_file_entry_file_manager_clicked (GtkWidget *widget,
|
|
PikaFileEntry *entry)
|
|
{
|
|
gchar *utf8;
|
|
GFile *file;
|
|
GError *error = NULL;
|
|
|
|
utf8 = gtk_editable_get_chars (GTK_EDITABLE (entry->priv->entry), 0, -1);
|
|
file = g_file_parse_name (utf8);
|
|
g_free (utf8);
|
|
|
|
if (! pika_file_show_in_file_manager (file, &error))
|
|
{
|
|
g_message (_("Can't show file in file manager: %s"),
|
|
error->message);
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
g_object_unref (file);
|
|
}
|
|
|
|
static void
|
|
pika_file_entry_browse_clicked (GtkWidget *widget,
|
|
PikaFileEntry *entry)
|
|
{
|
|
GtkFileChooser *chooser;
|
|
gchar *utf8;
|
|
gchar *filename;
|
|
|
|
utf8 = gtk_editable_get_chars (GTK_EDITABLE (entry->priv->entry), 0, -1);
|
|
filename = g_filename_from_utf8 (utf8, -1, NULL, NULL, NULL);
|
|
g_free (utf8);
|
|
|
|
if (! entry->priv->file_dialog)
|
|
{
|
|
const gchar *title = entry->priv->title;
|
|
|
|
if (! title)
|
|
{
|
|
if (entry->priv->dir_only)
|
|
title = _("Select Folder");
|
|
else
|
|
title = _("Select File");
|
|
}
|
|
|
|
entry->priv->file_dialog =
|
|
gtk_file_chooser_dialog_new (title, NULL,
|
|
entry->priv->dir_only ?
|
|
GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER :
|
|
GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
_("_OK"), GTK_RESPONSE_OK,
|
|
|
|
NULL);
|
|
|
|
pika_dialog_set_alternative_button_order (GTK_DIALOG (entry->priv->file_dialog),
|
|
GTK_RESPONSE_OK,
|
|
GTK_RESPONSE_CANCEL,
|
|
-1);
|
|
|
|
chooser = GTK_FILE_CHOOSER (entry->priv->file_dialog);
|
|
|
|
gtk_window_set_position (GTK_WINDOW (chooser), GTK_WIN_POS_MOUSE);
|
|
gtk_window_set_role (GTK_WINDOW (chooser),
|
|
"pika-file-entry-file-dialog");
|
|
|
|
g_signal_connect (chooser, "response",
|
|
G_CALLBACK (pika_file_entry_chooser_response),
|
|
entry);
|
|
g_signal_connect (chooser, "delete-event",
|
|
G_CALLBACK (gtk_true),
|
|
NULL);
|
|
|
|
g_signal_connect_swapped (entry, "unmap",
|
|
G_CALLBACK (gtk_widget_hide),
|
|
chooser);
|
|
}
|
|
else
|
|
{
|
|
chooser = GTK_FILE_CHOOSER (entry->priv->file_dialog);
|
|
}
|
|
|
|
gtk_file_chooser_set_filename (chooser, filename);
|
|
|
|
g_free (filename);
|
|
|
|
gtk_window_set_screen (GTK_WINDOW (chooser), gtk_widget_get_screen (widget));
|
|
gtk_window_present (GTK_WINDOW (chooser));
|
|
}
|
|
|
|
static void
|
|
pika_file_entry_check_filename (PikaFileEntry *entry)
|
|
{
|
|
gchar *utf8;
|
|
gchar *filename;
|
|
gboolean exists;
|
|
|
|
if (! entry->priv->check_valid || ! entry->priv->file_exists)
|
|
return;
|
|
|
|
utf8 = gtk_editable_get_chars (GTK_EDITABLE (entry->priv->entry), 0, -1);
|
|
filename = g_filename_from_utf8 (utf8, -1, NULL, NULL, NULL);
|
|
g_free (utf8);
|
|
|
|
if (entry->priv->dir_only)
|
|
exists = g_file_test (filename, G_FILE_TEST_IS_DIR);
|
|
else
|
|
exists = g_file_test (filename, G_FILE_TEST_IS_REGULAR);
|
|
|
|
g_free (filename);
|
|
|
|
gtk_image_set_from_icon_name (GTK_IMAGE (entry->priv->file_exists),
|
|
exists ? "gtk-yes" : "gtk-no",
|
|
GTK_ICON_SIZE_BUTTON);
|
|
}
|