Blame gtk/gtkfilechooserentry.c

Packit 98cdb6
/* GTK - The GIMP Toolkit
Packit 98cdb6
 * gtkfilechooserentry.c: Entry with filename completion
Packit 98cdb6
 * Copyright (C) 2003, Red Hat, Inc.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include "config.h"
Packit 98cdb6
Packit 98cdb6
#include "gtkfilechooserentry.h"
Packit 98cdb6
Packit 98cdb6
#include <string.h>
Packit 98cdb6
Packit 98cdb6
#include "gtkalignment.h"
Packit 98cdb6
#include "gtkcelllayout.h"
Packit 98cdb6
#include "gtkcellrenderertext.h"
Packit 98cdb6
#include "gtkentryprivate.h"
Packit 98cdb6
#include "gtkfilesystemmodel.h"
Packit 98cdb6
#include "gtklabel.h"
Packit 98cdb6
#include "gtkmain.h"
Packit 98cdb6
#include "gtkwindow.h"
Packit 98cdb6
#include "gtkintl.h"
Packit 98cdb6
#include "gtkalias.h"
Packit 98cdb6
Packit 98cdb6
#include "gdkkeysyms.h"
Packit 98cdb6
Packit 98cdb6
typedef struct _GtkFileChooserEntryClass GtkFileChooserEntryClass;
Packit 98cdb6
Packit 98cdb6
#define GTK_FILE_CHOOSER_ENTRY_CLASS(klass)     (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_FILE_CHOOSER_ENTRY, GtkFileChooserEntryClass))
Packit 98cdb6
#define GTK_IS_FILE_CHOOSER_ENTRY_CLASS(klass)  (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_FILE_CHOOSER_ENTRY))
Packit 98cdb6
#define GTK_FILE_CHOOSER_ENTRY_GET_CLASS(obj)   (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_FILE_CHOOSER_ENTRY, GtkFileChooserEntryClass))
Packit 98cdb6
Packit 98cdb6
struct _GtkFileChooserEntryClass
Packit 98cdb6
{
Packit 98cdb6
  GtkEntryClass parent_class;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
struct _GtkFileChooserEntry
Packit 98cdb6
{
Packit 98cdb6
  GtkEntry parent_instance;
Packit 98cdb6
Packit 98cdb6
  GtkFileChooserAction action;
Packit 98cdb6
Packit 98cdb6
  GFile *base_folder;
Packit 98cdb6
  GFile *current_folder_file;
Packit 98cdb6
  gchar *dir_part;
Packit 98cdb6
  gchar *file_part;
Packit 98cdb6
Packit 98cdb6
  GtkTreeModel *completion_store;
Packit 98cdb6
Packit 98cdb6
  guint current_folder_loaded : 1;
Packit 98cdb6
  guint complete_on_load : 1;
Packit 98cdb6
  guint eat_tabs       : 1;
Packit 98cdb6
  guint local_only     : 1;
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
enum
Packit 98cdb6
{
Packit 98cdb6
  DISPLAY_NAME_COLUMN,
Packit 98cdb6
  FULL_PATH_COLUMN,
Packit 98cdb6
  N_COLUMNS
Packit 98cdb6
};
Packit 98cdb6
Packit 98cdb6
static void     gtk_file_chooser_entry_finalize       (GObject          *object);
Packit 98cdb6
static void     gtk_file_chooser_entry_dispose        (GObject          *object);
Packit 98cdb6
static void     gtk_file_chooser_entry_grab_focus     (GtkWidget        *widget);
Packit 98cdb6
static gboolean gtk_file_chooser_entry_tab_handler    (GtkWidget *widget,
Packit 98cdb6
						       GdkEventKey *event);
Packit 98cdb6
static gboolean gtk_file_chooser_entry_focus_out_event (GtkWidget       *widget,
Packit 98cdb6
							GdkEventFocus   *event);
Packit 98cdb6
Packit 98cdb6
#ifdef G_OS_WIN32
Packit 98cdb6
static gint     insert_text_callback      (GtkFileChooserEntry *widget,
Packit 98cdb6
					   const gchar         *new_text,
Packit 98cdb6
					   gint                 new_text_length,
Packit 98cdb6
					   gint                *position,
Packit 98cdb6
					   gpointer             user_data);
Packit 98cdb6
static void     delete_text_callback      (GtkFileChooserEntry *widget,
Packit 98cdb6
					   gint                 start_pos,
Packit 98cdb6
					   gint                 end_pos,
Packit 98cdb6
					   gpointer             user_data);
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
static gboolean match_selected_callback   (GtkEntryCompletion  *completion,
Packit 98cdb6
					   GtkTreeModel        *model,
Packit 98cdb6
					   GtkTreeIter         *iter,
Packit 98cdb6
					   GtkFileChooserEntry *chooser_entry);
Packit 98cdb6
Packit 98cdb6
static void set_complete_on_load (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                                  gboolean             complete_on_load);
Packit 98cdb6
static void refresh_current_folder_and_file_part (GtkFileChooserEntry *chooser_entry);
Packit 98cdb6
static void set_completion_folder (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                                   GFile               *folder,
Packit 98cdb6
				   char                *dir_part);
Packit 98cdb6
static void finished_loading_cb (GtkFileSystemModel  *model,
Packit 98cdb6
                                 GError              *error,
Packit 98cdb6
		                 GtkFileChooserEntry *chooser_entry);
Packit 98cdb6
Packit 98cdb6
G_DEFINE_TYPE (GtkFileChooserEntry, _gtk_file_chooser_entry, GTK_TYPE_ENTRY)
Packit 98cdb6
Packit 98cdb6
static char *
Packit 98cdb6
gtk_file_chooser_entry_get_completion_text (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  GtkEditable *editable = GTK_EDITABLE (chooser_entry);
Packit 98cdb6
  int start, end;
Packit 98cdb6
Packit 98cdb6
  gtk_editable_get_selection_bounds (editable, &start, &end;;
Packit 98cdb6
  return gtk_editable_get_chars (editable, 0, MIN (start, end));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_file_chooser_entry_dispatch_properties_changed (GObject     *object,
Packit 98cdb6
                                                    guint        n_pspecs,
Packit 98cdb6
                                                    GParamSpec **pspecs)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry = GTK_FILE_CHOOSER_ENTRY (object);
Packit 98cdb6
  guint i;
Packit 98cdb6
Packit 98cdb6
  G_OBJECT_CLASS (_gtk_file_chooser_entry_parent_class)->dispatch_properties_changed (object, n_pspecs, pspecs);
Packit 98cdb6
Packit 98cdb6
  /* Don't do this during or after disposal */
