2006-07-13 14:30:44 +00:00
|
|
|
/*
|
2006-08-20 11:54:33 +00:00
|
|
|
* search.c - this file is part of Geany, a fast and lightweight IDE
|
2006-07-13 14:30:44 +00:00
|
|
|
*
|
2012-06-18 01:13:05 +02:00
|
|
|
* Copyright 2006-2012 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
|
|
|
|
* Copyright 2006-2012 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
|
2006-07-13 14:30:44 +00:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2012-08-24 19:25:57 +02:00
|
|
|
* 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.
|
2006-07-13 14:30:44 +00:00
|
|
|
*/
|
|
|
|
|
2007-02-24 11:41:56 +00:00
|
|
|
/*
|
|
|
|
* Find, Replace, Find in Files dialog related functions.
|
|
|
|
* Note that the basic text find functions are in document.c.
|
|
|
|
*/
|
|
|
|
|
2006-09-02 23:28:34 +00:00
|
|
|
#include <gdk/gdkkeysyms.h>
|
|
|
|
|
2006-07-13 14:30:44 +00:00
|
|
|
#include "geany.h"
|
|
|
|
#include "search.h"
|
2007-08-23 11:34:06 +00:00
|
|
|
#include "prefs.h"
|
2006-07-13 14:30:44 +00:00
|
|
|
#include "support.h"
|
|
|
|
#include "utils.h"
|
2006-08-11 21:12:49 +00:00
|
|
|
#include "document.h"
|
2008-06-12 20:09:57 +00:00
|
|
|
#include "msgwindow.h"
|
2006-08-11 21:12:49 +00:00
|
|
|
#include "sciwrappers.h"
|
2006-10-22 14:56:05 +00:00
|
|
|
#include "ui_utils.h"
|
2007-05-28 16:07:30 +00:00
|
|
|
#include "editor.h"
|
2008-11-13 16:29:56 +00:00
|
|
|
#include "encodings.h"
|
2008-11-16 17:53:33 +00:00
|
|
|
#include "project.h"
|
2008-12-16 13:01:47 +00:00
|
|
|
#include "keyfile.h"
|
|
|
|
#include "stash.h"
|
2010-09-21 16:16:15 +00:00
|
|
|
#include "toolbar.h"
|
2006-07-13 14:30:44 +00:00
|
|
|
|
|
|
|
#include <unistd.h>
|
2006-08-11 21:12:49 +00:00
|
|
|
#include <string.h>
|
2010-03-02 13:21:41 +00:00
|
|
|
#include <ctype.h>
|
2006-07-13 14:30:44 +00:00
|
|
|
|
|
|
|
#ifdef G_OS_UNIX
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <sys/wait.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2008-12-16 13:01:47 +00:00
|
|
|
enum
|
|
|
|
{
|
2006-11-01 15:26:41 +00:00
|
|
|
GEANY_RESPONSE_FIND = 1,
|
|
|
|
GEANY_RESPONSE_FIND_PREVIOUS,
|
|
|
|
GEANY_RESPONSE_FIND_IN_FILE,
|
|
|
|
GEANY_RESPONSE_FIND_IN_SESSION,
|
|
|
|
GEANY_RESPONSE_MARK,
|
|
|
|
GEANY_RESPONSE_REPLACE,
|
2006-10-22 14:56:05 +00:00
|
|
|
GEANY_RESPONSE_REPLACE_AND_FIND,
|
|
|
|
GEANY_RESPONSE_REPLACE_IN_SESSION,
|
|
|
|
GEANY_RESPONSE_REPLACE_IN_FILE,
|
|
|
|
GEANY_RESPONSE_REPLACE_IN_SEL
|
2006-08-11 21:12:49 +00:00
|
|
|
};
|
|
|
|
|
2006-09-20 11:03:33 +00:00
|
|
|
|
2011-04-30 21:50:37 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
FILES_MODE_ALL,
|
|
|
|
FILES_MODE_PROJECT,
|
|
|
|
FILES_MODE_CUSTOM
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
GeanySearchData search_data;
|
2008-05-16 12:08:39 +00:00
|
|
|
GeanySearchPrefs search_prefs;
|
|
|
|
|
2007-04-15 15:59:57 +00:00
|
|
|
|
2008-12-16 13:01:47 +00:00
|
|
|
static struct
|
|
|
|
{
|
2010-06-21 14:57:03 +00:00
|
|
|
gboolean fif_regexp;
|
2009-05-17 15:46:09 +00:00
|
|
|
gboolean fif_case_sensitive;
|
|
|
|
gboolean fif_match_whole_word;
|
|
|
|
gboolean fif_invert_results;
|
|
|
|
gboolean fif_recursive;
|
2009-05-17 17:49:18 +00:00
|
|
|
gboolean fif_use_extra_options;
|
2010-06-23 12:34:11 +00:00
|
|
|
gchar *fif_extra_options;
|
2011-04-30 21:50:37 +00:00
|
|
|
gint fif_files_mode;
|
2010-06-23 12:34:11 +00:00
|
|
|
gchar *fif_files;
|
2011-03-06 14:18:05 +00:00
|
|
|
gboolean find_regexp;
|
|
|
|
gboolean find_escape_sequences;
|
|
|
|
gboolean find_case_sensitive;
|
|
|
|
gboolean find_match_whole_word;
|
|
|
|
gboolean find_match_word_start;
|
|
|
|
gboolean find_close_dialog;
|
|
|
|
gboolean replace_regexp;
|
|
|
|
gboolean replace_escape_sequences;
|
|
|
|
gboolean replace_case_sensitive;
|
|
|
|
gboolean replace_match_whole_word;
|
|
|
|
gboolean replace_match_word_start;
|
|
|
|
gboolean replace_search_backwards;
|
|
|
|
gboolean replace_close_dialog;
|
2008-12-16 13:01:47 +00:00
|
|
|
}
|
2011-03-06 14:18:05 +00:00
|
|
|
settings;
|
2008-12-16 13:01:47 +00:00
|
|
|
|
2010-03-18 17:04:17 +00:00
|
|
|
static StashGroup *fif_prefs = NULL;
|
2011-03-06 14:18:05 +00:00
|
|
|
static StashGroup *find_prefs = NULL;
|
|
|
|
static StashGroup *replace_prefs = NULL;
|
2008-12-16 13:01:47 +00:00
|
|
|
|
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
static struct
|
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
GtkWidget *dialog;
|
2009-01-27 18:12:00 +00:00
|
|
|
GtkWidget *entry;
|
2009-01-27 18:03:58 +00:00
|
|
|
gboolean all_expanded;
|
2009-12-29 18:45:49 +00:00
|
|
|
gint position[2]; /* x, y */
|
2008-11-06 17:13:18 +00:00
|
|
|
}
|
2009-12-29 18:45:49 +00:00
|
|
|
find_dlg = {NULL, NULL, FALSE, {0, 0}};
|
2008-11-06 17:13:18 +00:00
|
|
|
|
2009-01-27 17:50:45 +00:00
|
|
|
static struct
|
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
GtkWidget *dialog;
|
|
|
|
GtkWidget *find_entry;
|
2009-02-27 14:05:50 +00:00
|
|
|
GtkWidget *replace_entry;
|
2009-01-27 18:03:58 +00:00
|
|
|
gboolean all_expanded;
|
2009-12-29 18:45:49 +00:00
|
|
|
gint position[2]; /* x, y */
|
2009-01-27 17:50:45 +00:00
|
|
|
}
|
2009-12-29 18:45:49 +00:00
|
|
|
replace_dlg = {NULL, NULL, NULL, FALSE, {0, 0}};
|
2009-01-27 17:50:45 +00:00
|
|
|
|
2009-01-22 17:19:15 +00:00
|
|
|
static struct
|
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
GtkWidget *dialog;
|
|
|
|
GtkWidget *dir_combo;
|
2010-06-23 16:06:10 +00:00
|
|
|
GtkWidget *files_combo;
|
2009-01-27 18:03:58 +00:00
|
|
|
GtkWidget *search_combo;
|
|
|
|
GtkWidget *encoding_combo;
|
2011-04-30 21:50:37 +00:00
|
|
|
GtkWidget *files_mode_combo;
|
2009-12-29 18:45:49 +00:00
|
|
|
gint position[2]; /* x, y */
|
2009-01-22 17:19:15 +00:00
|
|
|
}
|
2011-04-30 21:50:37 +00:00
|
|
|
fif_dlg = {NULL, NULL, NULL, NULL, NULL, NULL, {0, 0}};
|
2006-08-11 21:12:49 +00:00
|
|
|
|
|
|
|
|
2008-11-13 16:29:56 +00:00
|
|
|
static gboolean search_read_io(GIOChannel *source, GIOCondition condition, gpointer data);
|
2008-11-16 17:53:13 +00:00
|
|
|
static gboolean search_read_io_stderr(GIOChannel *source, GIOCondition condition, gpointer data);
|
2006-07-13 14:30:44 +00:00
|
|
|
|
|
|
|
static void search_close_pid(GPid child_pid, gint status, gpointer user_data);
|
|
|
|
|
|
|
|
static gchar **search_get_argv(const gchar **argv_prefix, const gchar *dir);
|
|
|
|
|
2012-01-03 16:21:12 +00:00
|
|
|
static GRegex *compile_regex(const gchar *str, gint sflags);
|
|
|
|
|
2006-07-13 14:30:44 +00:00
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
static void
|
|
|
|
on_find_replace_checkbutton_toggled(GtkToggleButton *togglebutton, gpointer user_data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_find_dialog_response(GtkDialog *dialog, gint response, gpointer user_data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_find_entry_activate(GtkEntry *entry, gpointer user_data);
|
|
|
|
|
2011-03-24 16:51:02 +00:00
|
|
|
static void
|
|
|
|
on_find_entry_activate_backward(GtkEntry *entry, gpointer user_data);
|
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
static void
|
|
|
|
on_replace_dialog_response(GtkDialog *dialog, gint response, gpointer user_data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_replace_entry_activate(GtkEntry *entry, gpointer user_data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_find_in_files_dialog_response(GtkDialog *dialog, gint response, gpointer user_data);
|
|
|
|
|
2006-09-20 11:03:33 +00:00
|
|
|
static gboolean
|
2008-11-13 16:29:56 +00:00
|
|
|
search_find_in_files(const gchar *utf8_search_text, const gchar *dir, const gchar *opts,
|
|
|
|
const gchar *enc);
|
2006-09-20 11:03:33 +00:00
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2008-12-16 13:01:47 +00:00
|
|
|
static void init_prefs(void)
|
|
|
|
{
|
2010-03-18 17:04:17 +00:00
|
|
|
StashGroup *group;
|
2008-12-16 13:01:47 +00:00
|
|
|
|
|
|
|
group = stash_group_new("search");
|
|
|
|
configuration_add_pref_group(group, TRUE);
|
2011-12-05 21:24:33 +02:00
|
|
|
stash_group_add_toggle_button(group, &search_prefs.always_wrap,
|
|
|
|
"pref_search_hide_find_dialog", FALSE, "check_always_wrap_search");
|
|
|
|
stash_group_add_toggle_button(group, &search_prefs.hide_find_dialog,
|
|
|
|
"pref_search_always_wrap", FALSE, "check_hide_find_dialog");
|
2008-12-16 13:01:47 +00:00
|
|
|
stash_group_add_toggle_button(group, &search_prefs.use_current_file_dir,
|
|
|
|
"pref_search_current_file_dir", TRUE, "check_fif_current_dir");
|
2009-01-27 17:50:45 +00:00
|
|
|
stash_group_add_boolean(group, &find_dlg.all_expanded, "find_all_expanded", FALSE);
|
2009-01-27 18:03:58 +00:00
|
|
|
stash_group_add_boolean(group, &replace_dlg.all_expanded, "replace_all_expanded", FALSE);
|
2009-12-29 18:45:49 +00:00
|
|
|
/* dialog positions */
|
|
|
|
stash_group_add_integer(group, &find_dlg.position[0], "position_find_x", -1);
|
|
|
|
stash_group_add_integer(group, &find_dlg.position[1], "position_find_y", -1);
|
|
|
|
stash_group_add_integer(group, &replace_dlg.position[0], "position_replace_x", -1);
|
|
|
|
stash_group_add_integer(group, &replace_dlg.position[1], "position_replace_y", -1);
|
|
|
|
stash_group_add_integer(group, &fif_dlg.position[0], "position_fif_x", -1);
|
|
|
|
stash_group_add_integer(group, &fif_dlg.position[1], "position_fif_y", -1);
|
2008-12-16 13:01:47 +00:00
|
|
|
|
2011-03-06 14:36:42 +00:00
|
|
|
memset(&settings, '\0', sizeof(settings));
|
2011-03-06 14:18:05 +00:00
|
|
|
|
2008-12-16 13:01:47 +00:00
|
|
|
group = stash_group_new("search");
|
|
|
|
fif_prefs = group;
|
|
|
|
configuration_add_pref_group(group, FALSE);
|
2010-06-21 14:57:03 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.fif_regexp,
|
|
|
|
"fif_regexp", FALSE, "check_regexp");
|
2009-05-17 15:46:09 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.fif_case_sensitive,
|
|
|
|
"fif_case_sensitive", TRUE, "check_case");
|
|
|
|
stash_group_add_toggle_button(group, &settings.fif_match_whole_word,
|
|
|
|
"fif_match_whole_word", FALSE, "check_wholeword");
|
|
|
|
stash_group_add_toggle_button(group, &settings.fif_invert_results,
|
|
|
|
"fif_invert_results", FALSE, "check_invert");
|
|
|
|
stash_group_add_toggle_button(group, &settings.fif_recursive,
|
|
|
|
"fif_recursive", FALSE, "check_recursive");
|
2010-06-23 12:34:11 +00:00
|
|
|
stash_group_add_entry(group, &settings.fif_extra_options,
|
2010-07-06 14:20:12 +00:00
|
|
|
"fif_extra_options", "", "entry_extra");
|
2009-05-17 17:49:18 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.fif_use_extra_options,
|
|
|
|
"fif_use_extra_options", FALSE, "check_extra");
|
2010-06-23 12:34:11 +00:00
|
|
|
stash_group_add_entry(group, &settings.fif_files,
|
|
|
|
"fif_files", "", "entry_files");
|
2011-04-30 21:50:37 +00:00
|
|
|
stash_group_add_combo_box(group, &settings.fif_files_mode,
|
|
|
|
"fif_files_mode", FILES_MODE_ALL, "combo_files_mode");
|
2011-03-06 14:18:05 +00:00
|
|
|
|
|
|
|
group = stash_group_new("search");
|
|
|
|
find_prefs = group;
|
|
|
|
configuration_add_pref_group(group, FALSE);
|
|
|
|
stash_group_add_toggle_button(group, &settings.find_regexp,
|
|
|
|
"find_regexp", FALSE, "check_regexp");
|
2011-04-08 16:49:38 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.find_case_sensitive,
|
|
|
|
"find_case_sensitive", FALSE, "check_case");
|
2011-03-06 14:18:05 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.find_escape_sequences,
|
|
|
|
"find_escape_sequences", FALSE, "check_escape");
|
|
|
|
stash_group_add_toggle_button(group, &settings.find_match_whole_word,
|
|
|
|
"find_match_whole_word", FALSE, "check_word");
|
|
|
|
stash_group_add_toggle_button(group, &settings.find_match_word_start,
|
|
|
|
"find_match_word_start", FALSE, "check_wordstart");
|
|
|
|
stash_group_add_toggle_button(group, &settings.find_close_dialog,
|
|
|
|
"find_close_dialog", TRUE, "check_close");
|
|
|
|
|
|
|
|
group = stash_group_new("search");
|
|
|
|
replace_prefs = group;
|
|
|
|
configuration_add_pref_group(group, FALSE);
|
|
|
|
stash_group_add_toggle_button(group, &settings.replace_regexp,
|
|
|
|
"replace_regexp", FALSE, "check_regexp");
|
2011-04-08 16:49:38 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.replace_case_sensitive,
|
|
|
|
"replace_case_sensitive", FALSE, "check_case");
|
2011-03-06 14:18:05 +00:00
|
|
|
stash_group_add_toggle_button(group, &settings.replace_escape_sequences,
|
|
|
|
"replace_escape_sequences", FALSE, "check_escape");
|
|
|
|
stash_group_add_toggle_button(group, &settings.replace_match_whole_word,
|
|
|
|
"replace_match_whole_word", FALSE, "check_word");
|
|
|
|
stash_group_add_toggle_button(group, &settings.replace_match_word_start,
|
|
|
|
"replace_match_word_start", FALSE, "check_wordstart");
|
|
|
|
stash_group_add_toggle_button(group, &settings.replace_search_backwards,
|
|
|
|
"replace_search_backwards", FALSE, "check_back");
|
|
|
|
stash_group_add_toggle_button(group, &settings.replace_close_dialog,
|
|
|
|
"replace_close_dialog", TRUE, "check_close");
|
2008-12-16 13:01:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 11:24:23 +00:00
|
|
|
void search_init(void)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2008-11-06 17:13:18 +00:00
|
|
|
search_data.text = NULL;
|
2011-06-03 13:40:42 +00:00
|
|
|
search_data.original_text = NULL;
|
2008-12-16 13:01:47 +00:00
|
|
|
init_prefs();
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define FREE_WIDGET(wid) \
|
|
|
|
if (wid && GTK_IS_WIDGET(wid)) gtk_widget_destroy(wid);
|
|
|
|
|
2008-02-20 11:24:23 +00:00
|
|
|
void search_finalize(void)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
FREE_WIDGET(find_dlg.dialog);
|
|
|
|
FREE_WIDGET(replace_dlg.dialog);
|
|
|
|
FREE_WIDGET(fif_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
g_free(search_data.text);
|
2011-06-03 13:40:42 +00:00
|
|
|
g_free(search_data.original_text);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-22 14:56:05 +00:00
|
|
|
static GtkWidget *add_find_checkboxes(GtkDialog *dialog)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2006-11-01 15:26:41 +00:00
|
|
|
GtkWidget *checkbox1, *checkbox2, *check_regexp, *check_back, *checkbox5,
|
2006-08-12 12:04:14 +00:00
|
|
|
*checkbox7, *hbox, *fbox, *mbox;
|
2006-08-11 21:12:49 +00:00
|
|
|
|
|
|
|
check_regexp = gtk_check_button_new_with_mnemonic(_("_Use regular expressions"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(dialog, check_regexp, "check_regexp");
|
2006-08-11 21:12:49 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_regexp), FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(check_regexp, _("Use POSIX-like regular expressions. "
|
2008-11-18 20:14:42 +00:00
|
|
|
"For detailed information about using regular expressions, please read the documentation."));
|
2008-07-18 13:40:48 +00:00
|
|
|
g_signal_connect(check_regexp, "toggled",
|
2011-04-05 16:16:47 +00:00
|
|
|
G_CALLBACK(on_find_replace_checkbutton_toggled), dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
if (dialog != GTK_DIALOG(find_dlg.dialog))
|
2006-11-01 15:26:41 +00:00
|
|
|
{
|
2007-06-29 16:01:40 +00:00
|
|
|
check_back = gtk_check_button_new_with_mnemonic(_("Search _backwards"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(dialog, check_back, "check_back");
|
2006-11-01 15:26:41 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_back), FALSE);
|
|
|
|
}
|
|
|
|
else
|
2008-02-27 13:17:29 +00:00
|
|
|
{ /* align the two checkboxes at the top of the hbox */
|
2006-11-01 15:26:41 +00:00
|
|
|
GtkSizeGroup *label_size;
|
|
|
|
check_back = gtk_label_new(NULL);
|
|
|
|
label_size = gtk_size_group_new(GTK_SIZE_GROUP_VERTICAL);
|
|
|
|
gtk_size_group_add_widget(GTK_SIZE_GROUP(label_size), check_back);
|
|
|
|
gtk_size_group_add_widget(GTK_SIZE_GROUP(label_size), check_regexp);
|
|
|
|
g_object_unref(label_size);
|
|
|
|
}
|
2006-08-12 12:04:14 +00:00
|
|
|
checkbox7 = gtk_check_button_new_with_mnemonic(_("Use _escape sequences"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(dialog, checkbox7, "check_escape");
|
2006-08-11 21:12:49 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox7), FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(checkbox7,
|
2006-08-11 21:12:49 +00:00
|
|
|
_("Replace \\\\, \\t, \\n, \\r and \\uXXXX (Unicode chararacters) with the "
|
2009-01-15 19:15:41 +00:00
|
|
|
"corresponding control characters"));
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* Search features */
|
2006-08-12 12:04:14 +00:00
|
|
|
fbox = gtk_vbox_new(FALSE, 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(fbox), check_regexp);
|
|
|
|
gtk_container_add(GTK_CONTAINER(fbox), checkbox7);
|
2006-11-01 15:26:41 +00:00
|
|
|
gtk_container_add(GTK_CONTAINER(fbox), check_back);
|
2006-08-12 12:04:14 +00:00
|
|
|
|
2007-06-29 16:01:40 +00:00
|
|
|
checkbox1 = gtk_check_button_new_with_mnemonic(_("C_ase sensitive"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(dialog, checkbox1, "check_case");
|
2006-08-12 12:04:14 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox1), FALSE);
|
|
|
|
|
|
|
|
checkbox2 = gtk_check_button_new_with_mnemonic(_("Match only a _whole word"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(dialog, checkbox2, "check_word");
|
2006-08-12 12:04:14 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox2), FALSE);
|
|
|
|
|
2006-12-01 12:51:33 +00:00
|
|
|
checkbox5 = gtk_check_button_new_with_mnemonic(_("Match from s_tart of word"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(dialog, checkbox5, "check_wordstart");
|
2006-08-12 12:04:14 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox5), FALSE);
|
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* Matching options */
|
2006-08-12 12:04:14 +00:00
|
|
|
mbox = gtk_vbox_new(FALSE, 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(mbox), checkbox1);
|
|
|
|
gtk_container_add(GTK_CONTAINER(mbox), checkbox2);
|
|
|
|
gtk_container_add(GTK_CONTAINER(mbox), checkbox5);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2006-08-27 17:49:10 +00:00
|
|
|
hbox = gtk_hbox_new(TRUE, 6);
|
2006-08-12 12:04:14 +00:00
|
|
|
gtk_container_add(GTK_CONTAINER(hbox), fbox);
|
|
|
|
gtk_container_add(GTK_CONTAINER(hbox), mbox);
|
2006-10-22 14:56:05 +00:00
|
|
|
return hbox;
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-22 14:56:05 +00:00
|
|
|
static void send_find_dialog_response(GtkButton *button, gpointer user_data)
|
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_dialog_response(GTK_DIALOG(find_dlg.dialog), GPOINTER_TO_INT(user_data));
|
2006-10-22 14:56:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* store text, clear search flags so we can use Search->Find Next/Previous */
|
2007-01-24 12:35:05 +00:00
|
|
|
static void setup_find_next(const gchar *text)
|
|
|
|
{
|
|
|
|
g_free(search_data.text);
|
2011-06-03 13:40:42 +00:00
|
|
|
g_free(search_data.original_text);
|
2007-01-24 12:35:05 +00:00
|
|
|
search_data.text = g_strdup(text);
|
2011-06-03 13:40:42 +00:00
|
|
|
search_data.original_text = g_strdup(text);
|
2007-01-24 12:35:05 +00:00
|
|
|
search_data.flags = 0;
|
|
|
|
search_data.backwards = FALSE;
|
2007-07-11 17:44:43 +00:00
|
|
|
search_data.search_bar = FALSE;
|
2007-01-24 12:35:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-23 12:22:44 +00:00
|
|
|
/* Search for next match of the current "selection".
|
|
|
|
* Optionally for X11 based systems, this will try to use the system-wide
|
|
|
|
* x-selection first.
|
|
|
|
* If it doesn't find a suitable search string it will try to use
|
|
|
|
* the current word instead, or just repeat the last search.
|
2007-01-23 17:51:30 +00:00
|
|
|
* Search flags are always zero.
|
|
|
|
*/
|
2008-06-15 13:35:48 +00:00
|
|
|
void search_find_selection(GeanyDocument *doc, gboolean search_backwards)
|
2007-01-23 17:51:30 +00:00
|
|
|
{
|
|
|
|
gchar *s = NULL;
|
2007-02-19 15:41:29 +00:00
|
|
|
|
2008-06-16 18:31:59 +00:00
|
|
|
g_return_if_fail(doc != NULL);
|
2007-01-23 17:51:30 +00:00
|
|
|
|
|
|
|
#ifdef G_OS_UNIX
|
2010-09-23 12:22:44 +00:00
|
|
|
if (search_prefs.find_selection_type == GEANY_FIND_SEL_X)
|
2007-01-23 17:51:30 +00:00
|
|
|
{
|
2010-09-23 12:22:44 +00:00
|
|
|
GtkClipboard *clipboard = gtk_clipboard_get(GDK_SELECTION_PRIMARY);
|
|
|
|
|
|
|
|
s = gtk_clipboard_wait_for_text(clipboard);
|
|
|
|
if (s && (strchr(s,'\n') || strchr(s, '\r')))
|
2007-01-23 17:51:30 +00:00
|
|
|
{
|
|
|
|
g_free(s);
|
2008-06-18 13:37:03 +00:00
|
|
|
s = NULL;
|
2007-01-23 17:51:30 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
#endif
|
2007-02-19 15:41:29 +00:00
|
|
|
|
2010-09-23 12:22:44 +00:00
|
|
|
if (!s && sci_has_selection(doc->editor->sci))
|
|
|
|
s = sci_get_selection_contents(doc->editor->sci);
|
|
|
|
|
|
|
|
if (!s && search_prefs.find_selection_type != GEANY_FIND_SEL_AGAIN)
|
|
|
|
{
|
|
|
|
/* get the current word */
|
2008-09-25 18:28:37 +00:00
|
|
|
s = editor_get_default_selection(doc->editor, TRUE, NULL);
|
2010-09-23 12:22:44 +00:00
|
|
|
}
|
2008-03-23 16:29:43 +00:00
|
|
|
|
2007-01-23 17:51:30 +00:00
|
|
|
if (s)
|
|
|
|
{
|
2008-02-27 13:17:29 +00:00
|
|
|
setup_find_next(s); /* allow find next/prev */
|
2008-02-22 13:30:16 +00:00
|
|
|
|
2011-06-03 13:40:42 +00:00
|
|
|
if (document_find_text(doc, s, NULL, 0, search_backwards, FALSE, NULL) > -1)
|
2008-07-08 13:53:08 +00:00
|
|
|
editor_display_current_line(doc->editor, 0.3F);
|
2007-01-23 17:51:30 +00:00
|
|
|
g_free(s);
|
|
|
|
}
|
2010-09-23 12:22:44 +00:00
|
|
|
else if (search_prefs.find_selection_type == GEANY_FIND_SEL_AGAIN)
|
2010-09-21 16:25:10 +00:00
|
|
|
{
|
|
|
|
/* Repeat last search (in case selection was lost) */
|
|
|
|
search_find_again(search_backwards);
|
|
|
|
}
|
2010-09-23 12:22:44 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
utils_beep();
|
|
|
|
}
|
2007-01-23 17:51:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-01-27 17:50:45 +00:00
|
|
|
static void on_expander_activated(GtkExpander *exp, gpointer data)
|
|
|
|
{
|
|
|
|
gboolean *setting = data;
|
|
|
|
|
|
|
|
*setting = gtk_expander_get_expanded(exp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-01-27 18:12:00 +00:00
|
|
|
static void create_find_dialog(void)
|
|
|
|
{
|
|
|
|
GtkWidget *label, *entry, *sbox, *vbox;
|
|
|
|
GtkWidget *exp, *bbox, *button, *check_close;
|
|
|
|
|
|
|
|
find_dlg.dialog = gtk_dialog_new_with_buttons(_("Find"),
|
|
|
|
GTK_WINDOW(main_widgets.window), GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
|
|
GTK_STOCK_CLOSE, GTK_RESPONSE_CANCEL, NULL);
|
|
|
|
vbox = ui_dialog_vbox_new(GTK_DIALOG(find_dlg.dialog));
|
|
|
|
gtk_widget_set_name(find_dlg.dialog, "GeanyDialogSearch");
|
|
|
|
gtk_box_set_spacing(GTK_BOX(vbox), 9);
|
|
|
|
|
|
|
|
button = ui_button_new_with_image(GTK_STOCK_GO_BACK, _("_Previous"));
|
|
|
|
gtk_dialog_add_action_widget(GTK_DIALOG(find_dlg.dialog), button,
|
|
|
|
GEANY_RESPONSE_FIND_PREVIOUS);
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(find_dlg.dialog, button, "btn_previous");
|
2009-01-27 18:12:00 +00:00
|
|
|
|
|
|
|
button = ui_button_new_with_image(GTK_STOCK_GO_FORWARD, _("_Next"));
|
|
|
|
gtk_dialog_add_action_widget(GTK_DIALOG(find_dlg.dialog), button,
|
|
|
|
GEANY_RESPONSE_FIND);
|
|
|
|
|
|
|
|
label = gtk_label_new_with_mnemonic(_("_Search for:"));
|
|
|
|
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
|
|
|
|
|
|
|
|
entry = gtk_combo_box_entry_new_text();
|
2009-09-21 16:46:16 +00:00
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry))));
|
2009-01-27 18:12:00 +00:00
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry))), 50);
|
2011-10-11 21:30:28 -07:00
|
|
|
find_dlg.entry = gtk_bin_get_child(GTK_BIN(entry));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(find_dlg.dialog, entry, "entry");
|
2009-01-27 18:12:00 +00:00
|
|
|
|
|
|
|
g_signal_connect(gtk_bin_get_child(GTK_BIN(entry)), "activate",
|
|
|
|
G_CALLBACK(on_find_entry_activate), NULL);
|
2011-03-24 16:51:02 +00:00
|
|
|
ui_entry_add_activate_backward_signal(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry))));
|
|
|
|
g_signal_connect(gtk_bin_get_child(GTK_BIN(entry)), "activate-backward",
|
|
|
|
G_CALLBACK(on_find_entry_activate_backward), NULL);
|
2009-01-27 18:12:00 +00:00
|
|
|
g_signal_connect(find_dlg.dialog, "response",
|
|
|
|
G_CALLBACK(on_find_dialog_response), entry);
|
|
|
|
g_signal_connect(find_dlg.dialog, "delete-event",
|
|
|
|
G_CALLBACK(gtk_widget_hide_on_delete), NULL);
|
|
|
|
|
|
|
|
sbox = gtk_hbox_new(FALSE, 6);
|
|
|
|
gtk_box_pack_start(GTK_BOX(sbox), label, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(sbox), entry, TRUE, TRUE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), sbox, TRUE, FALSE, 0);
|
|
|
|
|
|
|
|
gtk_container_add(GTK_CONTAINER(vbox),
|
|
|
|
add_find_checkboxes(GTK_DIALOG(find_dlg.dialog)));
|
|
|
|
|
|
|
|
/* Now add the multiple match options */
|
|
|
|
exp = gtk_expander_new_with_mnemonic(_("_Find All"));
|
|
|
|
gtk_expander_set_expanded(GTK_EXPANDER(exp), find_dlg.all_expanded);
|
|
|
|
g_signal_connect_after(exp, "activate",
|
|
|
|
G_CALLBACK(on_expander_activated), &find_dlg.all_expanded);
|
|
|
|
|
|
|
|
bbox = gtk_hbutton_box_new();
|
|
|
|
|
|
|
|
button = gtk_button_new_with_mnemonic(_("_Mark"));
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(button,
|
2009-01-27 18:12:00 +00:00
|
|
|
_("Mark all matches in the current document"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), button);
|
|
|
|
g_signal_connect(button, "clicked", G_CALLBACK(send_find_dialog_response),
|
|
|
|
GINT_TO_POINTER(GEANY_RESPONSE_MARK));
|
|
|
|
|
|
|
|
button = gtk_button_new_with_mnemonic(_("In Sessi_on"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), button);
|
|
|
|
g_signal_connect(button, "clicked", G_CALLBACK(send_find_dialog_response),
|
|
|
|
GINT_TO_POINTER(GEANY_RESPONSE_FIND_IN_SESSION));
|
|
|
|
|
|
|
|
button = gtk_button_new_with_mnemonic(_("_In Document"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), button);
|
|
|
|
g_signal_connect(button, "clicked", G_CALLBACK(send_find_dialog_response),
|
|
|
|
GINT_TO_POINTER(GEANY_RESPONSE_FIND_IN_FILE));
|
|
|
|
|
|
|
|
/* close window checkbox */
|
|
|
|
check_close = gtk_check_button_new_with_mnemonic(_("Close _dialog"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(find_dlg.dialog, check_close, "check_close");
|
2009-01-27 18:12:00 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_close), FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(check_close,
|
2009-01-27 18:12:00 +00:00
|
|
|
_("Disable this option to keep the dialog open"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), check_close);
|
|
|
|
gtk_button_box_set_child_secondary(GTK_BUTTON_BOX(bbox), check_close, TRUE);
|
|
|
|
|
|
|
|
ui_hbutton_box_copy_layout(
|
2012-09-11 16:48:57 +02:00
|
|
|
GTK_BUTTON_BOX(gtk_dialog_get_action_area(GTK_DIALOG(find_dlg.dialog))),
|
2009-01-27 18:12:00 +00:00
|
|
|
GTK_BUTTON_BOX(bbox));
|
|
|
|
gtk_container_add(GTK_CONTAINER(exp), bbox);
|
|
|
|
gtk_container_add(GTK_CONTAINER(vbox), exp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-29 18:45:49 +00:00
|
|
|
static void set_dialog_position(GtkWidget *dialog, gint *position)
|
|
|
|
{
|
|
|
|
if (position[0] >= 0)
|
|
|
|
gtk_window_move(GTK_WINDOW(dialog), position[0], position[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 11:24:23 +00:00
|
|
|
void search_show_find_dialog(void)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2008-06-15 13:35:48 +00:00
|
|
|
GeanyDocument *doc = document_get_current();
|
2006-08-11 21:12:49 +00:00
|
|
|
gchar *sel = NULL;
|
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
g_return_if_fail(doc != NULL);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2008-09-25 18:28:37 +00:00
|
|
|
sel = editor_get_default_selection(doc->editor, search_prefs.use_current_word, NULL);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
if (find_dlg.dialog == NULL)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2009-01-27 18:12:00 +00:00
|
|
|
create_find_dialog();
|
2011-03-06 14:18:05 +00:00
|
|
|
stash_group_display(find_prefs, find_dlg.dialog);
|
2009-01-27 18:12:00 +00:00
|
|
|
if (sel)
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(find_dlg.entry), sel);
|
2006-11-01 15:26:41 +00:00
|
|
|
|
2009-12-29 18:45:49 +00:00
|
|
|
set_dialog_position(find_dlg.dialog, find_dlg.position);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_show_all(find_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-02-27 13:17:29 +00:00
|
|
|
/* only set selection if the dialog is not already visible */
|
2009-01-27 18:03:58 +00:00
|
|
|
if (! GTK_WIDGET_VISIBLE(find_dlg.dialog) && sel)
|
2009-02-27 14:05:50 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(find_dlg.entry), sel);
|
|
|
|
gtk_widget_grab_focus(find_dlg.entry);
|
2009-12-29 18:45:49 +00:00
|
|
|
set_dialog_position(find_dlg.dialog, find_dlg.position);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_show(find_dlg.dialog);
|
2009-02-27 14:05:50 +00:00
|
|
|
if (sel != NULL) /* when we have a selection, reset the entry widget's background colour */
|
|
|
|
ui_set_search_entry_background(find_dlg.entry, TRUE);
|
2008-02-27 13:17:29 +00:00
|
|
|
/* bring the dialog back in the foreground in case it is already open but the focus is away */
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_window_present(GTK_WINDOW(find_dlg.dialog));
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
2009-12-29 18:45:49 +00:00
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
g_free(sel);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-22 14:56:05 +00:00
|
|
|
static void send_replace_dialog_response(GtkButton *button, gpointer user_data)
|
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_dialog_response(GTK_DIALOG(replace_dlg.dialog), GPOINTER_TO_INT(user_data));
|
2006-10-22 14:56:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-21 14:27:20 +00:00
|
|
|
static gboolean
|
|
|
|
on_widget_key_pressed_set_focus(GtkWidget *widget, GdkEventKey *event, gpointer user_data)
|
|
|
|
{
|
|
|
|
if (event->keyval == GDK_Tab)
|
|
|
|
{
|
|
|
|
gtk_widget_grab_focus(GTK_WIDGET(user_data));
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-01-22 17:19:15 +00:00
|
|
|
static void create_replace_dialog(void)
|
|
|
|
{
|
|
|
|
GtkWidget *label_find, *label_replace, *entry_find, *entry_replace,
|
|
|
|
*check_close, *button, *rbox, *fbox, *vbox, *exp, *bbox;
|
|
|
|
GtkSizeGroup *label_size;
|
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
replace_dlg.dialog = gtk_dialog_new_with_buttons(_("Replace"),
|
2009-01-22 17:19:15 +00:00
|
|
|
GTK_WINDOW(main_widgets.window), GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
|
|
GTK_STOCK_CLOSE, GTK_RESPONSE_CANCEL, NULL);
|
2009-01-27 18:03:58 +00:00
|
|
|
vbox = ui_dialog_vbox_new(GTK_DIALOG(replace_dlg.dialog));
|
2009-01-22 17:19:15 +00:00
|
|
|
gtk_box_set_spacing(GTK_BOX(vbox), 9);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_set_name(replace_dlg.dialog, "GeanyDialogSearch");
|
2009-01-22 17:19:15 +00:00
|
|
|
|
|
|
|
button = gtk_button_new_from_stock(GTK_STOCK_FIND);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_dialog_add_action_widget(GTK_DIALOG(replace_dlg.dialog), button,
|
2009-01-22 17:19:15 +00:00
|
|
|
GEANY_RESPONSE_FIND);
|
|
|
|
button = gtk_button_new_with_mnemonic(_("_Replace"));
|
2009-02-27 14:05:50 +00:00
|
|
|
gtk_button_set_image(GTK_BUTTON(button),
|
|
|
|
gtk_image_new_from_stock(GTK_STOCK_FIND_AND_REPLACE, GTK_ICON_SIZE_BUTTON));
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_dialog_add_action_widget(GTK_DIALOG(replace_dlg.dialog), button,
|
2009-01-22 17:19:15 +00:00
|
|
|
GEANY_RESPONSE_REPLACE);
|
|
|
|
button = gtk_button_new_with_mnemonic(_("Replace & Fi_nd"));
|
2009-02-27 14:05:50 +00:00
|
|
|
gtk_button_set_image(GTK_BUTTON(button),
|
|
|
|
gtk_image_new_from_stock(GTK_STOCK_FIND_AND_REPLACE, GTK_ICON_SIZE_BUTTON));
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_dialog_add_action_widget(GTK_DIALOG(replace_dlg.dialog), button,
|
2009-01-22 17:19:15 +00:00
|
|
|
GEANY_RESPONSE_REPLACE_AND_FIND);
|
|
|
|
|
|
|
|
label_find = gtk_label_new_with_mnemonic(_("_Search for:"));
|
|
|
|
gtk_misc_set_alignment(GTK_MISC(label_find), 0, 0.5);
|
|
|
|
|
|
|
|
label_replace = gtk_label_new_with_mnemonic(_("Replace wit_h:"));
|
|
|
|
gtk_misc_set_alignment(GTK_MISC(label_replace), 0, 0.5);
|
|
|
|
|
|
|
|
entry_find = gtk_combo_box_entry_new_text();
|
2009-09-21 16:46:16 +00:00
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry_find))));
|
2009-01-22 17:19:15 +00:00
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label_find), entry_find);
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry_find))), 50);
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(replace_dlg.dialog, entry_find, "entry_find");
|
2011-10-11 21:30:28 -07:00
|
|
|
replace_dlg.find_entry = gtk_bin_get_child(GTK_BIN(entry_find));
|
2009-01-22 17:19:15 +00:00
|
|
|
|
|
|
|
entry_replace = gtk_combo_box_entry_new_text();
|
2009-09-21 16:46:16 +00:00
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry_replace))));
|
2009-01-22 17:19:15 +00:00
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label_replace), entry_replace);
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(entry_replace))), 50);
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(replace_dlg.dialog, entry_replace, "entry_replace");
|
2011-10-11 21:30:28 -07:00
|
|
|
replace_dlg.replace_entry = gtk_bin_get_child(GTK_BIN(entry_replace));
|
2009-01-22 17:19:15 +00:00
|
|
|
|
2010-06-28 13:47:14 +00:00
|
|
|
/* tab from find to the replace entry */
|
2009-01-22 17:19:15 +00:00
|
|
|
g_signal_connect(gtk_bin_get_child(GTK_BIN(entry_find)),
|
|
|
|
"key-press-event", G_CALLBACK(on_widget_key_pressed_set_focus),
|
|
|
|
gtk_bin_get_child(GTK_BIN(entry_replace)));
|
|
|
|
g_signal_connect(gtk_bin_get_child(GTK_BIN(entry_replace)), "activate",
|
|
|
|
G_CALLBACK(on_replace_entry_activate), NULL);
|
2009-01-27 18:03:58 +00:00
|
|
|
g_signal_connect(replace_dlg.dialog, "response",
|
2009-01-22 17:19:15 +00:00
|
|
|
G_CALLBACK(on_replace_dialog_response), entry_replace);
|
2009-01-27 18:03:58 +00:00
|
|
|
g_signal_connect(replace_dlg.dialog, "delete-event",
|
2009-01-22 17:19:15 +00:00
|
|
|
G_CALLBACK(gtk_widget_hide_on_delete), NULL);
|
|
|
|
|
|
|
|
fbox = gtk_hbox_new(FALSE, 6);
|
|
|
|
gtk_box_pack_start(GTK_BOX(fbox), label_find, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(fbox), entry_find, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
rbox = gtk_hbox_new(FALSE, 6);
|
|
|
|
gtk_box_pack_start(GTK_BOX(rbox), label_replace, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(rbox), entry_replace, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
label_size = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
|
|
|
|
gtk_size_group_add_widget(label_size, label_find);
|
|
|
|
gtk_size_group_add_widget(label_size, label_replace);
|
|
|
|
g_object_unref(G_OBJECT(label_size)); /* auto destroy the size group */
|
|
|
|
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), fbox, TRUE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), rbox, TRUE, FALSE, 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(vbox),
|
2009-01-27 18:03:58 +00:00
|
|
|
add_find_checkboxes(GTK_DIALOG(replace_dlg.dialog)));
|
2009-01-22 17:19:15 +00:00
|
|
|
|
|
|
|
/* Now add the multiple replace options */
|
|
|
|
exp = gtk_expander_new_with_mnemonic(_("Re_place All"));
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_expander_set_expanded(GTK_EXPANDER(exp), replace_dlg.all_expanded);
|
2009-01-27 17:50:45 +00:00
|
|
|
g_signal_connect_after(exp, "activate",
|
2009-01-27 18:03:58 +00:00
|
|
|
G_CALLBACK(on_expander_activated), &replace_dlg.all_expanded);
|
2009-01-27 17:50:45 +00:00
|
|
|
|
2009-01-22 17:19:15 +00:00
|
|
|
bbox = gtk_hbutton_box_new();
|
|
|
|
|
|
|
|
button = gtk_button_new_with_mnemonic(_("In Sessi_on"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), button);
|
|
|
|
g_signal_connect(button, "clicked", G_CALLBACK(send_replace_dialog_response),
|
|
|
|
GINT_TO_POINTER(GEANY_RESPONSE_REPLACE_IN_SESSION));
|
|
|
|
|
|
|
|
button = gtk_button_new_with_mnemonic(_("_In Document"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), button);
|
|
|
|
g_signal_connect(button, "clicked", G_CALLBACK(send_replace_dialog_response),
|
|
|
|
GINT_TO_POINTER(GEANY_RESPONSE_REPLACE_IN_FILE));
|
|
|
|
|
2009-01-22 17:22:05 +00:00
|
|
|
button = gtk_button_new_with_mnemonic(_("In Se_lection"));
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(button,
|
2009-01-22 17:22:05 +00:00
|
|
|
_("Replace all matches found in the currently selected text"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), button);
|
|
|
|
g_signal_connect(button, "clicked", G_CALLBACK(send_replace_dialog_response),
|
|
|
|
GINT_TO_POINTER(GEANY_RESPONSE_REPLACE_IN_SEL));
|
|
|
|
|
2009-01-22 17:19:15 +00:00
|
|
|
/* close window checkbox */
|
|
|
|
check_close = gtk_check_button_new_with_mnemonic(_("Close _dialog"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(replace_dlg.dialog, check_close, "check_close");
|
2009-01-22 17:19:15 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_close), FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(check_close,
|
2009-01-22 17:19:15 +00:00
|
|
|
_("Disable this option to keep the dialog open"));
|
|
|
|
gtk_container_add(GTK_CONTAINER(bbox), check_close);
|
|
|
|
gtk_button_box_set_child_secondary(GTK_BUTTON_BOX(bbox), check_close, TRUE);
|
|
|
|
|
|
|
|
ui_hbutton_box_copy_layout(
|
2012-09-11 16:48:57 +02:00
|
|
|
GTK_BUTTON_BOX(gtk_dialog_get_action_area(GTK_DIALOG(replace_dlg.dialog))),
|
2009-01-22 17:19:15 +00:00
|
|
|
GTK_BUTTON_BOX(bbox));
|
|
|
|
gtk_container_add(GTK_CONTAINER(exp), bbox);
|
|
|
|
gtk_container_add(GTK_CONTAINER(vbox), exp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 11:24:23 +00:00
|
|
|
void search_show_replace_dialog(void)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2008-06-15 13:35:48 +00:00
|
|
|
GeanyDocument *doc = document_get_current();
|
2006-08-11 21:12:49 +00:00
|
|
|
gchar *sel = NULL;
|
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
if (doc == NULL)
|
|
|
|
return;
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2008-09-25 18:28:37 +00:00
|
|
|
sel = editor_get_default_selection(doc->editor, search_prefs.use_current_word, NULL);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
if (replace_dlg.dialog == NULL)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2009-01-22 17:19:15 +00:00
|
|
|
create_replace_dialog();
|
2011-03-06 14:18:05 +00:00
|
|
|
stash_group_display(replace_prefs, replace_dlg.dialog);
|
2009-01-22 17:19:15 +00:00
|
|
|
if (sel)
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(replace_dlg.find_entry), sel);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-12-29 18:45:49 +00:00
|
|
|
set_dialog_position(replace_dlg.dialog, replace_dlg.position);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_show_all(replace_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-02-27 13:17:29 +00:00
|
|
|
/* only set selection if the dialog is not already visible */
|
2009-01-27 18:03:58 +00:00
|
|
|
if (! GTK_WIDGET_VISIBLE(replace_dlg.dialog) && sel)
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(replace_dlg.find_entry), sel);
|
2009-02-27 14:05:50 +00:00
|
|
|
if (sel != NULL) /* when we have a selection, reset the entry widget's background colour */
|
|
|
|
ui_set_search_entry_background(replace_dlg.find_entry, TRUE);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_grab_focus(replace_dlg.find_entry);
|
2009-12-29 18:45:49 +00:00
|
|
|
set_dialog_position(replace_dlg.dialog, replace_dlg.position);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_show(replace_dlg.dialog);
|
2008-02-27 13:17:29 +00:00
|
|
|
/* bring the dialog back in the foreground in case it is already open but the focus is away */
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_window_present(GTK_WINDOW(replace_dlg.dialog));
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
2011-03-06 14:18:05 +00:00
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
g_free(sel);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-22 16:51:01 +00:00
|
|
|
static void on_widget_toggled_set_sensitive(GtkToggleButton *togglebutton, gpointer user_data)
|
2007-04-03 15:57:12 +00:00
|
|
|
{
|
2008-02-27 13:17:29 +00:00
|
|
|
/* disable extra option entry when checkbutton not checked */
|
2007-04-24 10:52:08 +00:00
|
|
|
gtk_widget_set_sensitive(GTK_WIDGET(user_data),
|
|
|
|
gtk_toggle_button_get_active(togglebutton));
|
2007-04-03 15:57:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-30 21:50:37 +00:00
|
|
|
static void update_file_patterns(GtkWidget *mode_combo, GtkWidget *fcombo)
|
|
|
|
{
|
|
|
|
gint selection;
|
|
|
|
GtkWidget *entry;
|
|
|
|
|
|
|
|
entry = gtk_bin_get_child(GTK_BIN(fcombo));
|
|
|
|
|
|
|
|
selection = gtk_combo_box_get_active(GTK_COMBO_BOX(mode_combo));
|
|
|
|
|
|
|
|
if (selection == FILES_MODE_ALL)
|
|
|
|
{
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), "");
|
|
|
|
gtk_widget_set_sensitive(fcombo, FALSE);
|
|
|
|
}
|
|
|
|
else if (selection == FILES_MODE_CUSTOM)
|
|
|
|
{
|
|
|
|
gtk_widget_set_sensitive(fcombo, TRUE);
|
|
|
|
}
|
|
|
|
else if (selection == FILES_MODE_PROJECT)
|
|
|
|
{
|
|
|
|
if (app->project && NZV(app->project->file_patterns))
|
|
|
|
{
|
|
|
|
gchar *patterns;
|
|
|
|
|
|
|
|
patterns = g_strjoinv(" ", app->project->file_patterns);
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), patterns);
|
|
|
|
g_free(patterns);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), "");
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_widget_set_sensitive(fcombo, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* creates the combo to choose which files include in the search */
|
|
|
|
static GtkWidget *create_fif_file_mode_combo(void)
|
|
|
|
{
|
|
|
|
GtkWidget *combo;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkListStore *store;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
/* text/sensitive */
|
|
|
|
store = gtk_list_store_new(2, G_TYPE_STRING, G_TYPE_BOOLEAN);
|
|
|
|
gtk_list_store_append(store, &iter);
|
|
|
|
gtk_list_store_set(store, &iter, 0, _("all"), 1, TRUE, -1);
|
|
|
|
gtk_list_store_append(store, &iter);
|
|
|
|
gtk_list_store_set(store, &iter, 0, _("project"), 1, app->project != NULL, -1);
|
|
|
|
gtk_list_store_append(store, &iter);
|
|
|
|
gtk_list_store_set(store, &iter, 0, _("custom"), 1, TRUE, -1);
|
|
|
|
|
|
|
|
combo = gtk_combo_box_new_with_model(GTK_TREE_MODEL(store));
|
|
|
|
g_object_unref(store);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(combo, _("All: search all files in the directory\n"
|
2011-04-30 21:50:37 +00:00
|
|
|
"Project: use file patterns defined in the project settings\n"
|
|
|
|
"Custom: specify file patterns manually"));
|
|
|
|
|
|
|
|
renderer = gtk_cell_renderer_text_new();
|
|
|
|
gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(combo), renderer, TRUE);
|
|
|
|
gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(combo), renderer, "text", 0, "sensitive", 1, NULL);
|
|
|
|
|
|
|
|
return combo;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* updates the sensitivity of the project combo item */
|
|
|
|
static void update_fif_file_mode_combo(void)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model = gtk_combo_box_get_model(GTK_COMBO_BOX(fif_dlg.files_mode_combo));
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
/* "1" refers to the second list entry, project */
|
|
|
|
if (gtk_tree_model_get_iter_from_string(model, &iter, "1"))
|
|
|
|
gtk_list_store_set(GTK_LIST_STORE(model), &iter, 1, app->project != NULL, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-01-11 18:29:39 +00:00
|
|
|
static void create_fif_dialog(void)
|
2008-11-06 17:13:18 +00:00
|
|
|
{
|
2010-06-22 16:51:01 +00:00
|
|
|
GtkWidget *dir_combo, *combo, *fcombo, *e_combo, *entry;
|
2011-04-30 21:50:37 +00:00
|
|
|
GtkWidget *label, *label1, *label2, *label3, *checkbox1, *checkbox2, *check_wholeword,
|
|
|
|
*check_recursive, *check_extra, *entry_extra, *check_regexp, *combo_files_mode;
|
2010-06-21 15:16:31 +00:00
|
|
|
GtkWidget *dbox, *sbox, *lbox, *rbox, *hbox, *vbox, *ebox;
|
2008-11-06 17:13:18 +00:00
|
|
|
GtkSizeGroup *size_group;
|
2008-11-13 16:29:56 +00:00
|
|
|
gchar *encoding_string;
|
|
|
|
guint i;
|
2008-11-06 17:13:18 +00:00
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
fif_dlg.dialog = gtk_dialog_new_with_buttons(
|
2008-11-06 17:13:18 +00:00
|
|
|
_("Find in Files"), GTK_WINDOW(main_widgets.window), GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, NULL);
|
2009-01-27 18:03:58 +00:00
|
|
|
vbox = ui_dialog_vbox_new(GTK_DIALOG(fif_dlg.dialog));
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_box_set_spacing(GTK_BOX(vbox), 9);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_set_name(fif_dlg.dialog, "GeanyDialogSearch");
|
2008-11-06 17:13:18 +00:00
|
|
|
|
2011-10-30 22:01:49 +01:00
|
|
|
gtk_dialog_add_button(GTK_DIALOG(fif_dlg.dialog), GTK_STOCK_FIND, GTK_RESPONSE_ACCEPT);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_dialog_set_default_response(GTK_DIALOG(fif_dlg.dialog),
|
2008-11-06 17:13:18 +00:00
|
|
|
GTK_RESPONSE_ACCEPT);
|
|
|
|
|
|
|
|
label = gtk_label_new_with_mnemonic(_("_Search for:"));
|
2010-06-22 16:51:01 +00:00
|
|
|
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
|
2008-11-06 17:13:18 +00:00
|
|
|
|
|
|
|
combo = gtk_combo_box_entry_new_text();
|
|
|
|
entry = gtk_bin_get_child(GTK_BIN(combo));
|
2009-09-21 16:46:16 +00:00
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(entry));
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(entry), 50);
|
|
|
|
gtk_entry_set_activates_default(GTK_ENTRY(entry), TRUE);
|
2009-01-27 18:03:58 +00:00
|
|
|
fif_dlg.search_combo = combo;
|
2008-11-06 17:13:18 +00:00
|
|
|
|
|
|
|
sbox = gtk_hbox_new(FALSE, 6);
|
|
|
|
gtk_box_pack_start(GTK_BOX(sbox), label, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(sbox), combo, TRUE, TRUE, 0);
|
|
|
|
|
2010-06-22 16:51:01 +00:00
|
|
|
/* make labels same width */
|
|
|
|
size_group = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
|
|
|
|
gtk_size_group_add_widget(size_group, label);
|
|
|
|
|
2011-04-30 21:50:37 +00:00
|
|
|
label3 = gtk_label_new_with_mnemonic(_("Fi_les:"));
|
|
|
|
gtk_misc_set_alignment(GTK_MISC(label3), 0, 0.5);
|
2010-06-22 16:51:01 +00:00
|
|
|
|
2011-04-30 21:50:37 +00:00
|
|
|
combo_files_mode = create_fif_file_mode_combo();
|
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label3), combo_files_mode);
|
|
|
|
ui_hookup_widget(fif_dlg.dialog, combo_files_mode, "combo_files_mode");
|
|
|
|
fif_dlg.files_mode_combo = combo_files_mode;
|
2010-06-28 13:47:14 +00:00
|
|
|
|
2010-06-22 16:51:01 +00:00
|
|
|
fcombo = gtk_combo_box_entry_new_text();
|
|
|
|
entry = gtk_bin_get_child(GTK_BIN(fcombo));
|
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(entry));
|
|
|
|
gtk_entry_set_activates_default(GTK_ENTRY(entry), TRUE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(entry, _("File patterns, e.g. *.c *.h"));
|
2010-06-22 16:51:01 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, entry, "entry_files");
|
2010-06-23 16:06:10 +00:00
|
|
|
fif_dlg.files_combo = fcombo;
|
2010-06-22 16:51:01 +00:00
|
|
|
|
2011-04-30 21:50:37 +00:00
|
|
|
/* update the entry when selection is changed */
|
|
|
|
g_signal_connect(combo_files_mode, "changed", G_CALLBACK(update_file_patterns), fcombo);
|
2010-06-22 16:51:01 +00:00
|
|
|
|
|
|
|
hbox = gtk_hbox_new(FALSE, 6);
|
2011-04-30 21:50:37 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), label3, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), combo_files_mode, FALSE, FALSE, 0);
|
2010-06-22 16:51:01 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), fcombo, TRUE, TRUE, 0);
|
|
|
|
|
2010-06-21 14:27:20 +00:00
|
|
|
label1 = gtk_label_new_with_mnemonic(_("_Directory:"));
|
2010-06-22 16:51:01 +00:00
|
|
|
gtk_misc_set_alignment(GTK_MISC(label1), 0, 0.5);
|
2010-06-21 14:27:20 +00:00
|
|
|
|
|
|
|
dir_combo = gtk_combo_box_entry_new_text();
|
|
|
|
entry = gtk_bin_get_child(GTK_BIN(dir_combo));
|
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(entry));
|
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label1), entry);
|
|
|
|
gtk_entry_set_width_chars(GTK_ENTRY(entry), 50);
|
|
|
|
fif_dlg.dir_combo = dir_combo;
|
|
|
|
|
2010-06-28 13:47:14 +00:00
|
|
|
/* tab from files to the dir entry */
|
|
|
|
g_signal_connect(gtk_bin_get_child(GTK_BIN(fcombo)), "key-press-event",
|
|
|
|
G_CALLBACK(on_widget_key_pressed_set_focus), entry);
|
|
|
|
|
2010-06-21 14:27:20 +00:00
|
|
|
dbox = ui_path_box_new(NULL, GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
|
|
|
|
GTK_ENTRY(entry));
|
|
|
|
gtk_box_pack_start(GTK_BOX(dbox), label1, FALSE, FALSE, 0);
|
|
|
|
|
2008-11-16 17:54:46 +00:00
|
|
|
label2 = gtk_label_new_with_mnemonic(_("E_ncoding:"));
|
2010-06-22 16:51:01 +00:00
|
|
|
gtk_misc_set_alignment(GTK_MISC(label2), 0, 0.5);
|
2008-11-13 16:29:56 +00:00
|
|
|
|
|
|
|
e_combo = gtk_combo_box_new_text();
|
|
|
|
for (i = 0; i < GEANY_ENCODINGS_MAX; i++)
|
|
|
|
{
|
|
|
|
encoding_string = encodings_to_string(&encodings[i]);
|
|
|
|
gtk_combo_box_append_text(GTK_COMBO_BOX(e_combo), encoding_string);
|
|
|
|
g_free(encoding_string);
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_wrap_width(GTK_COMBO_BOX(e_combo), 3);
|
2008-11-16 17:54:46 +00:00
|
|
|
gtk_label_set_mnemonic_widget(GTK_LABEL(label2), e_combo);
|
2009-01-27 18:03:58 +00:00
|
|
|
fif_dlg.encoding_combo = e_combo;
|
2008-11-13 16:29:56 +00:00
|
|
|
|
|
|
|
ebox = gtk_hbox_new(FALSE, 6);
|
2008-11-16 17:54:46 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(ebox), label2, FALSE, FALSE, 0);
|
2008-11-13 16:29:56 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(ebox), e_combo, TRUE, TRUE, 0);
|
|
|
|
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_size_group_add_widget(size_group, label1);
|
2008-11-16 17:54:46 +00:00
|
|
|
gtk_size_group_add_widget(size_group, label2);
|
2011-04-30 21:50:37 +00:00
|
|
|
gtk_size_group_add_widget(size_group, label3);
|
2008-11-06 17:13:18 +00:00
|
|
|
g_object_unref(G_OBJECT(size_group)); /* auto destroy the size group */
|
|
|
|
|
2010-06-22 16:51:01 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), sbox, TRUE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), hbox, TRUE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), dbox, TRUE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(vbox), ebox, TRUE, FALSE, 0);
|
|
|
|
|
2010-06-21 14:57:03 +00:00
|
|
|
check_regexp = gtk_check_button_new_with_mnemonic(_("_Use regular expressions"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, check_regexp, "check_regexp");
|
2010-06-21 14:57:03 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_regexp), FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(check_regexp, _("See grep's manual page for more information"));
|
2008-11-06 17:13:18 +00:00
|
|
|
|
|
|
|
check_recursive = gtk_check_button_new_with_mnemonic(_("_Recurse in subfolders"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, check_recursive, "check_recursive");
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_recursive), FALSE);
|
|
|
|
|
|
|
|
checkbox1 = gtk_check_button_new_with_mnemonic(_("C_ase sensitive"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, checkbox1, "check_case");
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox1), FALSE);
|
|
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbox1), TRUE);
|
|
|
|
|
|
|
|
check_wholeword = gtk_check_button_new_with_mnemonic(_("Match only a _whole word"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, check_wholeword, "check_wholeword");
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_wholeword), FALSE);
|
|
|
|
|
|
|
|
checkbox2 = gtk_check_button_new_with_mnemonic(_("_Invert search results"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, checkbox2, "check_invert");
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox2), FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(checkbox2,
|
2009-03-05 13:03:51 +00:00
|
|
|
_("Invert the sense of matching, to select non-matching lines"));
|
2008-11-06 17:13:18 +00:00
|
|
|
|
2010-06-21 15:16:31 +00:00
|
|
|
lbox = gtk_vbox_new(FALSE, 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(lbox), check_regexp);
|
|
|
|
gtk_container_add(GTK_CONTAINER(lbox), checkbox2);
|
|
|
|
gtk_container_add(GTK_CONTAINER(lbox), check_recursive);
|
|
|
|
|
|
|
|
rbox = gtk_vbox_new(FALSE, 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(rbox), checkbox1);
|
|
|
|
gtk_container_add(GTK_CONTAINER(rbox), check_wholeword);
|
|
|
|
gtk_container_add(GTK_CONTAINER(rbox), gtk_label_new(NULL));
|
2008-11-06 17:13:18 +00:00
|
|
|
|
|
|
|
hbox = gtk_hbox_new(FALSE, 6);
|
2010-06-21 15:16:31 +00:00
|
|
|
gtk_container_add(GTK_CONTAINER(hbox), lbox);
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_container_add(GTK_CONTAINER(hbox), rbox);
|
|
|
|
gtk_container_add(GTK_CONTAINER(vbox), hbox);
|
|
|
|
|
|
|
|
check_extra = gtk_check_button_new_with_mnemonic(_("E_xtra options:"));
|
2010-06-23 16:57:28 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, check_extra, "check_extra");
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(check_extra), FALSE);
|
|
|
|
|
|
|
|
entry_extra = gtk_entry_new();
|
2009-09-21 16:46:16 +00:00
|
|
|
ui_entry_add_clear_icon(GTK_ENTRY(entry_extra));
|
2008-11-06 17:13:18 +00:00
|
|
|
gtk_widget_set_sensitive(entry_extra, FALSE);
|
2011-06-13 18:41:50 +00:00
|
|
|
gtk_widget_set_tooltip_text(entry_extra, _("Other options to pass to Grep"));
|
2009-01-27 18:03:58 +00:00
|
|
|
ui_hookup_widget(fif_dlg.dialog, entry_extra, "entry_extra");
|
2008-11-06 17:13:18 +00:00
|
|
|
|
|
|
|
/* enable entry_extra when check_extra is checked */
|
|
|
|
g_signal_connect(check_extra, "toggled",
|
2010-06-22 16:51:01 +00:00
|
|
|
G_CALLBACK(on_widget_toggled_set_sensitive), entry_extra);
|
2008-11-06 17:13:18 +00:00
|
|
|
|
|
|
|
hbox = gtk_hbox_new(FALSE, 6);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), check_extra, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), entry_extra, TRUE, TRUE, 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(vbox), hbox);
|
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
g_signal_connect(fif_dlg.dialog, "response",
|
2008-11-06 17:13:18 +00:00
|
|
|
G_CALLBACK(on_find_in_files_dialog_response), NULL);
|
2009-01-27 18:03:58 +00:00
|
|
|
g_signal_connect(fif_dlg.dialog, "delete-event",
|
2008-11-06 17:13:18 +00:00
|
|
|
G_CALLBACK(gtk_widget_hide_on_delete), NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-14 00:24:42 +02:00
|
|
|
/* dir is the directory to search in (UTF-8 encoding), may be NULL to determine it the usual way
|
2007-12-02 10:52:19 +00:00
|
|
|
* by using the current file's path */
|
|
|
|
void search_show_find_in_files_dialog(const gchar *dir)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2008-11-06 17:13:18 +00:00
|
|
|
GtkWidget *entry; /* for child GtkEntry of a GtkComboBoxEntry */
|
2008-06-15 13:35:48 +00:00
|
|
|
GeanyDocument *doc = document_get_current();
|
2006-08-11 21:12:49 +00:00
|
|
|
gchar *sel = NULL;
|
2008-06-25 11:56:58 +00:00
|
|
|
gchar *cur_dir = NULL;
|
2008-11-13 16:29:56 +00:00
|
|
|
GeanyEncodingIndex enc_idx = GEANY_ENCODING_UTF_8;
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
if (fif_dlg.dialog == NULL)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2008-11-06 17:13:18 +00:00
|
|
|
create_fif_dialog();
|
2009-01-27 18:12:00 +00:00
|
|
|
gtk_widget_show_all(fif_dlg.dialog);
|
2009-10-14 14:07:28 +00:00
|
|
|
if (doc)
|
|
|
|
sel = editor_get_default_selection(doc->editor, search_prefs.use_current_word, NULL);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
2009-01-27 18:03:58 +00:00
|
|
|
stash_group_display(fif_prefs, fif_dlg.dialog);
|
2008-12-16 13:01:47 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* only set selection if the dialog is not already visible, or has just been created */
|
2009-10-14 14:07:28 +00:00
|
|
|
if (doc && ! sel && ! GTK_WIDGET_VISIBLE(fif_dlg.dialog))
|
|
|
|
sel = editor_get_default_selection(doc->editor, search_prefs.use_current_word, NULL);
|
2008-11-06 17:26:29 +00:00
|
|
|
|
2011-10-11 21:30:28 -07:00
|
|
|
entry = gtk_bin_get_child(GTK_BIN(fif_dlg.search_combo));
|
2006-12-17 16:47:08 +00:00
|
|
|
if (sel)
|
2006-12-15 18:33:29 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), sel);
|
2006-08-11 21:12:49 +00:00
|
|
|
g_free(sel);
|
|
|
|
|
2008-11-16 17:53:33 +00:00
|
|
|
/* add project's base path directory to the dir list, we do this here once
|
|
|
|
* (in create_fif_dialog() it would fail if a project is opened after dialog creation) */
|
2008-11-16 17:53:55 +00:00
|
|
|
if (app->project != NULL && NZV(app->project->base_path))
|
2008-11-16 17:53:33 +00:00
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
ui_combo_box_prepend_text_once(GTK_COMBO_BOX(fif_dlg.dir_combo),
|
2008-11-16 17:53:33 +00:00
|
|
|
app->project->base_path);
|
|
|
|
}
|
|
|
|
|
2011-10-11 21:30:28 -07:00
|
|
|
entry = gtk_bin_get_child(GTK_BIN(fif_dlg.dir_combo));
|
2007-12-02 10:52:19 +00:00
|
|
|
if (NZV(dir))
|
2008-06-25 11:56:58 +00:00
|
|
|
cur_dir = g_strdup(dir); /* custom directory argument passed */
|
2007-12-02 10:52:19 +00:00
|
|
|
else
|
2008-06-25 11:56:58 +00:00
|
|
|
{
|
2012-07-26 16:13:34 +01:00
|
|
|
if (search_prefs.use_current_file_dir)
|
2008-06-25 11:56:58 +00:00
|
|
|
{
|
2012-07-26 16:13:34 +01:00
|
|
|
static gchar *last_cur_dir = NULL;
|
|
|
|
static GeanyDocument *last_doc = NULL;
|
2008-06-25 11:56:58 +00:00
|
|
|
|
2012-07-26 16:13:34 +01:00
|
|
|
/* Only set the directory entry once for the current document */
|
|
|
|
cur_dir = utils_get_current_file_dir_utf8();
|
|
|
|
if (doc == last_doc && cur_dir && utils_str_equal(cur_dir, last_cur_dir))
|
|
|
|
{
|
|
|
|
/* in case the user now wants the current directory, add it to history */
|
|
|
|
ui_combo_box_add_to_history(
|
|
|
|
GTK_COMBO_BOX_ENTRY(fif_dlg.dir_combo), cur_dir, 0);
|
|
|
|
SETPTR(cur_dir, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
SETPTR(last_cur_dir, g_strdup(cur_dir));
|
|
|
|
|
|
|
|
last_doc = doc;
|
|
|
|
}
|
|
|
|
if (!cur_dir && ! NZV(gtk_entry_get_text(GTK_ENTRY(entry))))
|
|
|
|
{
|
2008-06-25 11:56:58 +00:00
|
|
|
/* use default_open_path if no directory could be determined
|
|
|
|
* (e.g. when no files are open) */
|
|
|
|
if (!cur_dir)
|
2008-12-16 13:38:10 +00:00
|
|
|
cur_dir = g_strdup(utils_get_default_dir_utf8());
|
|
|
|
if (!cur_dir)
|
|
|
|
cur_dir = g_get_current_dir();
|
2008-06-25 11:56:58 +00:00
|
|
|
}
|
|
|
|
}
|
2006-11-15 15:57:23 +00:00
|
|
|
if (cur_dir)
|
|
|
|
{
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), cur_dir);
|
|
|
|
g_free(cur_dir);
|
|
|
|
}
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2011-04-30 21:50:37 +00:00
|
|
|
update_fif_file_mode_combo();
|
|
|
|
update_file_patterns(fif_dlg.files_mode_combo, fif_dlg.files_combo);
|
|
|
|
|
2008-11-13 16:29:56 +00:00
|
|
|
/* set the encoding of the current file */
|
|
|
|
if (doc != NULL)
|
|
|
|
enc_idx = encodings_get_idx_from_charset(doc->encoding);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_combo_box_set_active(GTK_COMBO_BOX(fif_dlg.encoding_combo), enc_idx);
|
2008-11-13 16:29:56 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* put the focus to the directory entry if it is empty */
|
2006-12-07 16:09:45 +00:00
|
|
|
if (utils_str_equal(gtk_entry_get_text(GTK_ENTRY(entry)), ""))
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_grab_focus(fif_dlg.dir_combo);
|
2006-08-11 21:12:49 +00:00
|
|
|
else
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_grab_focus(fif_dlg.search_combo);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-12-29 18:45:49 +00:00
|
|
|
/* set dialog window position */
|
|
|
|
set_dialog_position(fif_dlg.dialog, fif_dlg.position);
|
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_show(fif_dlg.dialog);
|
2008-02-27 13:17:29 +00:00
|
|
|
/* bring the dialog back in the foreground in case it is already open but the focus is away */
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_window_present(GTK_WINDOW(fif_dlg.dialog));
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_find_replace_checkbutton_toggled(GtkToggleButton *togglebutton, gpointer user_data)
|
|
|
|
{
|
|
|
|
GtkWidget *dialog = GTK_WIDGET(user_data);
|
|
|
|
GtkToggleButton *chk_regexp = GTK_TOGGLE_BUTTON(
|
2008-12-18 21:21:53 +00:00
|
|
|
ui_lookup_widget(dialog, "check_regexp"));
|
2006-08-11 21:12:49 +00:00
|
|
|
|
|
|
|
if (togglebutton == chk_regexp)
|
|
|
|
{
|
|
|
|
gboolean regex_set = gtk_toggle_button_get_active(chk_regexp);
|
2008-12-18 21:21:53 +00:00
|
|
|
GtkWidget *check_word = ui_lookup_widget(dialog, "check_word");
|
|
|
|
GtkWidget *check_wordstart = ui_lookup_widget(dialog, "check_wordstart");
|
2010-03-01 17:07:56 +00:00
|
|
|
GtkWidget *check_escape = ui_lookup_widget(dialog, "check_escape");
|
2011-03-06 14:18:05 +00:00
|
|
|
gboolean replace = (dialog != find_dlg.dialog);
|
|
|
|
const char *back_button[2] = { "btn_previous" , "check_back" };
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* hide options that don't apply to regex searches */
|
2010-03-01 17:07:56 +00:00
|
|
|
gtk_widget_set_sensitive(check_escape, ! regex_set);
|
2011-03-06 14:18:05 +00:00
|
|
|
gtk_widget_set_sensitive(ui_lookup_widget(dialog, back_button[replace]), ! regex_set);
|
2006-08-11 21:12:49 +00:00
|
|
|
gtk_widget_set_sensitive(check_word, ! regex_set);
|
|
|
|
gtk_widget_set_sensitive(check_wordstart, ! regex_set);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-16 12:13:50 +00:00
|
|
|
/* Clears markers if text is null/empty.
|
|
|
|
* @return Number of matches marked. */
|
2009-07-10 15:17:27 +00:00
|
|
|
gint search_mark_all(GeanyDocument *doc, const gchar *search_text, gint flags)
|
2007-02-17 17:39:10 +00:00
|
|
|
{
|
2008-11-15 15:52:36 +00:00
|
|
|
gint pos, count = 0;
|
|
|
|
gsize len;
|
2009-07-04 09:20:28 +00:00
|
|
|
struct Sci_TextToFind ttf;
|
2007-02-17 17:39:10 +00:00
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
g_return_val_if_fail(doc != NULL, 0);
|
2007-02-17 17:39:10 +00:00
|
|
|
|
2008-11-15 15:53:40 +00:00
|
|
|
/* clear previous search indicators */
|
2008-11-16 17:54:28 +00:00
|
|
|
editor_indicator_clear(doc->editor, GEANY_INDICATOR_SEARCH);
|
2008-11-18 20:14:42 +00:00
|
|
|
|
2011-03-24 22:00:18 +00:00
|
|
|
if (G_UNLIKELY(! NZV(search_text)))
|
2009-07-16 12:21:56 +00:00
|
|
|
return 0;
|
|
|
|
|
2007-02-17 17:39:10 +00:00
|
|
|
ttf.chrg.cpMin = 0;
|
2008-07-14 11:13:54 +00:00
|
|
|
ttf.chrg.cpMax = sci_get_length(doc->editor->sci);
|
2007-02-17 17:39:10 +00:00
|
|
|
ttf.lpstrText = (gchar *)search_text;
|
2008-10-25 18:57:00 +00:00
|
|
|
while (TRUE)
|
2007-02-17 17:39:10 +00:00
|
|
|
{
|
2012-09-02 14:37:51 +02:00
|
|
|
pos = search_find_text(doc->editor->sci, flags, &ttf);
|
2007-02-17 17:39:10 +00:00
|
|
|
if (pos == -1) break;
|
|
|
|
|
2009-07-16 14:37:35 +00:00
|
|
|
len = ttf.chrgText.cpMax - ttf.chrgText.cpMin;
|
|
|
|
if (len)
|
|
|
|
editor_indicator_set_on_range(doc->editor, GEANY_INDICATOR_SEARCH, pos, pos + len);
|
2007-02-17 17:39:10 +00:00
|
|
|
|
2010-10-16 12:37:20 +00:00
|
|
|
ttf.chrg.cpMin = ttf.chrgText.cpMax;
|
2012-09-02 14:37:51 +02:00
|
|
|
/* make sure to advance even with empty matches (see find_document_usage()) */
|
|
|
|
if (len == 0)
|
|
|
|
ttf.chrg.cpMin ++;
|
2007-02-17 17:39:10 +00:00
|
|
|
count++;
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-04 11:26:31 +00:00
|
|
|
static void
|
|
|
|
on_find_entry_activate(GtkEntry *entry, gpointer user_data)
|
|
|
|
{
|
|
|
|
on_find_dialog_response(NULL, GEANY_RESPONSE_FIND,
|
2008-12-18 21:21:53 +00:00
|
|
|
ui_lookup_widget(GTK_WIDGET(entry), "entry"));
|
2006-11-04 11:26:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-24 16:51:02 +00:00
|
|
|
static void
|
|
|
|
on_find_entry_activate_backward(GtkEntry *entry, gpointer user_data)
|
|
|
|
{
|
|
|
|
/* can't search backwards with a regexp */
|
|
|
|
if (search_data.flags & SCFIND_REGEXP)
|
|
|
|
utils_beep();
|
|
|
|
else
|
|
|
|
on_find_dialog_response(NULL, GEANY_RESPONSE_FIND_PREVIOUS,
|
|
|
|
ui_lookup_widget(GTK_WIDGET(entry), "entry"));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-06 14:18:05 +00:00
|
|
|
#define int_search_flags(match_case, whole_word, regexp, word_start) \
|
|
|
|
((match_case ? SCFIND_MATCHCASE : 0) | \
|
|
|
|
(whole_word ? SCFIND_WHOLEWORD : 0) | \
|
|
|
|
(regexp ? SCFIND_REGEXP | SCFIND_POSIX : 0) | \
|
|
|
|
(word_start ? SCFIND_WORDSTART : 0))
|
2008-01-29 16:34:16 +00:00
|
|
|
|
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
static void
|
|
|
|
on_find_dialog_response(GtkDialog *dialog, gint response, gpointer user_data)
|
|
|
|
{
|
2009-12-29 18:45:49 +00:00
|
|
|
gtk_window_get_position(GTK_WINDOW(find_dlg.dialog),
|
|
|
|
&find_dlg.position[0], &find_dlg.position[1]);
|
|
|
|
|
2011-03-06 14:18:05 +00:00
|
|
|
stash_group_update(find_prefs, find_dlg.dialog);
|
|
|
|
|
2007-03-24 16:27:19 +00:00
|
|
|
if (response == GTK_RESPONSE_CANCEL || response == GTK_RESPONSE_DELETE_EVENT)
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_hide(find_dlg.dialog);
|
2006-11-01 15:26:41 +00:00
|
|
|
else
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2008-06-15 13:35:48 +00:00
|
|
|
GeanyDocument *doc = document_get_current();
|
2011-03-06 14:18:05 +00:00
|
|
|
gboolean check_close = settings.find_close_dialog;
|
2008-01-29 16:34:16 +00:00
|
|
|
|
2008-06-15 15:24:44 +00:00
|
|
|
if (doc == NULL)
|
|
|
|
return;
|
|
|
|
|
2006-11-01 15:26:41 +00:00
|
|
|
search_data.backwards = FALSE;
|
2007-07-11 17:44:43 +00:00
|
|
|
search_data.search_bar = FALSE;
|
2006-11-01 15:26:41 +00:00
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
g_free(search_data.text);
|
2011-06-03 13:40:42 +00:00
|
|
|
g_free(search_data.original_text);
|
2006-08-11 21:12:49 +00:00
|
|
|
search_data.text = g_strdup(gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(user_data)))));
|
2011-06-03 13:40:42 +00:00
|
|
|
search_data.original_text = g_strdup(search_data.text);
|
2011-03-06 14:18:05 +00:00
|
|
|
search_data.flags = int_search_flags(settings.find_case_sensitive,
|
|
|
|
settings.find_match_whole_word, settings.find_regexp, settings.find_match_word_start);
|
2010-03-01 17:07:56 +00:00
|
|
|
|
2011-03-24 22:00:18 +00:00
|
|
|
if (! NZV(search_data.text))
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2010-03-01 18:16:28 +00:00
|
|
|
fail:
|
2006-08-11 21:12:49 +00:00
|
|
|
utils_beep();
|
2009-02-27 14:05:50 +00:00
|
|
|
gtk_widget_grab_focus(find_dlg.entry);
|
2006-08-11 21:12:49 +00:00
|
|
|
return;
|
|
|
|
}
|
2012-01-03 16:21:12 +00:00
|
|
|
if (search_data.flags & SCFIND_REGEXP)
|
|
|
|
{
|
|
|
|
GRegex *regex = compile_regex(search_data.text, search_data.flags);
|
|
|
|
if (!regex)
|
|
|
|
goto fail;
|
2012-09-02 14:14:52 +02:00
|
|
|
else
|
|
|
|
g_regex_unref(regex);
|
2012-01-03 16:21:12 +00:00
|
|
|
}
|
|
|
|
else if (settings.find_escape_sequences)
|
2010-03-01 18:16:28 +00:00
|
|
|
{
|
2011-12-25 17:22:06 +00:00
|
|
|
if (! utils_str_replace_escape(search_data.text, FALSE))
|
2010-03-01 18:16:28 +00:00
|
|
|
goto fail;
|
|
|
|
}
|
2011-06-03 13:40:42 +00:00
|
|
|
ui_combo_box_add_to_history(GTK_COMBO_BOX_ENTRY(user_data), search_data.original_text, 0);
|
2008-01-29 16:34:16 +00:00
|
|
|
|
2006-11-01 15:26:41 +00:00
|
|
|
switch (response)
|
|
|
|
{
|
|
|
|
case GEANY_RESPONSE_FIND:
|
|
|
|
case GEANY_RESPONSE_FIND_PREVIOUS:
|
2009-02-27 14:05:50 +00:00
|
|
|
{
|
2011-06-03 13:40:42 +00:00
|
|
|
gint result = document_find_text(doc, search_data.text, search_data.original_text, search_data.flags,
|
2009-01-27 18:03:58 +00:00
|
|
|
(response == GEANY_RESPONSE_FIND_PREVIOUS), TRUE, GTK_WIDGET(find_dlg.dialog));
|
2009-02-27 14:05:50 +00:00
|
|
|
ui_set_search_entry_background(find_dlg.entry, (result > -1));
|
2011-12-05 21:24:33 +02:00
|
|
|
check_close = search_prefs.hide_find_dialog;
|
2007-02-17 17:39:10 +00:00
|
|
|
break;
|
2009-02-27 14:05:50 +00:00
|
|
|
}
|
2006-11-01 15:26:41 +00:00
|
|
|
case GEANY_RESPONSE_FIND_IN_FILE:
|
2011-06-03 13:40:42 +00:00
|
|
|
search_find_usage(search_data.text, search_data.original_text, search_data.flags, FALSE);
|
2007-02-17 17:39:10 +00:00
|
|
|
break;
|
2006-11-01 15:26:41 +00:00
|
|
|
|
|
|
|
case GEANY_RESPONSE_FIND_IN_SESSION:
|
2011-06-03 13:40:42 +00:00
|
|
|
search_find_usage(search_data.text, search_data.original_text, search_data.flags, TRUE);
|
2007-02-17 17:39:10 +00:00
|
|
|
break;
|
2006-11-01 15:26:41 +00:00
|
|
|
|
|
|
|
case GEANY_RESPONSE_MARK:
|
2008-06-15 13:35:48 +00:00
|
|
|
{
|
2009-07-10 15:17:27 +00:00
|
|
|
gint count = search_mark_all(doc, search_data.text, search_data.flags);
|
2008-06-15 13:35:48 +00:00
|
|
|
|
|
|
|
if (count == 0)
|
2011-06-03 13:40:42 +00:00
|
|
|
ui_set_statusbar(FALSE, _("No matches found for \"%s\"."), search_data.original_text);
|
2008-06-15 13:35:48 +00:00
|
|
|
else
|
|
|
|
ui_set_statusbar(FALSE,
|
|
|
|
ngettext("Found %d match for \"%s\".",
|
|
|
|
"Found %d matches for \"%s\".", count),
|
2011-06-03 13:40:42 +00:00
|
|
|
count, search_data.original_text);
|
2008-06-15 13:35:48 +00:00
|
|
|
}
|
|
|
|
break;
|
2006-11-01 15:26:41 +00:00
|
|
|
}
|
|
|
|
if (check_close)
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_hide(find_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2006-11-04 11:26:31 +00:00
|
|
|
on_replace_entry_activate(GtkEntry *entry, gpointer user_data)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2006-11-04 11:26:31 +00:00
|
|
|
on_replace_dialog_response(NULL, GEANY_RESPONSE_REPLACE, NULL);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-09 17:31:56 +00:00
|
|
|
static void replace_in_session(GeanyDocument *doc,
|
|
|
|
gint search_flags_re, gboolean search_replace_escape_re,
|
2011-06-03 13:40:42 +00:00
|
|
|
const gchar *find, const gchar *replace,
|
|
|
|
const gchar *original_find, const gchar *original_replace)
|
2009-12-09 17:31:56 +00:00
|
|
|
{
|
2010-01-18 17:05:13 +00:00
|
|
|
guint n, page_count, rep_count = 0, file_count = 0;
|
2009-12-09 17:31:56 +00:00
|
|
|
|
|
|
|
/* replace in all documents following notebook tab order */
|
|
|
|
page_count = gtk_notebook_get_n_pages(GTK_NOTEBOOK(main_widgets.notebook));
|
|
|
|
for (n = 0; n < page_count; n++)
|
|
|
|
{
|
|
|
|
GeanyDocument *tmp_doc = document_get_from_page(n);
|
2010-01-18 17:05:13 +00:00
|
|
|
gint reps = 0;
|
2009-12-09 17:31:56 +00:00
|
|
|
|
2011-06-03 13:40:42 +00:00
|
|
|
reps = document_replace_all(tmp_doc, find, replace, original_find, original_replace, search_flags_re);
|
2010-01-18 17:05:13 +00:00
|
|
|
rep_count += reps;
|
|
|
|
if (reps)
|
|
|
|
file_count++;
|
2009-12-09 17:31:56 +00:00
|
|
|
}
|
2010-01-18 17:05:13 +00:00
|
|
|
if (file_count == 0)
|
|
|
|
{
|
2009-12-09 17:31:56 +00:00
|
|
|
utils_beep();
|
2011-06-03 13:40:42 +00:00
|
|
|
ui_set_statusbar(FALSE, _("No matches found for \"%s\"."), original_find);
|
2010-01-18 17:05:13 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* if only one file was changed, don't override that document's status message
|
|
|
|
* so we don't have to translate 4 messages for ngettext */
|
|
|
|
if (file_count > 1)
|
|
|
|
ui_set_statusbar(FALSE, _("Replaced %u matches in %u documents."),
|
|
|
|
rep_count, file_count);
|
2009-12-09 17:31:56 +00:00
|
|
|
|
|
|
|
/* show which docs had replacements: */
|
|
|
|
gtk_notebook_set_current_page(GTK_NOTEBOOK(msgwindow.notebook), MSG_STATUS);
|
|
|
|
|
|
|
|
ui_save_buttons_toggle(doc->changed); /* update save all */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
static void
|
|
|
|
on_replace_dialog_response(GtkDialog *dialog, gint response, gpointer user_data)
|
|
|
|
{
|
2008-06-15 13:35:48 +00:00
|
|
|
GeanyDocument *doc = document_get_current();
|
2006-08-11 21:12:49 +00:00
|
|
|
gint search_flags_re;
|
2006-10-22 14:56:05 +00:00
|
|
|
gboolean search_backwards_re, search_replace_escape_re;
|
2011-06-03 13:40:42 +00:00
|
|
|
gchar *find, *replace, *original_find = NULL, *original_replace = NULL;
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2009-12-29 18:45:49 +00:00
|
|
|
gtk_window_get_position(GTK_WINDOW(replace_dlg.dialog),
|
|
|
|
&replace_dlg.position[0], &replace_dlg.position[1]);
|
|
|
|
|
2011-03-06 14:18:05 +00:00
|
|
|
stash_group_update(replace_prefs, replace_dlg.dialog);
|
|
|
|
|
2007-03-24 16:27:19 +00:00
|
|
|
if (response == GTK_RESPONSE_CANCEL || response == GTK_RESPONSE_DELETE_EVENT)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_hide(replace_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-03-06 14:18:05 +00:00
|
|
|
search_backwards_re = settings.replace_search_backwards;
|
|
|
|
search_replace_escape_re = settings.replace_escape_sequences;
|
2009-02-27 14:05:50 +00:00
|
|
|
find = g_strdup(gtk_entry_get_text(GTK_ENTRY(replace_dlg.find_entry)));
|
|
|
|
replace = g_strdup(gtk_entry_get_text(GTK_ENTRY(replace_dlg.replace_entry)));
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2011-03-06 14:18:05 +00:00
|
|
|
search_flags_re = int_search_flags(settings.replace_case_sensitive,
|
|
|
|
settings.replace_match_whole_word, settings.replace_regexp,
|
|
|
|
settings.replace_match_word_start);
|
2008-01-29 16:34:16 +00:00
|
|
|
|
|
|
|
if ((response != GEANY_RESPONSE_FIND) && (search_flags_re & SCFIND_MATCHCASE)
|
|
|
|
&& (strcmp(find, replace) == 0))
|
2011-05-17 13:47:36 +00:00
|
|
|
goto fail;
|
2011-06-03 13:40:42 +00:00
|
|
|
|
|
|
|
original_find = g_strdup(find);
|
|
|
|
original_replace = g_strdup(replace);
|
2011-12-25 17:22:06 +00:00
|
|
|
|
2011-12-26 13:01:54 +00:00
|
|
|
if (search_flags_re & SCFIND_REGEXP)
|
|
|
|
{
|
2012-01-03 16:21:12 +00:00
|
|
|
GRegex *regex = compile_regex(find, search_flags_re);
|
2012-09-02 14:14:52 +02:00
|
|
|
if (regex)
|
|
|
|
g_regex_unref(regex);
|
2011-12-26 13:01:54 +00:00
|
|
|
/* find escapes will be handled by GRegex */
|
2012-01-03 16:21:12 +00:00
|
|
|
if (!regex || !utils_str_replace_escape(replace, TRUE))
|
2011-12-26 13:01:54 +00:00
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
else if (search_replace_escape_re)
|
2010-03-01 18:16:28 +00:00
|
|
|
{
|
|
|
|
if (! utils_str_replace_escape(find, FALSE) ||
|
|
|
|
! utils_str_replace_escape(replace, FALSE))
|
|
|
|
goto fail;
|
|
|
|
}
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2010-06-18 12:20:15 +00:00
|
|
|
ui_combo_box_add_to_history(GTK_COMBO_BOX_ENTRY(
|
2011-06-03 13:41:00 +00:00
|
|
|
gtk_widget_get_parent(replace_dlg.find_entry)), original_find, 0);
|
2010-06-18 12:20:15 +00:00
|
|
|
ui_combo_box_add_to_history(GTK_COMBO_BOX_ENTRY(
|
2011-06-03 13:41:00 +00:00
|
|
|
gtk_widget_get_parent(replace_dlg.replace_entry)), original_replace, 0);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2006-10-22 14:56:05 +00:00
|
|
|
switch (response)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2006-10-22 14:56:05 +00:00
|
|
|
case GEANY_RESPONSE_REPLACE_AND_FIND:
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2011-06-03 13:40:42 +00:00
|
|
|
gint rep = document_replace_text(doc, find, original_find, replace, search_flags_re,
|
2006-10-22 14:56:05 +00:00
|
|
|
search_backwards_re);
|
|
|
|
if (rep != -1)
|
2011-06-03 13:40:42 +00:00
|
|
|
document_find_text(doc, find, original_find, search_flags_re, search_backwards_re,
|
2007-07-04 17:08:53 +00:00
|
|
|
TRUE, NULL);
|
2006-10-22 14:56:05 +00:00
|
|
|
break;
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
2006-10-22 14:56:05 +00:00
|
|
|
case GEANY_RESPONSE_REPLACE:
|
2011-06-03 13:40:42 +00:00
|
|
|
document_replace_text(doc, find, original_find, replace, search_flags_re,
|
2006-10-22 14:56:05 +00:00
|
|
|
search_backwards_re);
|
|
|
|
break;
|
2009-12-09 17:31:56 +00:00
|
|
|
|
2006-10-22 14:56:05 +00:00
|
|
|
case GEANY_RESPONSE_FIND:
|
|
|
|
{
|
2011-06-03 13:40:42 +00:00
|
|
|
gint result = document_find_text(doc, find, original_find, search_flags_re,
|
2009-02-27 14:05:50 +00:00
|
|
|
search_backwards_re, TRUE, GTK_WIDGET(dialog));
|
|
|
|
ui_set_search_entry_background(replace_dlg.find_entry, (result > -1));
|
2006-10-22 14:56:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case GEANY_RESPONSE_REPLACE_IN_FILE:
|
2011-06-03 13:40:42 +00:00
|
|
|
if (! document_replace_all(doc, find, replace, original_find, original_replace, search_flags_re))
|
2008-02-25 17:27:54 +00:00
|
|
|
utils_beep();
|
2006-10-22 14:56:05 +00:00
|
|
|
break;
|
2006-11-25 16:33:38 +00:00
|
|
|
|
2009-12-09 17:31:56 +00:00
|
|
|
case GEANY_RESPONSE_REPLACE_IN_SESSION:
|
2011-06-03 13:40:42 +00:00
|
|
|
replace_in_session(doc, search_flags_re, search_replace_escape_re, find, replace, original_find, original_replace);
|
2006-10-22 14:56:05 +00:00
|
|
|
break;
|
2009-12-09 17:31:56 +00:00
|
|
|
|
2006-10-22 14:56:05 +00:00
|
|
|
case GEANY_RESPONSE_REPLACE_IN_SEL:
|
2011-06-03 13:40:42 +00:00
|
|
|
document_replace_sel(doc, find, replace, original_find, original_replace, search_flags_re);
|
2006-10-22 14:56:05 +00:00
|
|
|
break;
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
2008-02-25 17:27:54 +00:00
|
|
|
switch (response)
|
|
|
|
{
|
|
|
|
case GEANY_RESPONSE_REPLACE_IN_SEL:
|
|
|
|
case GEANY_RESPONSE_REPLACE_IN_FILE:
|
|
|
|
case GEANY_RESPONSE_REPLACE_IN_SESSION:
|
2011-03-06 14:18:05 +00:00
|
|
|
if (settings.replace_close_dialog)
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_hide(replace_dlg.dialog);
|
2008-02-25 17:27:54 +00:00
|
|
|
}
|
2006-08-11 21:12:49 +00:00
|
|
|
g_free(find);
|
|
|
|
g_free(replace);
|
2011-06-03 13:40:42 +00:00
|
|
|
g_free(original_find);
|
|
|
|
g_free(original_replace);
|
2011-05-17 13:47:36 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
utils_beep();
|
|
|
|
gtk_widget_grab_focus(replace_dlg.find_entry);
|
|
|
|
g_free(find);
|
|
|
|
g_free(replace);
|
2011-06-03 13:40:42 +00:00
|
|
|
g_free(original_find);
|
|
|
|
g_free(original_replace);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 11:24:23 +00:00
|
|
|
static GString *get_grep_options(void)
|
2007-04-03 15:57:12 +00:00
|
|
|
{
|
2008-02-27 13:17:29 +00:00
|
|
|
GString *gstr = g_string_new("-nHI"); /* line numbers, filenames, ignore binaries */
|
2007-04-03 15:57:12 +00:00
|
|
|
|
2010-06-23 16:16:15 +00:00
|
|
|
if (settings.fif_invert_results)
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_append_c(gstr, 'v');
|
2010-06-23 16:16:15 +00:00
|
|
|
if (! settings.fif_case_sensitive)
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_append_c(gstr, 'i');
|
2010-06-23 16:16:15 +00:00
|
|
|
if (settings.fif_match_whole_word)
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_append_c(gstr, 'w');
|
2010-06-23 16:16:15 +00:00
|
|
|
if (settings.fif_recursive)
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_append_c(gstr, 'r');
|
|
|
|
|
2010-06-21 14:57:03 +00:00
|
|
|
if (!settings.fif_regexp)
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_append_c(gstr, 'F');
|
2010-06-21 14:57:03 +00:00
|
|
|
else
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_append_c(gstr, 'E');
|
|
|
|
|
2010-06-23 16:16:15 +00:00
|
|
|
if (settings.fif_use_extra_options)
|
2007-04-03 15:57:12 +00:00
|
|
|
{
|
2008-12-28 17:25:09 +00:00
|
|
|
g_strstrip(settings.fif_extra_options);
|
2007-04-03 15:57:12 +00:00
|
|
|
|
2008-12-28 17:25:09 +00:00
|
|
|
if (*settings.fif_extra_options != 0)
|
2007-04-03 15:57:12 +00:00
|
|
|
{
|
|
|
|
g_string_append_c(gstr, ' ');
|
2008-12-28 17:25:09 +00:00
|
|
|
g_string_append(gstr, settings.fif_extra_options);
|
2007-04-03 15:57:12 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-24 12:17:33 +00:00
|
|
|
g_strstrip(settings.fif_files);
|
2011-04-30 21:50:37 +00:00
|
|
|
if (settings.fif_files_mode != FILES_MODE_ALL && *settings.fif_files)
|
2010-06-23 12:34:11 +00:00
|
|
|
{
|
|
|
|
GString *tmp;
|
|
|
|
|
|
|
|
/* put --include= before each pattern */
|
|
|
|
tmp = g_string_new(settings.fif_files);
|
|
|
|
do {} while (utils_string_replace_all(tmp, " ", " "));
|
|
|
|
g_string_prepend_c(tmp, ' ');
|
|
|
|
utils_string_replace_all(tmp, " ", " --include=");
|
|
|
|
g_string_append(gstr, tmp->str);
|
|
|
|
g_string_free(tmp, TRUE);
|
|
|
|
}
|
2007-04-03 15:57:12 +00:00
|
|
|
return gstr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
static void
|
2008-11-06 17:13:18 +00:00
|
|
|
on_find_in_files_dialog_response(GtkDialog *dialog, gint response,
|
|
|
|
G_GNUC_UNUSED gpointer user_data)
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2009-12-29 18:45:49 +00:00
|
|
|
gtk_window_get_position(GTK_WINDOW(fif_dlg.dialog), &fif_dlg.position[0], &fif_dlg.position[1]);
|
|
|
|
|
2009-01-27 18:03:58 +00:00
|
|
|
stash_group_update(fif_prefs, fif_dlg.dialog);
|
2008-12-16 13:01:47 +00:00
|
|
|
|
2006-08-11 21:12:49 +00:00
|
|
|
if (response == GTK_RESPONSE_ACCEPT)
|
|
|
|
{
|
2009-01-27 18:03:58 +00:00
|
|
|
GtkWidget *search_combo = fif_dlg.search_combo;
|
2006-08-11 21:12:49 +00:00
|
|
|
const gchar *search_text =
|
2008-11-06 17:26:29 +00:00
|
|
|
gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(search_combo))));
|
2009-01-27 18:03:58 +00:00
|
|
|
GtkWidget *dir_combo = fif_dlg.dir_combo;
|
2006-08-11 21:12:49 +00:00
|
|
|
const gchar *utf8_dir =
|
2006-11-15 15:57:23 +00:00
|
|
|
gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(dir_combo))));
|
2008-11-13 16:29:56 +00:00
|
|
|
GeanyEncodingIndex enc_idx = gtk_combo_box_get_active(
|
2009-01-27 18:03:58 +00:00
|
|
|
GTK_COMBO_BOX(fif_dlg.encoding_combo));
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2011-03-24 22:00:18 +00:00
|
|
|
if (G_UNLIKELY(! NZV(utf8_dir)))
|
2007-10-24 10:52:48 +00:00
|
|
|
ui_set_statusbar(FALSE, _("Invalid directory for find in files."));
|
2008-11-06 17:13:18 +00:00
|
|
|
else if (NZV(search_text))
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
|
|
|
gchar *locale_dir;
|
2007-04-03 15:57:12 +00:00
|
|
|
GString *opts = get_grep_options();
|
2008-11-13 16:29:56 +00:00
|
|
|
const gchar *enc = (enc_idx == GEANY_ENCODING_UTF_8) ? NULL :
|
|
|
|
encodings_get_charset_from_index(enc_idx);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2006-08-13 09:07:10 +00:00
|
|
|
locale_dir = utils_get_locale_from_utf8(utf8_dir);
|
2006-08-11 21:12:49 +00:00
|
|
|
|
2008-11-13 16:29:56 +00:00
|
|
|
if (search_find_in_files(search_text, locale_dir, opts->str, enc))
|
2006-08-11 21:12:49 +00:00
|
|
|
{
|
2010-06-18 12:20:15 +00:00
|
|
|
ui_combo_box_add_to_history(GTK_COMBO_BOX_ENTRY(search_combo), search_text, 0);
|
2010-06-23 16:06:10 +00:00
|
|
|
ui_combo_box_add_to_history(GTK_COMBO_BOX_ENTRY(fif_dlg.files_combo), NULL, 0);
|
2010-06-18 12:20:15 +00:00
|
|
|
ui_combo_box_add_to_history(GTK_COMBO_BOX_ENTRY(dir_combo), utf8_dir, 0);
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_hide(fif_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
g_free(locale_dir);
|
2007-04-03 15:57:12 +00:00
|
|
|
g_string_free(opts, TRUE);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
else
|
2007-10-24 10:52:48 +00:00
|
|
|
ui_set_statusbar(FALSE, _("No text to find."));
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
else
|
2009-01-27 18:03:58 +00:00
|
|
|
gtk_widget_hide(fif_dlg.dialog);
|
2006-08-11 21:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-09-20 11:03:33 +00:00
|
|
|
static gboolean
|
2008-11-13 16:29:56 +00:00
|
|
|
search_find_in_files(const gchar *utf8_search_text, const gchar *dir, const gchar *opts,
|
|
|
|
const gchar *enc)
|
2006-07-13 14:30:44 +00:00
|
|
|
{
|
2007-04-03 15:57:12 +00:00
|
|
|
gchar **argv_prefix, **argv, **opts_argv;
|
2008-03-13 09:33:23 +00:00
|
|
|
gchar *command_grep;
|
2008-11-13 16:29:56 +00:00
|
|
|
gchar *search_text = NULL;
|
2012-04-10 04:07:16 +02:00
|
|
|
gint opts_argv_len, i;
|
2006-07-13 14:30:44 +00:00
|
|
|
GPid child_pid;
|
2007-10-23 15:23:08 +00:00
|
|
|
gint stdout_fd;
|
2008-11-16 17:53:13 +00:00
|
|
|
gint stderr_fd;
|
2006-07-13 14:30:44 +00:00
|
|
|
GError *error = NULL;
|
|
|
|
gboolean ret = FALSE;
|
2008-11-13 16:29:56 +00:00
|
|
|
gssize utf8_text_len;
|
2006-07-13 14:30:44 +00:00
|
|
|
|
2008-11-13 16:29:56 +00:00
|
|
|
if (! NZV(utf8_search_text) || ! dir) return TRUE;
|
2006-07-28 18:37:32 +00:00
|
|
|
|
2008-05-16 12:08:39 +00:00
|
|
|
command_grep = g_find_program_in_path(tool_prefs.grep_cmd);
|
2008-03-13 09:33:23 +00:00
|
|
|
if (command_grep == NULL)
|
2006-07-13 14:30:44 +00:00
|
|
|
{
|
2007-10-24 10:52:48 +00:00
|
|
|
ui_set_statusbar(TRUE, _("Cannot execute grep tool '%s';"
|
2008-05-16 12:08:39 +00:00
|
|
|
" check the path setting in Preferences."), tool_prefs.grep_cmd);
|
2006-07-13 14:30:44 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2012-04-10 04:07:16 +02:00
|
|
|
if (! g_shell_parse_argv(opts, &opts_argv_len, &opts_argv, &error))
|
|
|
|
{
|
|
|
|
ui_set_statusbar(TRUE, _("Cannot parse extra options: %s"), error->message);
|
|
|
|
g_error_free(error);
|
|
|
|
g_free(command_grep);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2008-11-13 16:29:56 +00:00
|
|
|
/* convert the search text in the preferred encoding (if the text is not valid UTF-8. assume
|
|
|
|
* it is already in the preferred encoding) */
|
|
|
|
utf8_text_len = strlen(utf8_search_text);
|
|
|
|
if (enc != NULL && g_utf8_validate(utf8_search_text, utf8_text_len, NULL))
|
|
|
|
{
|
|
|
|
search_text = g_convert(utf8_search_text, utf8_text_len, enc, "UTF-8", NULL, NULL, NULL);
|
|
|
|
}
|
|
|
|
if (search_text == NULL)
|
|
|
|
search_text = g_strdup(utf8_search_text);
|
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* set grep command and options */
|
|
|
|
argv_prefix = g_new0(gchar*, 1 + opts_argv_len + 3 + 1); /* last +1 for recursive arg */
|
2007-04-03 11:04:44 +00:00
|
|
|
|
2008-03-13 09:33:23 +00:00
|
|
|
argv_prefix[0] = command_grep;
|
2007-04-03 15:57:12 +00:00
|
|
|
for (i = 0; i < opts_argv_len; i++)
|
2006-07-28 18:37:32 +00:00
|
|
|
{
|
2007-04-03 15:57:12 +00:00
|
|
|
argv_prefix[i + 1] = g_strdup(opts_argv[i]);
|
2006-07-28 18:37:32 +00:00
|
|
|
}
|
2007-04-03 15:57:12 +00:00
|
|
|
g_strfreev(opts_argv);
|
2006-07-28 18:37:32 +00:00
|
|
|
|
2008-05-16 12:08:39 +00:00
|
|
|
i++; /* correct for tool_prefs.grep_cmd */
|
2007-04-03 11:04:44 +00:00
|
|
|
argv_prefix[i++] = g_strdup("--");
|
2010-08-12 17:14:47 +00:00
|
|
|
argv_prefix[i++] = search_text;
|
2007-04-03 11:04:44 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
/* finally add the arguments(files to be searched) */
|
|
|
|
if (strstr(argv_prefix[1], "r")) /* recursive option set */
|
2007-04-03 11:04:44 +00:00
|
|
|
{
|
2010-07-06 14:01:26 +00:00
|
|
|
/* Use '.' so we get relative paths in the output */
|
2007-04-03 11:04:44 +00:00
|
|
|
argv_prefix[i++] = g_strdup(".");
|
|
|
|
argv_prefix[i++] = NULL;
|
|
|
|
argv = argv_prefix;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
argv_prefix[i++] = NULL;
|
|
|
|
argv = search_get_argv((const gchar**)argv_prefix, dir);
|
|
|
|
g_strfreev(argv_prefix);
|
|
|
|
}
|
2006-09-20 11:03:33 +00:00
|
|
|
|
2008-02-27 13:17:29 +00:00
|
|
|
if (argv == NULL) /* no files */
|
2007-04-03 15:57:12 +00:00
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_list_store_clear(msgwindow.store_msg);
|
|
|
|
gtk_notebook_set_current_page(GTK_NOTEBOOK(msgwindow.notebook), MSG_MESSAGE);
|
2006-07-13 14:30:44 +00:00
|
|
|
|
2008-11-16 17:53:13 +00:00
|
|
|
if (! g_spawn_async_with_pipes(dir, (gchar**)argv, NULL, G_SPAWN_DO_NOT_REAP_CHILD,
|
2006-07-13 14:30:44 +00:00
|
|
|
NULL, NULL, &child_pid,
|
2008-11-16 17:53:13 +00:00
|
|
|
NULL, &stdout_fd, &stderr_fd, &error))
|
2006-07-13 14:30:44 +00:00
|
|
|
{
|
2009-01-21 22:49:21 +00:00
|
|
|
geany_debug("%s: g_spawn_async_with_pipes() failed: %s", G_STRFUNC, error->message);
|
2007-10-24 10:52:48 +00:00
|
|
|
ui_set_statusbar(TRUE, _("Process failed (%s)"), error->message);
|
2006-07-13 14:30:44 +00:00
|
|
|
g_error_free(error);
|
|
|
|
ret = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-02-16 17:09:53 +00:00
|
|
|
gchar *str, *utf8_str;
|
|
|
|
|
2009-01-27 20:19:43 +00:00
|
|
|
ui_progress_bar_start(_("Searching..."));
|
|
|
|
|
2010-09-16 15:31:23 +00:00
|
|
|
msgwin_set_messages_dir(dir);
|
2008-11-13 16:29:56 +00:00
|
|
|
/* we can pass 'enc' without strdup'ing it here because it's a global const string and
|
|
|
|
* always exits longer than the lifetime of this function */
|
2009-09-16 14:13:38 +00:00
|
|
|
utils_set_up_io_channel(stdout_fd, G_IO_IN | G_IO_PRI | G_IO_ERR | G_IO_HUP | G_IO_NVAL,
|
2008-11-13 16:29:56 +00:00
|
|
|
TRUE, search_read_io, (gpointer) enc);
|
2009-09-16 14:13:38 +00:00
|
|
|
utils_set_up_io_channel(stderr_fd, G_IO_IN | G_IO_PRI | G_IO_ERR | G_IO_HUP | G_IO_NVAL,
|
2008-11-16 17:53:13 +00:00
|
|
|
TRUE, search_read_io_stderr, (gpointer) enc);
|
2006-07-28 18:37:32 +00:00
|
|
|
g_child_watch_add(child_pid, search_close_pid, NULL);
|
2007-02-16 17:09:53 +00:00
|
|
|
|
2007-04-03 15:57:12 +00:00
|
|
|
str = g_strdup_printf(_("%s %s -- %s (in directory: %s)"),
|
2008-11-13 16:29:56 +00:00
|
|
|
tool_prefs.grep_cmd, opts, utf8_search_text, dir);
|
2007-02-16 17:09:53 +00:00
|
|
|
utf8_str = utils_get_utf8_from_locale(str);
|
2008-12-05 17:30:06 +00:00
|
|
|
msgwin_msg_add_string(COLOR_BLUE, -1, NULL, utf8_str);
|
2009-10-15 16:15:28 +00:00
|
|
|
utils_free_pointers(2, str, utf8_str, NULL);
|
2006-07-13 14:30:44 +00:00
|
|
|
ret = TRUE;
|
|
|
|
}
|
|
|
|
g_strfreev(argv);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-01 14:38:57 +00:00
|
|
|
static gboolean pattern_list_match(GSList *patterns, const gchar *str)
|
|
|
|
{
|
|
|
|
GSList *item;
|
|
|
|
|
|
|
|
foreach_slist(item, patterns)
|
|
|
|
{
|
|
|
|
if (g_pattern_match_string(item->data, str))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-13 14:30:44 +00:00
|
|
|
/* Creates an argument vector of strings, copying argv_prefix[] values for
|
|
|
|
* the first arguments, then followed by filenames found in dir.
|
|
|
|
* Returns NULL if no files were found, otherwise returned vector should be fully freed. */
|
|
|
|
static gchar **search_get_argv(const gchar **argv_prefix, const gchar *dir)
|
|
|
|
{
|
2011-05-01 14:38:57 +00:00
|
|
|
guint prefix_len, list_len, i, j;
|
2006-07-13 14:30:44 +00:00
|
|
|
gchar **argv;
|
2011-05-01 14:38:57 +00:00
|
|
|
GSList *list, *item, *patterns = NULL;
|
2007-05-11 11:42:00 +00:00
|
|
|
GError *error = NULL;
|
2006-09-20 11:03:33 +00:00
|
|
|
|
2006-07-13 14:30:44 +00:00
|
|
|
g_return_val_if_fail(dir != NULL, NULL);
|
|
|
|
|
|
|
|
prefix_len = g_strv_length((gchar**)argv_prefix);
|
2007-05-11 11:42:00 +00:00
|
|
|
list = utils_get_file_list(dir, &list_len, &error);
|
|
|
|
if (error)
|
|
|
|
{
|
2007-10-24 10:52:48 +00:00
|
|
|
ui_set_statusbar(TRUE, _("Could not open directory (%s)"), error->message);
|
2007-05-11 11:42:00 +00:00
|
|
|
g_error_free(error);
|
|
|
|
return NULL;
|
|
|
|
}
|
2010-09-13 15:03:18 +00:00
|
|
|
if (list == NULL)
|
|
|
|
return NULL;
|
2006-07-13 14:30:44 +00:00
|
|
|
|
|
|
|
argv = g_new(gchar*, prefix_len + list_len + 1);
|
|
|
|
|
2011-05-01 14:38:57 +00:00
|
|
|
for (i = 0, j = 0; i < prefix_len; i++)
|
|
|
|
{
|
|
|
|
if (g_str_has_prefix(argv_prefix[i], "--include="))
|
|
|
|
{
|
|
|
|
const gchar *pat = &(argv_prefix[i][10]); /* the pattern part of the argument */
|
|
|
|
|
|
|
|
patterns = g_slist_prepend(patterns, g_pattern_spec_new(pat));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
argv[j++] = g_strdup(argv_prefix[i]);
|
|
|
|
}
|
2006-07-13 14:30:44 +00:00
|
|
|
|
2011-05-01 14:38:57 +00:00
|
|
|
if (patterns)
|
|
|
|
{
|
|
|
|
GSList *pat;
|
|
|
|
|
|
|
|
foreach_slist(item, list)
|
|
|
|
{
|
|
|
|
if (pattern_list_match(patterns, item->data))
|
|
|
|
argv[j++] = item->data;
|
|
|
|
else
|
|
|
|
g_free(item->data);
|
|
|
|
}
|
|
|
|
foreach_slist(pat, patterns)
|
|
|
|
g_pattern_spec_free(pat->data);
|
|
|
|
g_slist_free(patterns);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
foreach_slist(item, list)
|
|
|
|
argv[j++] = item->data;
|
|
|
|
}
|
2006-07-13 14:30:44 +00:00
|
|
|
|
2011-05-01 14:38:57 +00:00
|
|
|
argv[j] = NULL;
|
2006-07-13 14:30:44 +00:00
|
|
|
g_slist_free(list);
|
|
|
|
return argv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-15 18:00:32 +00:00
|
|
|
static gboolean read_fif_io(GIOChannel *source, GIOCondition condition, gchar *enc, gint msg_color)
|
2006-07-13 14:30:44 +00:00
|
|
|
{
|
|
|
|
if (condition & (G_IO_IN | G_IO_PRI))
|
|
|
|
{
|
2008-11-15 15:53:05 +00:00
|
|
|
gchar *msg, *utf8_msg;
|
2006-07-13 14:30:44 +00:00
|
|
|
|
|
|
|
while (g_io_channel_read_line(source, &msg, NULL, NULL, NULL) && msg)
|
|
|
|
{
|
2008-11-15 15:53:05 +00:00
|
|
|
utf8_msg = NULL;
|
2008-11-18 20:14:42 +00:00
|
|
|
|
2008-11-13 16:29:56 +00:00
|
|
|
g_strstrip(msg);
|
2008-11-19 17:32:24 +00:00
|
|
|
/* enc is NULL when encoding is set to UTF-8, so we can skip any conversion */
|
|
|
|
if (enc != NULL)
|
2008-11-13 16:29:56 +00:00
|
|
|
{
|
2008-11-19 17:32:24 +00:00
|
|
|
if (! g_utf8_validate(msg, -1, NULL))
|
|
|
|
{
|
|
|
|
utf8_msg = g_convert(msg, -1, "UTF-8", enc, NULL, NULL, NULL);
|
|
|
|
}
|
|
|
|
if (utf8_msg == NULL)
|
|
|
|
utf8_msg = msg;
|
2008-11-13 16:29:56 +00:00
|
|
|
}
|
2008-11-19 17:32:24 +00:00
|
|
|
else
|
|
|
|
utf8_msg = msg;
|
|
|
|
|
2009-09-15 18:00:32 +00:00
|
|
|
msgwin_msg_add_string(msg_color, -1, NULL, utf8_msg);
|
2008-11-19 17:32:24 +00:00
|
|
|
|
|
|
|
if (utf8_msg != msg)
|
|
|
|
g_free(utf8_msg);
|
2006-07-13 14:30:44 +00:00
|
|
|
g_free(msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (condition & (G_IO_ERR | G_IO_HUP | G_IO_NVAL))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-15 18:00:32 +00:00
|
|
|
static gboolean search_read_io(GIOChannel *source, GIOCondition condition, gpointer data)
|
2008-11-16 17:53:13 +00:00
|
|
|
{
|
2009-09-15 18:00:32 +00:00
|
|
|
return read_fif_io(source, condition, data, COLOR_BLACK);
|
|
|
|
}
|
2008-11-19 17:32:24 +00:00
|
|
|
|
2008-11-16 17:53:13 +00:00
|
|
|
|
2009-09-15 18:00:32 +00:00
|
|
|
static gboolean search_read_io_stderr(GIOChannel *source, GIOCondition condition, gpointer data)
|
|
|
|
{
|
|
|
|
return read_fif_io(source, condition, data, COLOR_DARK_RED);
|
2008-11-16 17:53:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-13 14:30:44 +00:00
|
|
|
static void search_close_pid(GPid child_pid, gint status, gpointer user_data)
|
|
|
|
{
|
2009-09-15 18:00:32 +00:00
|
|
|
const gchar *msg = _("Search failed.");
|
2012-11-27 14:16:30 +00:00
|
|
|
#ifdef G_OS_UNIX
|
2008-11-19 17:31:54 +00:00
|
|
|
gint exit_status = 1;
|
2006-07-13 14:30:44 +00:00
|
|
|
|
|
|
|
if (WIFEXITED(status))
|
|
|
|
{
|
2008-11-19 17:31:54 +00:00
|
|
|
exit_status = WEXITSTATUS(status);
|
|
|
|
}
|
|
|
|
else if (WIFSIGNALED(status))
|
|
|
|
{
|
|
|
|
exit_status = -1;
|
|
|
|
g_warning("Find in Files: The command failed unexpectedly (signal received).");
|
|
|
|
}
|
2012-11-27 14:16:30 +00:00
|
|
|
#else
|
|
|
|
gint exit_status = status;
|
|
|
|
#endif
|
2008-11-19 17:31:54 +00:00
|
|
|
|
|
|
|
switch (exit_status)
|
|
|
|
{
|
|
|
|
case 0:
|
2006-07-14 15:50:29 +00:00
|
|
|
{
|
2008-11-19 17:31:54 +00:00
|
|
|
gint count = gtk_tree_model_iter_n_children(
|
|
|
|
GTK_TREE_MODEL(msgwindow.store_msg), NULL) - 1;
|
|
|
|
gchar *text = ngettext(
|
|
|
|
"Search completed with %d match.",
|
|
|
|
"Search completed with %d matches.", count);
|
|
|
|
|
2008-12-05 17:30:06 +00:00
|
|
|
msgwin_msg_add(COLOR_BLUE, -1, NULL, text, count);
|
2008-11-19 17:31:54 +00:00
|
|
|
ui_set_statusbar(FALSE, text, count);
|
|
|
|
break;
|
2006-07-14 15:50:29 +00:00
|
|
|
}
|
2008-11-19 17:31:54 +00:00
|
|
|
case 1:
|
|
|
|
msg = _("No matches found.");
|
|
|
|
default:
|
2009-09-15 18:00:32 +00:00
|
|
|
msgwin_msg_add_string(COLOR_BLUE, -1, NULL, msg);
|
2008-11-19 17:31:54 +00:00
|
|
|
ui_set_statusbar(FALSE, "%s", msg);
|
|
|
|
break;
|
2006-07-13 14:30:44 +00:00
|
|
|
}
|
|
|
|
utils_beep();
|
|
|
|
g_spawn_close_pid(child_pid);
|
2009-01-27 20:19:43 +00:00
|
|
|
ui_progress_bar_stop();
|
2006-07-13 14:30:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
static GRegex *compile_regex(const gchar *str, gint sflags)
|
2010-03-02 13:21:41 +00:00
|
|
|
{
|
2011-12-24 17:41:09 +00:00
|
|
|
GRegex *regex;
|
|
|
|
GError *error = NULL;
|
|
|
|
gint rflags = G_REGEX_MULTILINE;
|
2010-03-02 13:21:41 +00:00
|
|
|
|
|
|
|
if (~sflags & SCFIND_MATCHCASE)
|
2011-12-24 17:41:09 +00:00
|
|
|
rflags |= G_REGEX_CASELESS;
|
2010-05-11 12:10:31 +00:00
|
|
|
if (sflags & (SCFIND_WHOLEWORD | SCFIND_WORDSTART))
|
|
|
|
{
|
2010-07-02 13:54:36 +00:00
|
|
|
geany_debug("%s: Unsupported regex flags found!", G_STRFUNC);
|
2010-05-11 12:10:31 +00:00
|
|
|
}
|
2010-03-02 13:21:41 +00:00
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
regex = g_regex_new(str, rflags, 0, &error);
|
|
|
|
if (!regex)
|
2010-03-02 13:21:41 +00:00
|
|
|
{
|
2011-12-24 17:41:09 +00:00
|
|
|
ui_set_statusbar(FALSE, _("Bad regex: %s"), error->message);
|
|
|
|
g_error_free(error);
|
2010-03-02 13:21:41 +00:00
|
|
|
}
|
2011-12-24 17:41:09 +00:00
|
|
|
return regex;
|
2010-03-02 13:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
typedef struct CharOffsets
|
|
|
|
{
|
|
|
|
gint start, end;
|
|
|
|
} CharOffsets;
|
|
|
|
|
|
|
|
static CharOffsets regex_matches[10];
|
|
|
|
|
2010-03-02 13:21:41 +00:00
|
|
|
/* groups that don't exist are handled OK as len = end - start = (-1) - (-1) = 0 */
|
2011-12-24 17:41:09 +00:00
|
|
|
static gchar *get_regex_match_string(const gchar *text, CharOffsets *match)
|
2010-03-02 13:21:41 +00:00
|
|
|
{
|
2011-12-24 17:41:09 +00:00
|
|
|
return g_strndup(&text[match->start], match->end - match->start);
|
2010-03-02 13:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
/* All matching text from regex_matches[0].start to regex_matches[0].end */
|
2010-03-02 13:21:41 +00:00
|
|
|
static gchar *regex_match_text = NULL;
|
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
static gint find_regex(ScintillaObject *sci, guint pos, GRegex *regex)
|
2010-03-02 13:21:41 +00:00
|
|
|
{
|
|
|
|
const gchar *text;
|
2011-12-24 17:41:09 +00:00
|
|
|
GMatchInfo *minfo;
|
|
|
|
gint ret = -1;
|
2010-03-02 13:21:41 +00:00
|
|
|
|
2012-07-25 13:13:49 +02:00
|
|
|
g_return_val_if_fail(pos <= (guint)sci_get_length(sci), -1);
|
2010-03-02 13:21:41 +00:00
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
/* clear old match */
|
2012-01-25 16:26:16 +00:00
|
|
|
SETPTR(regex_match_text, NULL);
|
2011-12-24 17:41:09 +00:00
|
|
|
|
2010-03-31 11:53:37 +00:00
|
|
|
/* Warning: any SCI calls will invalidate 'text' after calling SCI_GETCHARACTERPOINTER */
|
2010-03-02 13:21:41 +00:00
|
|
|
text = (void*)scintilla_send_message(sci, SCI_GETCHARACTERPOINTER, 0, 0);
|
2010-03-31 11:53:37 +00:00
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
/* Warning: minfo will become invalid when 'text' does! */
|
2011-12-25 13:42:19 +00:00
|
|
|
if (g_regex_match_full(regex, text, -1, pos, 0, &minfo, NULL))
|
2010-03-02 13:21:41 +00:00
|
|
|
{
|
2011-12-25 14:44:32 -08:00
|
|
|
guint i;
|
2011-12-24 17:41:09 +00:00
|
|
|
|
|
|
|
/* copy whole match text and offsets before they become invalid */
|
|
|
|
regex_match_text = g_match_info_fetch(minfo, 0);
|
|
|
|
|
|
|
|
foreach_range(i, G_N_ELEMENTS(regex_matches))
|
|
|
|
{
|
|
|
|
gint start = -1, end = -1;
|
|
|
|
|
2011-12-25 14:44:32 -08:00
|
|
|
g_match_info_fetch_pos(minfo, (gint)i, &start, &end);
|
2011-12-24 17:41:09 +00:00
|
|
|
regex_matches[i].start = start;
|
|
|
|
regex_matches[i].end = end;
|
|
|
|
}
|
2011-12-25 13:42:19 +00:00
|
|
|
ret = regex_matches[0].start;
|
2010-03-02 13:21:41 +00:00
|
|
|
}
|
2011-12-24 17:41:09 +00:00
|
|
|
g_match_info_free(minfo);
|
|
|
|
return ret;
|
2010-03-02 13:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
gint search_find_next(ScintillaObject *sci, const gchar *str, gint flags)
|
|
|
|
{
|
2011-12-24 17:41:09 +00:00
|
|
|
GRegex *regex;
|
2010-03-02 13:21:41 +00:00
|
|
|
gint ret = -1;
|
|
|
|
gint pos;
|
|
|
|
|
|
|
|
if (~flags & SCFIND_REGEXP)
|
|
|
|
return sci_search_next(sci, flags, str);
|
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
regex = compile_regex(str, flags);
|
|
|
|
if (!regex)
|
2010-03-02 13:21:41 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
pos = sci_get_current_position(sci);
|
2011-12-24 17:41:09 +00:00
|
|
|
ret = find_regex(sci, pos, regex);
|
2012-07-28 00:15:24 +02:00
|
|
|
/* avoid re-matching the same position in case of empty matches */
|
|
|
|
if (ret == pos && regex_matches[0].start == regex_matches[0].end)
|
|
|
|
ret = find_regex(sci, pos + 1, regex);
|
2010-03-02 13:21:41 +00:00
|
|
|
if (ret >= 0)
|
2011-12-25 13:42:19 +00:00
|
|
|
sci_set_selection(sci, ret, regex_matches[0].end);
|
2010-03-02 13:21:41 +00:00
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
g_regex_unref(regex);
|
2010-03-02 13:21:41 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
gint search_replace_target(ScintillaObject *sci, const gchar *replace_text,
|
|
|
|
gboolean regex)
|
|
|
|
{
|
|
|
|
GString *str;
|
|
|
|
gint ret = 0;
|
|
|
|
gint i = 0;
|
|
|
|
|
|
|
|
if (!regex)
|
|
|
|
return sci_replace_target(sci, replace_text, FALSE);
|
|
|
|
|
|
|
|
str = g_string_new(replace_text);
|
|
|
|
while (str->str[i])
|
|
|
|
{
|
|
|
|
gchar *ptr = &str->str[i];
|
|
|
|
gchar *grp;
|
|
|
|
gchar c;
|
|
|
|
|
|
|
|
if (ptr[0] != '\\')
|
|
|
|
{
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
c = ptr[1];
|
|
|
|
/* backslash or unnecessary escape */
|
|
|
|
if (c == '\\' || !isdigit(c))
|
|
|
|
{
|
|
|
|
g_string_erase(str, i, 1);
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* digit escape */
|
|
|
|
g_string_erase(str, i, 2);
|
|
|
|
/* fix match offsets by subtracting index of whole match start from the string */
|
2011-12-24 17:41:09 +00:00
|
|
|
grp = get_regex_match_string(regex_match_text - regex_matches[0].start,
|
|
|
|
®ex_matches[c - '0']);
|
2010-03-02 13:21:41 +00:00
|
|
|
g_string_insert(str, i, grp);
|
|
|
|
i += strlen(grp);
|
|
|
|
g_free(grp);
|
|
|
|
}
|
|
|
|
ret = sci_replace_target(sci, str->str, FALSE);
|
|
|
|
g_string_free(str, TRUE);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-05 12:50:59 +00:00
|
|
|
gint search_find_text(ScintillaObject *sci, gint flags, struct Sci_TextToFind *ttf)
|
2010-03-08 12:27:31 +00:00
|
|
|
{
|
2011-12-24 17:41:09 +00:00
|
|
|
GRegex *regex;
|
2010-03-08 12:27:31 +00:00
|
|
|
gint pos;
|
|
|
|
gint ret;
|
|
|
|
|
|
|
|
if (~flags & SCFIND_REGEXP)
|
|
|
|
return sci_find_text(sci, flags, ttf);
|
|
|
|
|
2011-12-24 17:41:09 +00:00
|
|
|
regex = compile_regex(ttf->lpstrText, flags);
|
|
|
|
if (!regex)
|
2010-03-08 12:27:31 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
pos = ttf->chrg.cpMin;
|
2011-12-24 17:41:09 +00:00
|
|
|
ret = find_regex(sci, pos, regex);
|
2010-05-05 12:50:59 +00:00
|
|
|
|
2012-12-10 22:37:20 +01:00
|
|
|
if (ret >= ttf->chrg.cpMax)
|
|
|
|
ret = -1;
|
|
|
|
else if (ret >= 0)
|
2010-03-08 12:27:31 +00:00
|
|
|
{
|
2011-12-25 13:42:19 +00:00
|
|
|
ttf->chrgText.cpMin = regex_matches[0].start;
|
|
|
|
ttf->chrgText.cpMax = regex_matches[0].end;
|
2010-03-08 12:27:31 +00:00
|
|
|
}
|
2011-12-24 17:41:09 +00:00
|
|
|
g_regex_unref(regex);
|
|
|
|
return ret;
|
2010-03-08 12:27:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
static gint find_document_usage(GeanyDocument *doc, const gchar *search_text, gint flags)
|
2006-11-01 15:26:41 +00:00
|
|
|
{
|
2007-02-28 12:08:40 +00:00
|
|
|
gchar *buffer, *short_file_name;
|
2009-07-04 09:20:28 +00:00
|
|
|
struct Sci_TextToFind ttf;
|
2007-06-28 16:00:29 +00:00
|
|
|
gint count = 0;
|
2009-03-16 13:43:12 +00:00
|
|
|
gint prev_line = -1;
|
2006-11-01 15:26:41 +00:00
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
g_return_val_if_fail(doc != NULL, 0);
|
2006-11-01 15:26:41 +00:00
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
short_file_name = g_path_get_basename(DOC_FILENAME(doc));
|
2007-06-28 16:00:29 +00:00
|
|
|
|
2006-11-01 15:26:41 +00:00
|
|
|
ttf.chrg.cpMin = 0;
|
2008-07-14 11:13:54 +00:00
|
|
|
ttf.chrg.cpMax = sci_get_length(doc->editor->sci);
|
2006-11-01 15:26:41 +00:00
|
|
|
ttf.lpstrText = (gchar *)search_text;
|
|
|
|
while (1)
|
|
|
|
{
|
2012-07-28 00:15:24 +02:00
|
|
|
gint pos, line;
|
2007-06-28 16:00:29 +00:00
|
|
|
|
2010-03-08 16:27:51 +00:00
|
|
|
pos = search_find_text(doc->editor->sci, flags, &ttf);
|
2007-06-28 16:00:29 +00:00
|
|
|
if (pos == -1)
|
2008-02-27 13:17:29 +00:00
|
|
|
break; /* no more matches */
|
2006-11-01 15:26:41 +00:00
|
|
|
|
Fix search and replacement of empty matches (again)
f4eb89cd7d79738a9c6c45e29abdd9d15d22e4fd was partially wrong and
removed legitimate re-matches when two different matches ends at the
same position. Particularly, the replacement changes are reverted.
Interestingly, Perl and Python does not agree on how to do such
replacements. Python does what I did in the above-cited commit, e.g.
doesn't replace twice if the match end overlaps, but Perl does.
Perl looks more legitimate here since both Python and Perl does find
the overlapping matches when performing a search, so Python is the odd
guy here doing it differently on replace than it does upon search.
For example, replacing using the pattern "a?(?=b)" and the replacement
string "_":
Python: ababcdb -> _b_bcd_b
Perl: ababcdb -> __b__bcd_b
But finding using the same pattern on the same input gives the same
results on both:
Python: ababcdb -> ['a', '', 'a', '', '']
Perl: ababcdb -> ['a', '', 'a', '', '']
Anyway, GLib and us claim to support "Perl-compatible regular
expressions", so we gotta follow Perl, especially in such doubtful
situations.
2012-07-28 19:23:37 +02:00
|
|
|
count++;
|
|
|
|
line = sci_get_line_from_position(doc->editor->sci, pos);
|
|
|
|
if (line != prev_line)
|
2009-03-13 17:23:56 +00:00
|
|
|
{
|
Fix search and replacement of empty matches (again)
f4eb89cd7d79738a9c6c45e29abdd9d15d22e4fd was partially wrong and
removed legitimate re-matches when two different matches ends at the
same position. Particularly, the replacement changes are reverted.
Interestingly, Perl and Python does not agree on how to do such
replacements. Python does what I did in the above-cited commit, e.g.
doesn't replace twice if the match end overlaps, but Perl does.
Perl looks more legitimate here since both Python and Perl does find
the overlapping matches when performing a search, so Python is the odd
guy here doing it differently on replace than it does upon search.
For example, replacing using the pattern "a?(?=b)" and the replacement
string "_":
Python: ababcdb -> _b_bcd_b
Perl: ababcdb -> __b__bcd_b
But finding using the same pattern on the same input gives the same
results on both:
Python: ababcdb -> ['a', '', 'a', '', '']
Perl: ababcdb -> ['a', '', 'a', '', '']
Anyway, GLib and us claim to support "Perl-compatible regular
expressions", so we gotta follow Perl, especially in such doubtful
situations.
2012-07-28 19:23:37 +02:00
|
|
|
buffer = sci_get_line(doc->editor->sci, line);
|
|
|
|
msgwin_msg_add(COLOR_BLACK, line + 1, doc,
|
|
|
|
"%s:%d: %s", short_file_name, line + 1, g_strstrip(buffer));
|
|
|
|
g_free(buffer);
|
|
|
|
prev_line = line;
|
2009-03-13 17:23:56 +00:00
|
|
|
}
|
2007-06-28 16:00:29 +00:00
|
|
|
|
Fix search and replacement of empty matches (again)
f4eb89cd7d79738a9c6c45e29abdd9d15d22e4fd was partially wrong and
removed legitimate re-matches when two different matches ends at the
same position. Particularly, the replacement changes are reverted.
Interestingly, Perl and Python does not agree on how to do such
replacements. Python does what I did in the above-cited commit, e.g.
doesn't replace twice if the match end overlaps, but Perl does.
Perl looks more legitimate here since both Python and Perl does find
the overlapping matches when performing a search, so Python is the odd
guy here doing it differently on replace than it does upon search.
For example, replacing using the pattern "a?(?=b)" and the replacement
string "_":
Python: ababcdb -> _b_bcd_b
Perl: ababcdb -> __b__bcd_b
But finding using the same pattern on the same input gives the same
results on both:
Python: ababcdb -> ['a', '', 'a', '', '']
Perl: ababcdb -> ['a', '', 'a', '', '']
Anyway, GLib and us claim to support "Perl-compatible regular
expressions", so we gotta follow Perl, especially in such doubtful
situations.
2012-07-28 19:23:37 +02:00
|
|
|
ttf.chrg.cpMin = ttf.chrgText.cpMax;
|
|
|
|
/* avoid rematching with empty matches like "(?=[a-z])" or "^$".
|
|
|
|
* note we cannot assume a match will always be empty or not and then break out, since
|
|
|
|
* matches like "a?(?=b)" will me sometimes empty and sometimes not */
|
|
|
|
if ((ttf.chrgText.cpMax - ttf.chrgText.cpMin) == 0)
|
2012-07-28 00:15:24 +02:00
|
|
|
ttf.chrg.cpMin ++;
|
2006-11-01 15:26:41 +00:00
|
|
|
}
|
2007-06-28 16:00:29 +00:00
|
|
|
g_free(short_file_name);
|
2006-11-01 15:26:41 +00:00
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-03 13:40:42 +00:00
|
|
|
void search_find_usage(const gchar *search_text, const gchar *original_search_text,
|
|
|
|
gint flags, gboolean in_session)
|
2006-11-01 15:26:41 +00:00
|
|
|
{
|
2008-06-15 13:35:48 +00:00
|
|
|
GeanyDocument *doc;
|
2009-03-16 13:43:12 +00:00
|
|
|
gint count = 0;
|
2006-11-01 15:26:41 +00:00
|
|
|
|
2008-06-15 13:35:48 +00:00
|
|
|
doc = document_get_current();
|
|
|
|
g_return_if_fail(doc != NULL);
|
2006-11-01 15:26:41 +00:00
|
|
|
|
2011-03-24 22:00:18 +00:00
|
|
|
if (G_UNLIKELY(! NZV(search_text)))
|
2008-12-19 17:15:52 +00:00
|
|
|
{
|
|
|
|
utils_beep();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-11-01 15:26:41 +00:00
|
|
|
gtk_notebook_set_current_page(GTK_NOTEBOOK(msgwindow.notebook), MSG_MESSAGE);
|
|
|
|
gtk_list_store_clear(msgwindow.store_msg);
|
|
|
|
|
|
|
|
if (! in_session)
|
2008-02-27 13:17:29 +00:00
|
|
|
{ /* use current document */
|
2009-03-16 13:43:12 +00:00
|
|
|
count = find_document_usage(doc, search_text, flags);
|
2006-11-01 15:26:41 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
guint i;
|
2008-05-29 17:00:54 +00:00
|
|
|
for (i = 0; i < documents_array->len; i++)
|
2006-11-01 15:26:41 +00:00
|
|
|
{
|
2008-06-18 14:18:26 +00:00
|
|
|
if (documents[i]->is_valid)
|
2009-03-16 13:43:12 +00:00
|
|
|
{
|
|
|
|
count += find_document_usage(documents[i], search_text, flags);
|
|
|
|
}
|
2006-11-01 15:26:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-16 13:43:12 +00:00
|
|
|
if (count == 0) /* no matches were found */
|
2006-11-01 15:26:41 +00:00
|
|
|
{
|
2011-06-03 13:40:42 +00:00
|
|
|
ui_set_statusbar(FALSE, _("No matches found for \"%s\"."), original_search_text);
|
|
|
|
msgwin_msg_add(COLOR_BLUE, -1, NULL, _("No matches found for \"%s\"."), original_search_text);
|
2007-02-28 12:08:40 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-04-22 14:44:45 +00:00
|
|
|
ui_set_statusbar(FALSE, ngettext(
|
2008-10-12 13:07:56 +00:00
|
|
|
"Found %d match for \"%s\".", "Found %d matches for \"%s\".", count),
|
2011-06-03 13:40:42 +00:00
|
|
|
count, original_search_text);
|
2008-12-05 17:30:06 +00:00
|
|
|
msgwin_msg_add(COLOR_BLUE, -1, NULL, ngettext(
|
2008-10-12 13:07:56 +00:00
|
|
|
"Found %d match for \"%s\".", "Found %d matches for \"%s\".", count),
|
2011-06-03 13:40:42 +00:00
|
|
|
count, original_search_text);
|
2006-11-01 15:26:41 +00:00
|
|
|
}
|
|
|
|
}
|
2007-02-17 17:39:10 +00:00
|
|
|
|
|
|
|
|
2010-03-25 13:43:37 +00:00
|
|
|
/* ttf is updated to include the last match position (ttf->chrg.cpMin) and
|
|
|
|
* the new search range end (ttf->chrg.cpMax).
|
2010-03-08 13:28:04 +00:00
|
|
|
* Note: Normally you would call sci_start/end_undo_action() around this call. */
|
2010-03-08 13:45:14 +00:00
|
|
|
/* Warning: Scintilla recommends caching replacements to do all at once to avoid
|
|
|
|
* performance issues with SCI_GETCHARACTERPOINTER. */
|
2010-03-08 13:28:04 +00:00
|
|
|
guint search_replace_range(ScintillaObject *sci, struct Sci_TextToFind *ttf,
|
|
|
|
gint flags, const gchar *replace_text)
|
|
|
|
{
|
|
|
|
gint count = 0;
|
|
|
|
const gchar *find_text = ttf->lpstrText;
|
|
|
|
gint start = ttf->chrg.cpMin;
|
|
|
|
gint end = ttf->chrg.cpMax;
|
|
|
|
|
|
|
|
g_return_val_if_fail(sci != NULL && find_text != NULL && replace_text != NULL, 0);
|
|
|
|
if (! *find_text)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
{
|
|
|
|
gint search_pos;
|
|
|
|
gint find_len = 0, replace_len = 0;
|
|
|
|
|
2010-03-08 16:27:51 +00:00
|
|
|
search_pos = search_find_text(sci, flags, ttf);
|
2010-03-08 13:28:04 +00:00
|
|
|
find_len = ttf->chrgText.cpMax - ttf->chrgText.cpMin;
|
|
|
|
if (search_pos == -1)
|
|
|
|
break; /* no more matches */
|
|
|
|
|
|
|
|
if (search_pos + find_len > end)
|
|
|
|
break; /* found text is partly out of range */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint movepastEOL = 0;
|
|
|
|
|
|
|
|
sci_set_target_start(sci, search_pos);
|
|
|
|
sci_set_target_end(sci, search_pos + find_len);
|
|
|
|
|
|
|
|
if (find_len <= 0)
|
|
|
|
{
|
|
|
|
gchar chNext = sci_get_char_at(sci, sci_get_target_end(sci));
|
|
|
|
|
|
|
|
if (chNext == '\r' || chNext == '\n')
|
|
|
|
movepastEOL = 1;
|
|
|
|
}
|
Fix search and replacement of empty matches (again)
f4eb89cd7d79738a9c6c45e29abdd9d15d22e4fd was partially wrong and
removed legitimate re-matches when two different matches ends at the
same position. Particularly, the replacement changes are reverted.
Interestingly, Perl and Python does not agree on how to do such
replacements. Python does what I did in the above-cited commit, e.g.
doesn't replace twice if the match end overlaps, but Perl does.
Perl looks more legitimate here since both Python and Perl does find
the overlapping matches when performing a search, so Python is the odd
guy here doing it differently on replace than it does upon search.
For example, replacing using the pattern "a?(?=b)" and the replacement
string "_":
Python: ababcdb -> _b_bcd_b
Perl: ababcdb -> __b__bcd_b
But finding using the same pattern on the same input gives the same
results on both:
Python: ababcdb -> ['a', '', 'a', '', '']
Perl: ababcdb -> ['a', '', 'a', '', '']
Anyway, GLib and us claim to support "Perl-compatible regular
expressions", so we gotta follow Perl, especially in such doubtful
situations.
2012-07-28 19:23:37 +02:00
|
|
|
replace_len = search_replace_target(sci, replace_text,
|
|
|
|
flags & SCFIND_REGEXP);
|
|
|
|
count++;
|
2010-03-08 13:28:04 +00:00
|
|
|
if (search_pos == end)
|
|
|
|
break; /* Prevent hang when replacing regex $ */
|
|
|
|
|
|
|
|
/* make the next search start after the replaced text */
|
|
|
|
start = search_pos + replace_len + movepastEOL;
|
|
|
|
if (find_len == 0)
|
|
|
|
start = sci_get_position_after(sci, start); /* prevent '[ ]*' regex rematching part of replaced text */
|
|
|
|
ttf->chrg.cpMin = start;
|
|
|
|
end += replace_len - find_len; /* update end of range now text has changed */
|
|
|
|
ttf->chrg.cpMax = end;
|
|
|
|
}
|
2012-07-28 00:15:24 +02:00
|
|
|
|
2010-03-08 13:28:04 +00:00
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-21 16:16:15 +00:00
|
|
|
void search_find_again(gboolean change_direction)
|
|
|
|
{
|
|
|
|
GeanyDocument *doc = document_get_current();
|
|
|
|
|
|
|
|
g_return_if_fail(doc != NULL);
|
|
|
|
|
|
|
|
if (search_data.text)
|
|
|
|
{
|
|
|
|
gboolean forward = ! search_data.backwards;
|
2011-06-03 13:40:42 +00:00
|
|
|
gint result = document_find_text(doc, search_data.text, search_data.original_text, search_data.flags,
|
2010-09-21 16:16:15 +00:00
|
|
|
change_direction ? forward : !forward, FALSE, NULL);
|
|
|
|
|
|
|
|
if (result > -1)
|
|
|
|
editor_display_current_line(doc->editor, 0.3F);
|
|
|
|
|
|
|
|
if (search_data.search_bar)
|
|
|
|
ui_set_search_entry_background(
|
|
|
|
toolbar_get_widget_child_by_name("SearchEntry"), (result > -1));
|
|
|
|
}
|
|
|
|
}
|