gnome-commander r1663 - in branches/gcmd-1-3: . doc/C src
- From: epiotr svn gnome org
- To: svn-commits-list gnome org
- Subject: gnome-commander r1663 - in branches/gcmd-1-3: . doc/C src
- Date: Wed, 26 Mar 2008 16:43:07 +0000 (GMT)
Author: epiotr
Date: Wed Mar 26 16:43:07 2008
New Revision: 1663
URL: http://svn.gnome.org/viewvc/gnome-commander?rev=1663&view=rev
Log:
Bugfixes + added accelerators to the search dlg
Modified:
branches/gcmd-1-3/ChangeLog
branches/gcmd-1-3/NEWS
branches/gcmd-1-3/doc/C/gnome-commander.xml
branches/gcmd-1-3/src/gnome-cmd-search-dialog.cc
Modified: branches/gcmd-1-3/NEWS
==============================================================================
--- branches/gcmd-1-3/NEWS (original)
+++ branches/gcmd-1-3/NEWS Wed Mar 26 16:43:07 2008
@@ -10,6 +10,7 @@
* Fixed problem #360175 (crash while entering a dir with %)
* Fixed problem #371948 and #388970 (file path not escaped)
* Fixed problem #412162 (build with --enable-python=no)
+ * Revamped search dialog
* Build fixes
New features:
Modified: branches/gcmd-1-3/doc/C/gnome-commander.xml
==============================================================================
--- branches/gcmd-1-3/doc/C/gnome-commander.xml (original)
+++ branches/gcmd-1-3/doc/C/gnome-commander.xml Wed Mar 26 16:43:07 2008
@@ -5830,6 +5830,9 @@
<para>Creating directories 'mkdir -p' like</para>
</listitem>
<listitem>
+ <para>Revamped search dialog</para>
+ </listitem>
+ <listitem>
<para>Updated help docs</para>
</listitem>
<listitem>
Modified: branches/gcmd-1-3/src/gnome-cmd-search-dialog.cc
==============================================================================
--- branches/gcmd-1-3/src/gnome-cmd-search-dialog.cc (original)
+++ branches/gcmd-1-3/src/gnome-cmd-search-dialog.cc Wed Mar 26 16:43:07 2008
@@ -36,7 +36,10 @@
static GnomeCmdDialogClass *parent_class = NULL;
-#define PBAR_MAX 50
+#define PBAR_MAX 50
+
+#define SEARCH_JUMP_SIZE 4096U
+#define SEARCH_BUFFER_SIZE (SEARCH_JUMP_SIZE * 10U)
struct ProtectedData
{
@@ -67,9 +70,18 @@
gboolean search_done;
gboolean stopped; // stops the search routine if set to TRUE. This is done by the stop_button
- gboolean dialog_destroyed; // set when the search dialog is destroyed, also stopps the search of course
+ gboolean dialog_destroyed; // set when the search dialog is destroyed, also stops the search of course
+
+ gchar *search_mem; // memory to search in the content of a file
};
+struct SearchFileData
+{
+ gchar *uri_str;
+ GnomeVFSHandle *handle;
+ gint offset;
+ guint len;
+};
struct _GnomeCmdSearchDialogPrivate
{
@@ -78,11 +90,9 @@
GnomeCmdCon *con;
GtkWidget *pattern_combo;
- GtkWidget *pattern_entry;
GtkWidget *dir_browser;
GtkWidget *dir_entry;
GtkWidget *find_text_combo;
- GtkWidget *find_text_entry;
GtkWidget *find_text_check;
GtkWidget *result_list;
GtkWidget *statusbar;
@@ -99,22 +109,6 @@
};
-static void on_search (GtkButton *button, GnomeCmdSearchDialog *dialog);
-
-
-static gboolean on_pattern_entry_keypressed (GtkWidget *entry, GdkEventKey *event, GnomeCmdSearchDialog *dialog)
-{
- if (state_is_blank (event->state) && event->keyval == GDK_Return)
- {
- on_search (NULL, dialog);
- event->keyval = GDK_Escape;
- return TRUE;
- }
-
- return FALSE;
-}
-
-
static void on_list_file_clicked (GnomeCmdFileList *fl, GnomeCmdFile *finfo, GdkEventButton *button, GnomeCmdSearchDialog *dialog)
{
if (dialog->priv->data->search_done)
@@ -134,76 +128,123 @@
data->context_id, msg);
}
+/**
+ *
+ */
+inline void search_file_data_free (SearchFileData *searchfile_data)
+{
+ if (searchfile_data->handle != NULL)
+ gnome_vfs_close (searchfile_data->handle);
+
+ g_free (searchfile_data->uri_str);
+ g_free (searchfile_data);
+}
/**
- * Loads a file and returns the content.
- * FIXME: Recode this so not the whole file has
- * to be loaded in memory.
+ * Loads a file in chunks and returns the content.
*/
-static gchar *load_file (GnomeCmdFile *finfo)
+static SearchFileData *read_search_file (SearchData *data,
+ SearchFileData *searchfile_data,
+ GnomeCmdFile *finfo)
{
g_return_val_if_fail (finfo != NULL, NULL);
g_return_val_if_fail (finfo->info != NULL, NULL);
- gchar *buf = (gchar *) g_malloc (finfo->info->size+1);
- gchar *uri_str = gnome_cmd_file_get_uri_str (finfo);
- GnomeVFSHandle *handle;
- GnomeVFSResult result = gnome_vfs_open (&handle, uri_str, GNOME_VFS_OPEN_READ);
+ GnomeVFSResult result;
- if (result != GNOME_VFS_OK)
+ if (searchfile_data == NULL)
+ {
+ searchfile_data = g_new0 (SearchFileData, 1);
+ searchfile_data->uri_str = gnome_cmd_file_get_uri_str (finfo);
+ result = gnome_vfs_open (&searchfile_data->handle, searchfile_data->uri_str, GNOME_VFS_OPEN_READ);
+
+ if (result != GNOME_VFS_OK)
+ {
+ warn_print (_("Failed to open file %s: %s\n"), searchfile_data->uri_str, gnome_vfs_result_to_string (result));
+ search_file_data_free (searchfile_data);
+ return NULL;
+ }
+ }
+
+ // If the stop button was pressed let's abort here
+ if (data->stopped)
{
- warn_print (_("Failed to open file %s: %s\n"), uri_str, gnome_vfs_result_to_string (result));
- g_free (uri_str);
- g_free (buf);
+ search_file_data_free (searchfile_data);
return NULL;
}
+ if (searchfile_data->len != NULL)
+ {
+ if ((searchfile_data->offset + searchfile_data->len) >= finfo->info->size)
+ { // end, all has been read
+ search_file_data_free (searchfile_data);
+ return NULL;
+ }
+ else
+ { // jump a big step backward to give the regex a chance
+ searchfile_data->offset += searchfile_data->len - SEARCH_JUMP_SIZE;
+ if (finfo->info->size < (searchfile_data->offset + (SEARCH_BUFFER_SIZE - 1)))
+ searchfile_data->len = finfo->info->size - searchfile_data->offset;
+ else
+ searchfile_data->len = SEARCH_BUFFER_SIZE - 1;
+ }
+ }
+ else
+ { // first time call of this function
+ if (finfo->info->size < (SEARCH_BUFFER_SIZE - 1))
+ searchfile_data->len = finfo->info->size;
+ else
+ searchfile_data->len = SEARCH_BUFFER_SIZE - 1;
+ }
+
GnomeVFSFileSize ret;
- result = gnome_vfs_read (handle, buf, finfo->info->size, &ret);
+ result = gnome_vfs_seek (searchfile_data->handle, GNOME_VFS_SEEK_START, searchfile_data->offset);
if (result != GNOME_VFS_OK)
{
- warn_print (_("Failed to read from file %s: %s\n"),
- uri_str, gnome_vfs_result_to_string (result));
- g_free (uri_str);
- g_free (buf);
- gnome_vfs_close (handle);
+ warn_print (_("Failed to seek in file %s: %s\n"), searchfile_data->uri_str, gnome_vfs_result_to_string (result));
+ search_file_data_free (searchfile_data);
return NULL;
}
+ result = gnome_vfs_read (searchfile_data->handle, data->search_mem, searchfile_data->len, &ret);
+ if (result != GNOME_VFS_OK)
+ {
+ warn_print (_("Failed to read from file %s: %s\n"), searchfile_data->uri_str, gnome_vfs_result_to_string (result));
+ search_file_data_free (searchfile_data);
+ return NULL;
+ }
+ data->search_mem[searchfile_data->len] = '\0';
- buf[ret] = '\0';
-
- g_free (uri_str);
- gnome_vfs_close (handle);
-
- return buf;
+ return searchfile_data;
}
/**
- * Determinates if the name of a file matches an regexp
+ * Determines if the content of a file matches an regexp
*
*/
-inline gboolean name_matches (gchar *name, SearchData *data)
+inline gboolean content_matches (GnomeCmdFile *finfo, SearchData *data)
{
- return filter_match (data->name_filter, name);
-}
+ gint ret = REG_NOMATCH;
+ if (finfo->info->size > 0)
+ {
+ regmatch_t match;
+ SearchFileData *search_file = NULL;
+
+ while ((search_file = read_search_file (data, search_file, finfo)) != NULL)
+ ret = regexec (data->content_regex, data->search_mem, 1, &match, 0);
+ }
+
+ return ret != REG_NOMATCH;
+}
/**
- * Determinates if the content of a file matches an regexp
+ * Determines if the name of a file matches an regexp
*
*/
-inline gboolean content_matches (GnomeCmdFile *finfo, SearchData *data)
+inline gboolean name_matches (gchar *name, SearchData *data)
{
- static regmatch_t match;
- gint ret = REG_NOMATCH;
- gchar *buf = load_file (finfo);
- if (buf)
- {
- ret = regexec (data->content_regex, buf, 1, &match, 0);
- g_free (buf);
- }
- return ret == 0;
+ return filter_match (data->name_filter, name);
}
@@ -389,6 +430,9 @@
gtk_widget_set_sensitive (data->dialog->priv->search_button, TRUE);
gtk_widget_set_sensitive (data->dialog->priv->stop_button, FALSE);
+ // set focus to result list
+ gtk_widget_grab_focus (data->dialog->priv->result_list);
+
gtk_widget_hide (data->dialog->priv->pbar);
}
@@ -411,6 +455,11 @@
if (data->thread)
g_thread_join (data->thread);
+ if (data->pdata.mutex != NULL)
+ g_mutex_free (data->pdata.mutex);
+
+ g_free (data->search_mem);
+
g_free (data);
return FALSE;
@@ -468,8 +517,8 @@
}
data->dialog = dialog;
- data->name_pattern = gtk_entry_get_text (GTK_ENTRY (dialog->priv->pattern_entry));
- data->content_pattern = gtk_entry_get_text (GTK_ENTRY (dialog->priv->find_text_entry));
+ data->name_pattern = gtk_combo_box_get_active_text (GTK_COMBO_BOX (dialog->priv->pattern_combo));
+ data->content_pattern = gtk_combo_box_get_active_text (GTK_COMBO_BOX (dialog->priv->find_text_combo));
data->dir = gtk_entry_get_text (GTK_ENTRY (dialog->priv->dir_entry));
data->context_id = gtk_statusbar_get_context_id (GTK_STATUSBAR (data->dialog->priv->statusbar), "info");
data->content_search = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (dialog->priv->find_text_check));
@@ -497,14 +546,15 @@
// create an re for filename matching
data->name_filter = filter_new (data->name_pattern, data->case_sens, gnome_cmd_data_get_filter_type ());
-
// if we're going to search through file content create an re for that too
if (data->content_search)
{
data->content_regex = g_new0 (regex_t, 1);
- regcomp (data->content_regex, data->content_pattern, 0);
+ regcomp (data->content_regex, data->content_pattern, data->case_sens ? 0 : REG_ICASE);
}
+ if (dialog->priv->data->search_mem == NULL)
+ dialog->priv->data->search_mem = (gchar *) g_malloc(SEARCH_BUFFER_SIZE);
// start the search
path = gnome_cmd_con_create_path (dialog->priv->con, data->dir);
@@ -512,7 +562,10 @@
gnome_cmd_dir_ref (data->start_dir);
data->search_done = FALSE;
- data->pdata.mutex = g_mutex_new ();
+
+ if (data->pdata.mutex == NULL)
+ data->pdata.mutex = g_mutex_new ();
+
data->thread = g_thread_create ((GThreadFunc)perform_search_operation, data, TRUE, NULL);
gtk_widget_show (data->dialog->priv->pbar);
@@ -613,13 +666,19 @@
return FALSE;
}
-
-static gboolean on_list_keypressed (GnomeCmdFileList *fl, GdkEventKey *event, GnomeCmdSearchDialog *dialog)
+/*
+ *
+ */
+static gboolean on_list_keypressed (GtkWidget *result_list, GdkEventKey *event, gpointer dialog)
{
- if (gnome_cmd_file_list_keypressed (fl, event) || handle_list_keypress (fl, event, dialog))
- stop_kp (GTK_OBJECT (fl));
+ if (gnome_cmd_file_list_keypressed (GNOME_CMD_FILE_LIST (result_list), event) ||
+ handle_list_keypress (GNOME_CMD_FILE_LIST (result_list), event, GNOME_CMD_SEARCH_DIALOG (dialog)))
+ {
+ stop_kp (GTK_OBJECT (result_list));
+ return TRUE;
+ }
- return TRUE;
+ return FALSE;
}
@@ -632,7 +691,7 @@
if (gtk_toggle_button_get_active (togglebutton))
{
gtk_widget_set_sensitive (dialog->priv->find_text_combo, TRUE);
- gtk_widget_grab_focus (dialog->priv->find_text_entry);
+ gtk_widget_grab_focus (dialog->priv->find_text_combo);
}
else
gtk_widget_set_sensitive (dialog->priv->find_text_combo, FALSE);
@@ -646,8 +705,12 @@
static void destroy (GtkObject *object)
{
GnomeCmdSearchDialog *dialog = GNOME_CMD_SEARCH_DIALOG (object);
- if (dialog->priv)
+
+ if (dialog->priv != NULL)
+ {
g_free (dialog->priv);
+ dialog->priv = NULL;
+ }
if (GTK_OBJECT_CLASS (parent_class)->destroy)
(*GTK_OBJECT_CLASS (parent_class)->destroy) (object);
@@ -673,6 +736,64 @@
widget_class->map = ::map;
}
+/*
+ * create a label with keyboard shortcut and a widget to activate
+ * if shortcut is pressed.
+ */
+inline GtkWidget *create_label_with_mnemonic (GtkWidget *parent, const gchar *text, GtkWidget *for_widget)
+{
+ GtkWidget *label;
+
+ label = gtk_label_new_with_mnemonic (text);
+ if (for_widget != NULL)
+ gtk_label_set_mnemonic_widget (GTK_LABEL (label), for_widget);
+
+ gtk_widget_ref (label);
+ gtk_object_set_data_full (GTK_OBJECT (parent), "label", label,
+ (GtkDestroyNotify) gtk_widget_unref);
+ gtk_widget_show (label);
+ gtk_misc_set_alignment (GTK_MISC (label), 0, 0.5);
+
+ return label;
+}
+
+/*
+ * create a check_box_button with keyboard shortcut
+ */
+inline GtkWidget *create_check_with_mnemonic (GtkWidget *parent, gchar *text, gchar *name)
+{
+ GtkWidget *btn = gtk_check_button_new_with_mnemonic (text);
+
+ gtk_widget_ref (btn);
+ gtk_object_set_data_full (GTK_OBJECT (parent), name, btn, (GtkDestroyNotify) gtk_widget_unref);
+ gtk_widget_show (btn);
+
+ return btn;
+}
+
+/*
+ * create a gtk_combo_box_entry_new_text.
+ * gtk_combo is deprecated.
+ */
+inline GtkWidget *create_combo_box_entry (GtkWidget *parent)
+{
+ GtkWidget *combo = gtk_combo_box_entry_new_text ();
+ gtk_widget_ref (combo);
+ gtk_object_set_data_full (GTK_OBJECT (parent), "combo", combo,
+ (GtkDestroyNotify) gtk_widget_unref);
+ gtk_widget_show (combo);
+ return combo;
+}
+
+/*
+ * callback function for 'g_list_foreach' to add default value
+ * to dropdownbox
+ */
+static void combo_box_insert_text (gpointer data, gpointer user_data)
+{
+ gtk_combo_box_append_text (GTK_COMBO_BOX (user_data), (gchar *) data);
+}
+
static void init (GnomeCmdSearchDialog *dialog)
{
@@ -687,13 +808,6 @@
dialog->priv = g_new0 (GnomeCmdSearchDialogPrivate, 1);
dialog->priv->data = g_new0 (SearchData, 1);
- dialog->priv->data->match_dirs = NULL;
- dialog->priv->data->thread = NULL;
- dialog->priv->data->search_done = TRUE;
-
- dialog->priv->data->pdata.files = NULL;
- dialog->priv->data->pdata.msg = NULL;
- dialog->priv->data->pdata.mutex = NULL;
window = GTK_WIDGET (dialog);
gtk_object_set_data (GTK_OBJECT (window), "window", window);
@@ -710,22 +824,22 @@
gtk_box_pack_start (GTK_BOX (vbox1), table, FALSE, TRUE, 0);
// Search for
- label = create_label (window, _("Search for: "));
+ dialog->priv->pattern_combo = create_combo_box_entry (window);
+ label = create_label_with_mnemonic (window, _("Search _for: "), dialog->priv->pattern_combo);
table_add (table, label, 0, 0, GTK_FILL);
- dialog->priv->pattern_combo = create_combo (window);
table_add (table, dialog->priv->pattern_combo, 1, 0, (GtkAttachOptions) (GTK_EXPAND|GTK_FILL));
if (defaults->name_patterns)
- gtk_combo_set_popdown_strings (GTK_COMBO (dialog->priv->pattern_combo), defaults->name_patterns);
+ g_list_foreach (defaults->name_patterns, combo_box_insert_text, dialog->priv->pattern_combo);
- dialog->priv->pattern_entry = GTK_COMBO (dialog->priv->pattern_combo)->entry;
- gnome_cmd_dialog_editable_enters (GNOME_CMD_DIALOG (dialog), GTK_EDITABLE (dialog->priv->pattern_entry));
+ gtk_combo_box_set_active (GTK_COMBO_BOX (dialog->priv->pattern_combo), 0);
+ gnome_cmd_dialog_editable_enters (GNOME_CMD_DIALOG (dialog), GTK_EDITABLE (gtk_bin_get_child (GTK_BIN (dialog->priv->pattern_combo))));
// Search in
- label = create_label (window, _("Search in: "));
+ dialog->priv->dir_browser = create_file_entry (window, "dir_browser", "");
+ label = create_label_with_mnemonic (window, _("Search _in: "), dialog->priv->dir_browser);
table_add (table, label, 0, 1, GTK_FILL);
- dialog->priv->dir_browser = create_file_entry (window, "dir_browser", "");
table_add (table, dialog->priv->dir_browser, 1, 1, (GtkAttachOptions) (GTK_EXPAND|GTK_FILL));
if (defaults->directories)
gtk_combo_set_popdown_strings (
@@ -736,28 +850,28 @@
gnome_file_entry_gtk_entry (GNOME_FILE_ENTRY (dialog->priv->dir_browser));
// Find text
- dialog->priv->find_text_check = create_check (window, _("Find text: "), "find_text");
+ dialog->priv->find_text_check = create_check_with_mnemonic (window, _("Find _text: "), "find_text");
table_add (table, dialog->priv->find_text_check, 0, 2, GTK_FILL);
- dialog->priv->find_text_combo = create_combo (window);
+ dialog->priv->find_text_combo = create_combo_box_entry (window);
table_add (table, dialog->priv->find_text_combo, 1, 2, (GtkAttachOptions) (GTK_EXPAND|GTK_FILL));
gtk_widget_set_sensitive (dialog->priv->find_text_combo, FALSE);
if (defaults->content_patterns)
- gtk_combo_set_popdown_strings (GTK_COMBO (dialog->priv->find_text_combo), defaults->content_patterns);
+ g_list_foreach (defaults->content_patterns, combo_box_insert_text, dialog->priv->find_text_combo);
- dialog->priv->find_text_entry = GTK_COMBO (dialog->priv->find_text_combo)->entry;
- gnome_cmd_dialog_editable_enters (GNOME_CMD_DIALOG (dialog), GTK_EDITABLE (dialog->priv->find_text_entry));
+ gtk_combo_box_set_active (GTK_COMBO_BOX (dialog->priv->find_text_combo), 0);
+ gnome_cmd_dialog_editable_enters (GNOME_CMD_DIALOG (dialog), GTK_EDITABLE (gtk_bin_get_child (GTK_BIN (dialog->priv->find_text_combo))));
// Recurse check
- dialog->priv->recurse_check = create_check (window, _("Search Recursively"), "recurse_check");
+ dialog->priv->recurse_check = create_check_with_mnemonic (window, _("Search _recursively"), "recurse_check");
gtk_table_attach (GTK_TABLE (table), dialog->priv->recurse_check, 0, 2, 3, 4,
(GtkAttachOptions) (GTK_FILL),
(GtkAttachOptions) (0), 0, 0);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (dialog->priv->recurse_check), defaults->recursive);
// Case check
- dialog->priv->case_check = create_check (window, _("Case Sensitive"), "case_check");
+ dialog->priv->case_check = create_check_with_mnemonic (window, _("_Case sensitive"), "case_check");
gtk_table_attach (GTK_TABLE (table), dialog->priv->case_check, 0, 2, 4, 5,
(GtkAttachOptions) (GTK_FILL),
(GtkAttachOptions) (0), 0, 0);
@@ -808,17 +922,16 @@
dialog->priv->pbar = pbar;
- gtk_signal_connect (GTK_OBJECT (dialog), "destroy", GTK_SIGNAL_FUNC (on_dialog_destroy), NULL);
- gtk_signal_connect (GTK_OBJECT (dialog), "size-allocate", GTK_SIGNAL_FUNC (on_dialog_size_allocate), NULL);
- gtk_signal_connect (GTK_OBJECT (dialog->priv->result_list), "key-press-event", GTK_SIGNAL_FUNC (on_list_keypressed), dialog);
- gtk_signal_connect (GTK_OBJECT (dialog->priv->result_list), "file-clicked", GTK_SIGNAL_FUNC (on_list_file_clicked), dialog);
-
- gtk_signal_connect (GTK_OBJECT (dialog->priv->pattern_entry),
- "key-press-event",
- GTK_SIGNAL_FUNC (on_pattern_entry_keypressed), dialog);
+ g_signal_connect (G_OBJECT (dialog), "destroy", G_CALLBACK (on_dialog_destroy), NULL);
+ g_signal_connect (G_OBJECT (dialog), "size-allocate", G_CALLBACK (on_dialog_size_allocate), NULL);
+ g_signal_connect (G_OBJECT (dialog->priv->result_list), "key-press-event", G_CALLBACK (on_list_keypressed), dialog);
+ g_signal_connect (G_OBJECT (dialog->priv->result_list), "file-clicked", G_CALLBACK (on_list_file_clicked), dialog);
+
gtk_signal_connect (GTK_OBJECT (dialog->priv->find_text_check), "toggled", GTK_SIGNAL_FUNC (find_text_toggled), dialog);
- gtk_widget_grab_focus (dialog->priv->pattern_entry);
+ gtk_window_set_keep_above (GTK_WINDOW (dialog), FALSE);
+
+ gtk_widget_grab_focus (dialog->priv->pattern_combo);
gnome_cmd_file_list_update_style (GNOME_CMD_FILE_LIST (dialog->priv->result_list));
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]