Packit 98cdb6
  if (gtk_widget_get_parent (GTK_WIDGET (object)) != NULL)
Packit 98cdb6
    {
Packit 98cdb6
      /* What we are after: The text in front of the cursor was modified.
Packit 98cdb6
       * Unfortunately, there's no other way to catch this.
Packit 98cdb6
       */
Packit 98cdb6
      for (i = 0; i < n_pspecs; i++)
Packit 98cdb6
        {
Packit 98cdb6
          if (pspecs[i]->name == I_("cursor-position") ||
Packit 98cdb6
              pspecs[i]->name == I_("selection-bound") ||
Packit 98cdb6
              pspecs[i]->name == I_("text"))
Packit 98cdb6
            {
Packit 98cdb6
              set_complete_on_load (chooser_entry, FALSE);
Packit 98cdb6
              refresh_current_folder_and_file_part (chooser_entry);
Packit 98cdb6
              break;
Packit 98cdb6
            }
Packit 98cdb6
        }
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
_gtk_file_chooser_entry_class_init (GtkFileChooserEntryClass *class)
Packit 98cdb6
{
Packit 98cdb6
  GObjectClass *gobject_class = G_OBJECT_CLASS (class);
Packit 98cdb6
  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
Packit 98cdb6
Packit 98cdb6
  gobject_class->finalize = gtk_file_chooser_entry_finalize;
Packit 98cdb6
  gobject_class->dispose = gtk_file_chooser_entry_dispose;
Packit 98cdb6
  gobject_class->dispatch_properties_changed = gtk_file_chooser_entry_dispatch_properties_changed;
Packit 98cdb6
Packit 98cdb6
  widget_class->grab_focus = gtk_file_chooser_entry_grab_focus;
Packit 98cdb6
  widget_class->focus_out_event = gtk_file_chooser_entry_focus_out_event;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
_gtk_file_chooser_entry_init (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  GtkEntryCompletion *comp;
Packit 98cdb6
  GtkCellRenderer *cell;
Packit 98cdb6
Packit 98cdb6
  chooser_entry->local_only = TRUE;
Packit 98cdb6
Packit 98cdb6
  g_object_set (chooser_entry, "truncate-multiline", TRUE, NULL);
Packit 98cdb6
Packit 98cdb6
  comp = gtk_entry_completion_new ();
Packit 98cdb6
  gtk_entry_completion_set_popup_single_match (comp, FALSE);
Packit 98cdb6
  gtk_entry_completion_set_minimum_key_length (comp, 0);
Packit 98cdb6
  /* see docs for gtk_entry_completion_set_text_column() */
Packit 98cdb6
  g_object_set (comp, "text-column", FULL_PATH_COLUMN, NULL);
Packit 98cdb6
Packit 98cdb6
  /* Need a match func here or entry completion uses a wrong one.
Packit 98cdb6
   * We do our own filtering after all. */
Packit 98cdb6
  gtk_entry_completion_set_match_func (comp,
Packit 98cdb6
				       (GtkEntryCompletionMatchFunc) gtk_true,
Packit 98cdb6
				       chooser_entry,
Packit 98cdb6
				       NULL);
Packit 98cdb6
Packit 98cdb6
  cell = gtk_cell_renderer_text_new ();
Packit 98cdb6
  gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (comp),
Packit 98cdb6
                              cell, TRUE);
Packit 98cdb6
  gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (comp),
Packit 98cdb6
                                 cell,
Packit 98cdb6
                                 "text", DISPLAY_NAME_COLUMN);
Packit 98cdb6
Packit 98cdb6
  g_signal_connect (comp, "match-selected",
Packit 98cdb6
		    G_CALLBACK (match_selected_callback), chooser_entry);
Packit 98cdb6
Packit 98cdb6
  gtk_entry_set_completion (GTK_ENTRY (chooser_entry), comp);
Packit 98cdb6
  g_object_unref (comp);
Packit 98cdb6
  /* NB: This needs to happen after the completion is set, so this handler
Packit 98cdb6
   * runs before the handler installed by entrycompletion */
Packit 98cdb6
  g_signal_connect (chooser_entry, "key-press-event",
Packit 98cdb6
                    G_CALLBACK (gtk_file_chooser_entry_tab_handler), NULL);
Packit 98cdb6
Packit 98cdb6
#ifdef G_OS_WIN32
Packit 98cdb6
  g_signal_connect (chooser_entry, "insert-text",
Packit 98cdb6
		    G_CALLBACK (insert_text_callback), NULL);
Packit 98cdb6
  g_signal_connect (chooser_entry, "delete-text",
Packit 98cdb6
		    G_CALLBACK (delete_text_callback), NULL);
Packit 98cdb6
#endif
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_file_chooser_entry_finalize (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry = GTK_FILE_CHOOSER_ENTRY (object);
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->base_folder)
Packit 98cdb6
    g_object_unref (chooser_entry->base_folder);
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->current_folder_file)
Packit 98cdb6
    g_object_unref (chooser_entry->current_folder_file);
Packit 98cdb6
Packit 98cdb6
  g_free (chooser_entry->dir_part);
Packit 98cdb6
  g_free (chooser_entry->file_part);
Packit 98cdb6
Packit 98cdb6
  G_OBJECT_CLASS (_gtk_file_chooser_entry_parent_class)->finalize (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_file_chooser_entry_dispose (GObject *object)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry = GTK_FILE_CHOOSER_ENTRY (object);
Packit 98cdb6
Packit 98cdb6
  set_completion_folder (chooser_entry, NULL, NULL);
Packit 98cdb6
Packit 98cdb6
  G_OBJECT_CLASS (_gtk_file_chooser_entry_parent_class)->dispose (object);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Match functions for the GtkEntryCompletion */
Packit 98cdb6
static gboolean
Packit 98cdb6
match_selected_callback (GtkEntryCompletion  *completion,
Packit 98cdb6
                         GtkTreeModel        *model,
Packit 98cdb6
                         GtkTreeIter         *iter,
Packit 98cdb6
                         GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  char *path;
Packit 98cdb6
  gint pos;
Packit 98cdb6
Packit 98cdb6
  gtk_tree_model_get (model, iter,
Packit 98cdb6
                      FULL_PATH_COLUMN, &path,
Packit 98cdb6
                      -1);
Packit 98cdb6
Packit 98cdb6
  gtk_editable_delete_text (GTK_EDITABLE (chooser_entry),
Packit 98cdb6
                            0,
Packit 98cdb6
                            gtk_editable_get_position (GTK_EDITABLE (chooser_entry)));
Packit 98cdb6
  pos = 0;
Packit 98cdb6
  gtk_editable_insert_text (GTK_EDITABLE (chooser_entry),
Packit 98cdb6
                            path,
Packit 98cdb6
                            -1,
Packit 98cdb6
                            &pos;;
Packit 98cdb6
Packit 98cdb6
  gtk_editable_set_position (GTK_EDITABLE (chooser_entry), pos);
Packit 98cdb6
Packit 98cdb6
  g_free (path);
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_complete_on_load (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                      gboolean             complete_on_load)
Packit 98cdb6
{
Packit 98cdb6
  /* a completion was triggered, but we couldn't do it.
Packit 98cdb6
   * So no text was inserted when pressing tab, so we beep
Packit 98cdb6
   */
Packit 98cdb6
  if (chooser_entry->complete_on_load && !complete_on_load)
Packit 98cdb6
    gtk_widget_error_bell (GTK_WIDGET (chooser_entry));
Packit 98cdb6
Packit 98cdb6
  chooser_entry->complete_on_load = complete_on_load;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
is_valid_scheme_character (char c)
Packit 98cdb6
{
Packit 98cdb6
  return g_ascii_isalnum (c) || c == '+' || c == '-' || c == '.';
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
has_uri_scheme (const char *str)
Packit 98cdb6
{
Packit 98cdb6
  const char *p;
Packit 98cdb6
Packit 98cdb6
  p = str;
Packit 98cdb6
Packit 98cdb6
  if (!is_valid_scheme_character (*p))
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  do
Packit 98cdb6
    p++;
Packit 98cdb6
  while (is_valid_scheme_character (*p));
Packit 98cdb6
Packit 98cdb6
  return (strncmp (p, "://", 3) == 0);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GFile *
Packit 98cdb6
gtk_file_chooser_get_file_for_text (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                                    const gchar         *str)
Packit 98cdb6
{
Packit 98cdb6
  GFile *file;
Packit 98cdb6
Packit 98cdb6
  if (str[0] == '~' || g_path_is_absolute (str) || has_uri_scheme (str))
Packit 98cdb6
    file = g_file_parse_name (str);
Packit 98cdb6
  else if (chooser_entry->base_folder != NULL)
Packit 98cdb6
    file = g_file_resolve_relative_path (chooser_entry->base_folder, str);
Packit 98cdb6
  else
Packit 98cdb6
    file = NULL;
Packit 98cdb6
Packit 98cdb6
  return file;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
is_directory_shortcut (const char *text)
Packit 98cdb6
{
Packit 98cdb6
  return strcmp (text, ".") == 0 ||
Packit 98cdb6
         strcmp (text, "..") == 0 ||
Packit 98cdb6
         strcmp (text, "~" ) == 0;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GFile *
Packit 98cdb6
gtk_file_chooser_get_directory_for_text (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                                         const char *         text)
Packit 98cdb6
{
Packit 98cdb6
  GFile *file, *parent;
Packit 98cdb6
Packit 98cdb6
  file = gtk_file_chooser_get_file_for_text (chooser_entry, text);
Packit 98cdb6
Packit 98cdb6
  if (file == NULL)
Packit 98cdb6
    return NULL;
Packit 98cdb6
Packit 98cdb6
  if (text[0] == 0 || text[strlen (text) - 1] == G_DIR_SEPARATOR ||
Packit 98cdb6
      is_directory_shortcut (text))
Packit 98cdb6
    return file;
Packit 98cdb6
Packit 98cdb6
  parent = g_file_get_parent (file);
Packit 98cdb6
  g_object_unref (file);
Packit 98cdb6
Packit 98cdb6
  return parent;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Finds a common prefix based on the contents of the entry
Packit 98cdb6
 * and mandatorily appends it
Packit 98cdb6
 */
Packit 98cdb6
static void
Packit 98cdb6
explicitly_complete (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  chooser_entry->complete_on_load = FALSE;
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->completion_store)
Packit 98cdb6
    {
Packit 98cdb6
      char *completion, *text;
Packit 98cdb6
      gsize completion_len, text_len;
Packit 98cdb6
Packit 98cdb6
      text = gtk_file_chooser_entry_get_completion_text (chooser_entry);
Packit 98cdb6
      text_len = strlen (text);
Packit 98cdb6
      completion = _gtk_entry_completion_compute_prefix (gtk_entry_get_completion (GTK_ENTRY (chooser_entry)), text);
Packit 98cdb6
      completion_len = completion ? strlen (completion) : 0;
Packit 98cdb6
Packit 98cdb6
      if (completion_len > text_len)
Packit 98cdb6
        {
Packit 98cdb6
          GtkEditable *editable = GTK_EDITABLE (chooser_entry);
Packit 98cdb6
          int pos = gtk_editable_get_position (editable);
Packit 98cdb6
Packit 98cdb6
          gtk_editable_insert_text (editable,
Packit 98cdb6
                                    completion + text_len,
Packit 98cdb6
                                    completion_len - text_len,
Packit 98cdb6
                                    &pos;;
Packit 98cdb6
          gtk_editable_set_position (editable, pos);
Packit 98cdb6
          return;
Packit 98cdb6
        }
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gtk_widget_error_bell (GTK_WIDGET (chooser_entry));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_file_chooser_entry_grab_focus (GtkWidget *widget)
Packit 98cdb6
{
Packit 98cdb6
  GTK_WIDGET_CLASS (_gtk_file_chooser_entry_parent_class)->grab_focus (widget);
Packit 98cdb6
  _gtk_file_chooser_entry_select_filename (GTK_FILE_CHOOSER_ENTRY (widget));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
start_explicit_completion (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  if (chooser_entry->current_folder_loaded)
Packit 98cdb6
    explicitly_complete (chooser_entry);
Packit 98cdb6
  else
Packit 98cdb6
    set_complete_on_load (chooser_entry, TRUE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_file_chooser_entry_tab_handler (GtkWidget *widget,
Packit 98cdb6
				    GdkEventKey *event)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry;
Packit 98cdb6
  GtkEditable *editable;
Packit 98cdb6
  GdkModifierType state;
Packit 98cdb6
  gint start, end;
Packit 98cdb6
Packit 98cdb6
  chooser_entry = GTK_FILE_CHOOSER_ENTRY (widget);
Packit 98cdb6
  editable = GTK_EDITABLE (widget);
Packit 98cdb6
Packit 98cdb6
  if (!chooser_entry->eat_tabs)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  if (event->keyval != GDK_KEY_Tab)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  if (gtk_get_current_event_state (&state) &&
Packit 98cdb6
      (state & GDK_CONTROL_MASK) == GDK_CONTROL_MASK)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  /* This is a bit evil -- it makes Tab never leave the entry. It basically
Packit 98cdb6
   * makes it 'safe' for people to hit. */
Packit 98cdb6
  gtk_editable_get_selection_bounds (editable, &start, &end;;
Packit 98cdb6
      
Packit 98cdb6
  if (start != end)
Packit 98cdb6
    gtk_editable_set_position (editable, MAX (start, end));
Packit 98cdb6
  else
Packit 98cdb6
    start_explicit_completion (chooser_entry);
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_file_chooser_entry_focus_out_event (GtkWidget     *widget,
Packit 98cdb6
					GdkEventFocus *event)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry = GTK_FILE_CHOOSER_ENTRY (widget);
Packit 98cdb6
Packit 98cdb6
  set_complete_on_load (chooser_entry, FALSE);
Packit 98cdb6
 
Packit 98cdb6
  return GTK_WIDGET_CLASS (_gtk_file_chooser_entry_parent_class)->focus_out_event (widget, event);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
update_inline_completion (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  GtkEntryCompletion *completion = gtk_entry_get_completion (GTK_ENTRY (chooser_entry));
Packit 98cdb6
Packit 98cdb6
  if (!chooser_entry->current_folder_loaded)
Packit 98cdb6
    {
Packit 98cdb6
      gtk_entry_completion_set_inline_completion (completion, FALSE);
Packit 98cdb6
      return;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  switch (chooser_entry->action)
Packit 98cdb6
    {
Packit 98cdb6
    case GTK_FILE_CHOOSER_ACTION_OPEN:
Packit 98cdb6
    case GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER:
Packit 98cdb6
      gtk_entry_completion_set_inline_completion (completion, TRUE);
Packit 98cdb6
      break;
Packit 98cdb6
    case GTK_FILE_CHOOSER_ACTION_SAVE:
Packit 98cdb6
    case GTK_FILE_CHOOSER_ACTION_CREATE_FOLDER:
Packit 98cdb6
      gtk_entry_completion_set_inline_completion (completion, FALSE);
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
discard_completion_store (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  if (!chooser_entry->completion_store)
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  gtk_entry_completion_set_model (gtk_entry_get_completion (GTK_ENTRY (chooser_entry)), NULL);
Packit 98cdb6
  update_inline_completion (chooser_entry);
Packit 98cdb6
  g_object_unref (chooser_entry->completion_store);
Packit 98cdb6
  chooser_entry->completion_store = NULL;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
completion_store_set (GtkFileSystemModel  *model,
Packit 98cdb6
                      GFile               *file,
Packit 98cdb6
                      GFileInfo           *info,
Packit 98cdb6
                      int                  column,
Packit 98cdb6
                      GValue              *value,
Packit 98cdb6
                      gpointer             data)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry = data;
Packit 98cdb6
Packit 98cdb6
  const char *prefix = "";
Packit 98cdb6
  const char *suffix = "";
Packit 98cdb6
Packit 98cdb6
  switch (column)
Packit 98cdb6
    {
Packit 98cdb6
    case FULL_PATH_COLUMN:
Packit 98cdb6
      prefix = chooser_entry->dir_part;
Packit 98cdb6
      /* fall through */
Packit 98cdb6
    case DISPLAY_NAME_COLUMN:
Packit 98cdb6
      if (_gtk_file_info_consider_as_directory (info))
Packit 98cdb6
        suffix = G_DIR_SEPARATOR_S;
Packit 98cdb6
Packit 98cdb6
      g_value_take_string (value,
Packit 98cdb6
			   g_strconcat (prefix,
Packit 98cdb6
					g_file_info_get_display_name (info),
Packit 98cdb6
					suffix,
Packit 98cdb6
					NULL));
Packit 98cdb6
      break;
Packit 98cdb6
    default:
Packit 98cdb6
      g_assert_not_reached ();
Packit 98cdb6
      break;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Fills the completion store from the contents of the current folder */
Packit 98cdb6
static void
Packit 98cdb6
populate_completion_store (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  chooser_entry->completion_store = GTK_TREE_MODEL (
Packit 98cdb6
      _gtk_file_system_model_new_for_directory (chooser_entry->current_folder_file,
Packit 98cdb6
                                                "standard::name,standard::display-name,standard::type",
Packit 98cdb6
                                                completion_store_set,
Packit 98cdb6
                                                chooser_entry,
Packit 98cdb6
                                                N_COLUMNS,
Packit 98cdb6
                                                G_TYPE_STRING,
Packit 98cdb6
                                                G_TYPE_STRING));
Packit 98cdb6
  g_signal_connect (chooser_entry->completion_store, "finished-loading",
Packit 98cdb6
		    G_CALLBACK (finished_loading_cb), chooser_entry);
Packit 98cdb6
Packit 98cdb6
  _gtk_file_system_model_set_filter_folders (GTK_FILE_SYSTEM_MODEL (chooser_entry->completion_store),
Packit 98cdb6
                                             TRUE);
Packit 98cdb6
  _gtk_file_system_model_set_show_files (GTK_FILE_SYSTEM_MODEL (chooser_entry->completion_store),
Packit 98cdb6
                                         chooser_entry->action == GTK_FILE_CHOOSER_ACTION_OPEN ||
Packit 98cdb6
                                         chooser_entry->action == GTK_FILE_CHOOSER_ACTION_SAVE);
Packit 98cdb6
  gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (chooser_entry->completion_store),
Packit 98cdb6
					DISPLAY_NAME_COLUMN, GTK_SORT_ASCENDING);
Packit 98cdb6
Packit 98cdb6
  gtk_entry_completion_set_model (gtk_entry_get_completion (GTK_ENTRY (chooser_entry)),
Packit 98cdb6
				  chooser_entry->completion_store);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/* Callback when the current folder finishes loading */
Packit 98cdb6
static void
Packit 98cdb6
finished_loading_cb (GtkFileSystemModel  *model,
Packit 98cdb6
                     GError              *error,
Packit 98cdb6
		     GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  GtkEntryCompletion *completion;
Packit 98cdb6
Packit 98cdb6
  chooser_entry->current_folder_loaded = TRUE;
Packit 98cdb6
Packit 98cdb6
  if (error)
Packit 98cdb6
    {
Packit 98cdb6
      discard_completion_store (chooser_entry);
Packit 98cdb6
      set_complete_on_load (chooser_entry, FALSE);
Packit 98cdb6
      return;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->complete_on_load)
Packit 98cdb6
    explicitly_complete (chooser_entry);
Packit 98cdb6
Packit 98cdb6
  gtk_widget_set_tooltip_text (GTK_WIDGET (chooser_entry), NULL);
Packit 98cdb6
Packit 98cdb6
  completion = gtk_entry_get_completion (GTK_ENTRY (chooser_entry));
Packit 98cdb6
  update_inline_completion (chooser_entry);
Packit 98cdb6
Packit 98cdb6
  if (gtk_widget_has_focus (GTK_WIDGET (chooser_entry)))
Packit 98cdb6
    {
Packit 98cdb6
      gtk_entry_completion_complete (completion);
Packit 98cdb6
      gtk_entry_completion_insert_prefix (completion);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
set_completion_folder (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                       GFile               *folder_file,
Packit 98cdb6
		       char                *dir_part)
Packit 98cdb6
{
Packit 98cdb6
  if (folder_file &&
Packit 98cdb6
      chooser_entry->local_only
Packit 98cdb6
      && !_gtk_file_has_native_path (folder_file))
Packit 98cdb6
    folder_file = NULL;
Packit 98cdb6
Packit 98cdb6
  if (((chooser_entry->current_folder_file
Packit 98cdb6
	&& folder_file
Packit 98cdb6
	&& g_file_equal (folder_file, chooser_entry->current_folder_file))
Packit 98cdb6
       || chooser_entry->current_folder_file == folder_file)
Packit 98cdb6
      && g_strcmp0 (dir_part, chooser_entry->dir_part) == 0)
Packit 98cdb6
    {
Packit 98cdb6
      return;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->current_folder_file)
Packit 98cdb6
    {
Packit 98cdb6
      g_object_unref (chooser_entry->current_folder_file);
Packit 98cdb6
      chooser_entry->current_folder_file = NULL;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_free (chooser_entry->dir_part);
Packit 98cdb6
  chooser_entry->dir_part = g_strdup (dir_part);
Packit 98cdb6
  
Packit 98cdb6
  chooser_entry->current_folder_loaded = FALSE;
Packit 98cdb6
Packit 98cdb6
  discard_completion_store (chooser_entry);
Packit 98cdb6
Packit 98cdb6
  if (folder_file)
Packit 98cdb6
    {
Packit 98cdb6
      chooser_entry->current_folder_file = g_object_ref (folder_file);
Packit 98cdb6
      populate_completion_store (chooser_entry);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
refresh_current_folder_and_file_part (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  GFile *folder_file;
Packit 98cdb6
  char *text, *last_slash, *old_file_part;
Packit 98cdb6
  char *dir_part;
Packit 98cdb6
Packit 98cdb6
  old_file_part = chooser_entry->file_part;
Packit 98cdb6
Packit 98cdb6
  text = gtk_file_chooser_entry_get_completion_text (chooser_entry);
Packit 98cdb6
Packit 98cdb6
  last_slash = strrchr (text, G_DIR_SEPARATOR);
Packit 98cdb6
  if (last_slash)
Packit 98cdb6
    {
Packit 98cdb6
      dir_part = g_strndup (text, last_slash - text + 1);
Packit 98cdb6
      chooser_entry->file_part = g_strdup (last_slash + 1);
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      dir_part = g_strdup ("");
Packit 98cdb6
      chooser_entry->file_part = g_strdup (text);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  folder_file = gtk_file_chooser_get_directory_for_text (chooser_entry, text);
Packit 98cdb6
Packit 98cdb6
  set_completion_folder (chooser_entry, folder_file, dir_part);
Packit 98cdb6
Packit 98cdb6
  if (folder_file)
Packit 98cdb6
    g_object_unref (folder_file);
Packit 98cdb6
Packit 98cdb6
  g_free (dir_part);
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->completion_store &&
Packit 98cdb6
      (g_strcmp0 (old_file_part, chooser_entry->file_part) != 0))
Packit 98cdb6
    {
Packit 98cdb6
      GtkFileFilter *filter;
Packit 98cdb6
      char *pattern;
Packit 98cdb6
Packit 98cdb6
      filter = gtk_file_filter_new ();
Packit 98cdb6
      pattern = g_strconcat (chooser_entry->file_part, "*", NULL);
Packit 98cdb6
      gtk_file_filter_add_pattern (filter, pattern);
Packit 98cdb6
Packit 98cdb6
      g_object_ref_sink (filter);
Packit 98cdb6
Packit 98cdb6
      _gtk_file_system_model_set_filter (GTK_FILE_SYSTEM_MODEL (chooser_entry->completion_store),
Packit 98cdb6
                                         filter);
Packit 98cdb6
Packit 98cdb6
      g_free (pattern);
Packit 98cdb6
      g_object_unref (filter);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  g_free (text);
Packit 98cdb6
  g_free (old_file_part);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#ifdef G_OS_WIN32
Packit 98cdb6
static gint
Packit 98cdb6
insert_text_callback (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
		      const gchar	  *new_text,
Packit 98cdb6
		      gint       	   new_text_length,
Packit 98cdb6
		      gint       	  *position,
Packit 98cdb6
		      gpointer   	   user_data)
Packit 98cdb6
{
Packit 98cdb6
  const gchar *colon = memchr (new_text, ':', new_text_length);
Packit 98cdb6
  gint i;
Packit 98cdb6
Packit 98cdb6
  /* Disallow these characters altogether */
Packit 98cdb6
  for (i = 0; i < new_text_length; i++)
Packit 98cdb6
    {
Packit 98cdb6
      if (new_text[i] == '<' ||
Packit 98cdb6
	  new_text[i] == '>' ||
Packit 98cdb6
	  new_text[i] == '"' ||
Packit 98cdb6
	  new_text[i] == '|' ||
Packit 98cdb6
	  new_text[i] == '*' ||
Packit 98cdb6
	  new_text[i] == '?')
Packit 98cdb6
	break;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (i < new_text_length ||
Packit 98cdb6
      /* Disallow entering text that would cause a colon to be anywhere except
Packit 98cdb6
       * after a drive letter.
Packit 98cdb6
       */
Packit 98cdb6
      (colon != NULL &&
Packit 98cdb6
       *position + (colon - new_text) != 1) ||
Packit 98cdb6
      (new_text_length > 0 &&
Packit 98cdb6
       *position <= 1 &&
Packit 98cdb6
       gtk_entry_get_text_length (GTK_ENTRY (chooser_entry)) >= 2 &&
Packit 98cdb6
       gtk_entry_get_text (GTK_ENTRY (chooser_entry))[1] == ':'))
Packit 98cdb6
    {
Packit 98cdb6
      gtk_widget_error_bell (GTK_WIDGET (chooser_entry));
Packit 98cdb6
      g_signal_stop_emission_by_name (chooser_entry, "insert_text");
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
delete_text_callback (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
		      gint                 start_pos,
Packit 98cdb6
		      gint                 end_pos,
Packit 98cdb6
		      gpointer             user_data)
Packit 98cdb6
{
Packit 98cdb6
  /* If deleting a drive letter, delete the colon, too */
Packit 98cdb6
  if (start_pos == 0 && end_pos == 1 &&
Packit 98cdb6
      gtk_entry_get_text_length (GTK_ENTRY (chooser_entry)) >= 2 &&
Packit 98cdb6
      gtk_entry_get_text (GTK_ENTRY (chooser_entry))[1] == ':')
Packit 98cdb6
    {
Packit 98cdb6
      g_signal_handlers_block_by_func (chooser_entry,
Packit 98cdb6
				       G_CALLBACK (delete_text_callback),
Packit 98cdb6
				       user_data);
Packit 98cdb6
      gtk_editable_delete_text (GTK_EDITABLE (chooser_entry), 0, 1);
Packit 98cdb6
      g_signal_handlers_unblock_by_func (chooser_entry,
Packit 98cdb6
					 G_CALLBACK (delete_text_callback),
Packit 98cdb6
					 user_data);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_file_chooser_entry_new:
Packit 98cdb6
 * @eat_tabs: If %FALSE, allow focus navigation with the tab key.
Packit 98cdb6
 *
Packit 98cdb6
 * Creates a new #GtkFileChooserEntry object. #GtkFileChooserEntry
Packit 98cdb6
 * is an internal implementation widget for the GTK+ file chooser
Packit 98cdb6
 * which is an entry with completion with respect to a
Packit 98cdb6
 * #GtkFileSystem object.
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: the newly created #GtkFileChooserEntry
Packit 98cdb6
 **/
Packit 98cdb6
GtkWidget *
Packit 98cdb6
_gtk_file_chooser_entry_new (gboolean       eat_tabs)
Packit 98cdb6
{
Packit 98cdb6
  GtkFileChooserEntry *chooser_entry;
Packit 98cdb6
Packit 98cdb6
  chooser_entry = g_object_new (GTK_TYPE_FILE_CHOOSER_ENTRY, NULL);
Packit 98cdb6
  chooser_entry->eat_tabs = (eat_tabs != FALSE);
Packit 98cdb6
Packit 98cdb6
  return GTK_WIDGET (chooser_entry);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_file_chooser_entry_set_base_folder:
Packit 98cdb6
 * @chooser_entry: a #GtkFileChooserEntry
Packit 98cdb6
 * @file: file for a folder in the chooser entries current file system.
Packit 98cdb6
 *
Packit 98cdb6
 * Sets the folder with respect to which completions occur.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gtk_file_chooser_entry_set_base_folder (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
					 GFile               *file)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (GTK_IS_FILE_CHOOSER_ENTRY (chooser_entry));
Packit 98cdb6
  g_return_if_fail (file == NULL || G_IS_FILE (file));
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->base_folder == file ||
Packit 98cdb6
      (file != NULL && chooser_entry->base_folder != NULL 
Packit 98cdb6
       && g_file_equal (chooser_entry->base_folder, file)))
Packit 98cdb6
    return;
Packit 98cdb6
Packit 98cdb6
  if (file)
Packit 98cdb6
    g_object_ref (file);
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->base_folder)
Packit 98cdb6
    g_object_unref (chooser_entry->base_folder);
Packit 98cdb6
Packit 98cdb6
  chooser_entry->base_folder = file;
Packit 98cdb6
Packit 98cdb6
  refresh_current_folder_and_file_part (chooser_entry);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_file_chooser_entry_get_current_folder:
Packit 98cdb6
 * @chooser_entry: a #GtkFileChooserEntry
Packit 98cdb6
 *
Packit 98cdb6
 * Gets the current folder for the #GtkFileChooserEntry. If the
Packit 98cdb6
 * user has only entered a filename, this will be in the base folder
Packit 98cdb6
 * (see _gtk_file_chooser_entry_set_base_folder()), but if the
Packit 98cdb6
 * user has entered a relative or absolute path, then it will
Packit 98cdb6
 * be different.  If the user has entered unparsable text, or text which
Packit 98cdb6
 * the entry cannot handle, this will return %NULL.
Packit 98cdb6
 *
Packit 98cdb6
 * Return value: the file for the current folder - you must g_object_unref()
Packit 98cdb6
 *   the value after use.
Packit 98cdb6
 **/
Packit 98cdb6
GFile *
Packit 98cdb6
_gtk_file_chooser_entry_get_current_folder (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_FILE_CHOOSER_ENTRY (chooser_entry), NULL);
Packit 98cdb6
Packit 98cdb6
  return gtk_file_chooser_get_directory_for_text (chooser_entry,
Packit 98cdb6
                                                  gtk_entry_get_text (GTK_ENTRY (chooser_entry)));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_file_chooser_entry_get_file_part:
Packit 98cdb6
 * @chooser_entry: a #GtkFileChooserEntry
Packit 98cdb6
 *
Packit 98cdb6
 * Gets the non-folder portion of whatever the user has entered
Packit 98cdb6
 * into the file selector. What is returned is a UTF-8 string,
Packit 98cdb6
 * and if a filename path is needed, g_file_get_child_for_display_name()
Packit 98cdb6
 * must be used
Packit 98cdb6
  *
Packit 98cdb6
 * Return value: the entered filename - this value is owned by the
Packit 98cdb6
 *  chooser entry and must not be modified or freed.
Packit 98cdb6
 **/
Packit 98cdb6
const gchar *
Packit 98cdb6
_gtk_file_chooser_entry_get_file_part (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  const char *last_slash, *text;
Packit 98cdb6
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_FILE_CHOOSER_ENTRY (chooser_entry), NULL);
Packit 98cdb6
Packit 98cdb6
  text = gtk_entry_get_text (GTK_ENTRY (chooser_entry));
Packit 98cdb6
  last_slash = strrchr (text, G_DIR_SEPARATOR);
Packit 98cdb6
  if (last_slash)
Packit 98cdb6
    return last_slash + 1;
Packit 98cdb6
  else if (is_directory_shortcut (text))
Packit 98cdb6
    return "";
Packit 98cdb6
  else
Packit 98cdb6
    return text;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_file_chooser_entry_set_action:
Packit 98cdb6
 * @chooser_entry: a #GtkFileChooserEntry
Packit 98cdb6
 * @action: the action which is performed by the file selector using this entry
Packit 98cdb6
 *
Packit 98cdb6
 * Sets action which is performed by the file selector using this entry. 
Packit 98cdb6
 * The #GtkFileChooserEntry will use different completion strategies for 
Packit 98cdb6
 * different actions.
Packit 98cdb6
 **/
Packit 98cdb6
void
Packit 98cdb6
_gtk_file_chooser_entry_set_action (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
				    GtkFileChooserAction action)
Packit 98cdb6
{
Packit 98cdb6
  g_return_if_fail (GTK_IS_FILE_CHOOSER_ENTRY (chooser_entry));
Packit 98cdb6
  
Packit 98cdb6
  if (chooser_entry->action != action)
Packit 98cdb6
    {
Packit 98cdb6
      GtkEntryCompletion *comp;
Packit 98cdb6
Packit 98cdb6
      chooser_entry->action = action;
Packit 98cdb6
Packit 98cdb6
      comp = gtk_entry_get_completion (GTK_ENTRY (chooser_entry));
Packit 98cdb6
Packit 98cdb6
      /* FIXME: do we need to actually set the following? */
Packit 98cdb6
Packit 98cdb6
      switch (action)
Packit 98cdb6
	{
Packit 98cdb6
	case GTK_FILE_CHOOSER_ACTION_OPEN:
Packit 98cdb6
	case GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER:
Packit 98cdb6
	  gtk_entry_completion_set_popup_single_match (comp, FALSE);
Packit 98cdb6
	  break;
Packit 98cdb6
	case GTK_FILE_CHOOSER_ACTION_SAVE:
Packit 98cdb6
	case GTK_FILE_CHOOSER_ACTION_CREATE_FOLDER:
Packit 98cdb6
	  gtk_entry_completion_set_popup_single_match (comp, TRUE);
Packit 98cdb6
	  break;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (chooser_entry->completion_store)
Packit 98cdb6
        _gtk_file_system_model_set_show_files (GTK_FILE_SYSTEM_MODEL (chooser_entry->completion_store),
Packit 98cdb6
                                               action == GTK_FILE_CHOOSER_ACTION_OPEN ||
Packit 98cdb6
                                               action == GTK_FILE_CHOOSER_ACTION_SAVE);
Packit 98cdb6
Packit 98cdb6
      update_inline_completion (chooser_entry);
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
/**
Packit 98cdb6
 * _gtk_file_chooser_entry_get_action:
Packit 98cdb6
 * @chooser_entry: a #GtkFileChooserEntry
Packit 98cdb6
 *
Packit 98cdb6
 * Gets the action for this entry. 
Packit 98cdb6
 *
Packit 98cdb6
 * Returns: the action
Packit 98cdb6
 **/
Packit 98cdb6
GtkFileChooserAction
Packit 98cdb6
_gtk_file_chooser_entry_get_action (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  g_return_val_if_fail (GTK_IS_FILE_CHOOSER_ENTRY (chooser_entry),
Packit 98cdb6
			GTK_FILE_CHOOSER_ACTION_OPEN);
Packit 98cdb6
  
Packit 98cdb6
  return chooser_entry->action;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gboolean
Packit 98cdb6
_gtk_file_chooser_entry_get_is_folder (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
				       GFile               *file)
Packit 98cdb6
{
Packit 98cdb6
  GtkTreeIter iter;
Packit 98cdb6
  GFileInfo *info;
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->completion_store == NULL ||
Packit 98cdb6
      !_gtk_file_system_model_get_iter_for_file (GTK_FILE_SYSTEM_MODEL (chooser_entry->completion_store),
Packit 98cdb6
                                                 &iter,
Packit 98cdb6
                                                 file))
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  info = _gtk_file_system_model_get_info (GTK_FILE_SYSTEM_MODEL (chooser_entry->completion_store),
Packit 98cdb6
                                          &iter);
Packit 98cdb6
Packit 98cdb6
  return _gtk_file_info_consider_as_directory (info);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
Packit 98cdb6
/*
Packit 98cdb6
 * _gtk_file_chooser_entry_select_filename:
Packit 98cdb6
 * @chooser_entry: a #GtkFileChooserEntry
Packit 98cdb6
 *
Packit 98cdb6
 * Selects the filename (without the extension) for user edition.
Packit 98cdb6
 */
Packit 98cdb6
void
Packit 98cdb6
_gtk_file_chooser_entry_select_filename (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  const gchar *str, *ext;
Packit 98cdb6
  glong len = -1;
Packit 98cdb6
Packit 98cdb6
  if (chooser_entry->action == GTK_FILE_CHOOSER_ACTION_SAVE)
Packit 98cdb6
    {
Packit 98cdb6
      str = gtk_entry_get_text (GTK_ENTRY (chooser_entry));
Packit 98cdb6
      ext = g_strrstr (str, ".");
Packit 98cdb6
Packit 98cdb6
      if (ext)
Packit 98cdb6
       len = g_utf8_pointer_to_offset (str, ext);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  gtk_editable_select_region (GTK_EDITABLE (chooser_entry), 0, (gint) len);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
_gtk_file_chooser_entry_set_local_only (GtkFileChooserEntry *chooser_entry,
Packit 98cdb6
                                        gboolean             local_only)
Packit 98cdb6
{
Packit 98cdb6
  chooser_entry->local_only = local_only;
Packit 98cdb6
  refresh_current_folder_and_file_part (chooser_entry);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
gboolean
Packit 98cdb6
_gtk_file_chooser_entry_get_local_only (GtkFileChooserEntry *chooser_entry)
Packit 98cdb6
{
Packit 98cdb6
  return chooser_entry->local_only;
Packit 98cdb6
}