486 lines
14 KiB
C
486 lines
14 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*-
|
|
* test-search.c
|
|
* This file is part of GtkSourceView
|
|
*
|
|
* Copyright (C) 2013 - Sébastien Wilmet <swilmet@gnome.org>
|
|
*
|
|
* GtkSourceView is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* GtkSourceView 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <gtk/gtk.h>
|
|
#include <gtksourceview/gtksource.h>
|
|
|
|
#define TEST_TYPE_SEARCH (test_search_get_type ())
|
|
#define TEST_SEARCH(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), TEST_TYPE_SEARCH, TestSearch))
|
|
#define TEST_SEARCH_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), TEST_TYPE_SEARCH, TestSearchClass))
|
|
#define TEST_IS_SEARCH(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), TEST_TYPE_SEARCH))
|
|
#define TEST_IS_SEARCH_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), TEST_TYPE_SEARCH))
|
|
#define TEST_SEARCH_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), TEST_TYPE_SEARCH, TestSearchClass))
|
|
|
|
typedef struct _TestSearch TestSearch;
|
|
typedef struct _TestSearchClass TestSearchClass;
|
|
typedef struct _TestSearchPrivate TestSearchPrivate;
|
|
|
|
struct _TestSearch
|
|
{
|
|
GtkGrid parent;
|
|
TestSearchPrivate *priv;
|
|
};
|
|
|
|
struct _TestSearchClass
|
|
{
|
|
GtkGridClass parent_class;
|
|
};
|
|
|
|
struct _TestSearchPrivate
|
|
{
|
|
GtkSourceView *source_view;
|
|
GtkSourceBuffer *source_buffer;
|
|
GtkSourceSearchContext *search_context;
|
|
GtkSourceSearchSettings *search_settings;
|
|
GtkEntry *replace_entry;
|
|
GtkLabel *label_occurrences;
|
|
GtkLabel *label_regex_error;
|
|
|
|
guint idle_update_label_id;
|
|
};
|
|
|
|
GType test_search_get_type (void);
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (TestSearch, test_search, GTK_TYPE_GRID)
|
|
|
|
static void
|
|
open_file (TestSearch *search,
|
|
const gchar *filename)
|
|
{
|
|
gchar *contents;
|
|
GError *error = NULL;
|
|
GtkSourceLanguageManager *language_manager;
|
|
GtkSourceLanguage *language;
|
|
GtkTextIter iter;
|
|
|
|
/* In a realistic application you would use GtkSourceFile of course. */
|
|
if (!g_file_get_contents (filename, &contents, NULL, &error))
|
|
{
|
|
g_error ("Impossible to load file: %s", error->message);
|
|
}
|
|
|
|
gtk_text_buffer_set_text (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
contents,
|
|
-1);
|
|
|
|
language_manager = gtk_source_language_manager_get_default ();
|
|
language = gtk_source_language_manager_get_language (language_manager, "c");
|
|
gtk_source_buffer_set_language (search->priv->source_buffer, language);
|
|
|
|
gtk_text_buffer_get_start_iter (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
&iter);
|
|
|
|
gtk_text_buffer_select_range (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
&iter,
|
|
&iter);
|
|
|
|
g_free (contents);
|
|
}
|
|
|
|
static void
|
|
update_label_occurrences (TestSearch *search)
|
|
{
|
|
gint occurrences_count;
|
|
GtkTextIter select_start;
|
|
GtkTextIter select_end;
|
|
gint occurrence_pos;
|
|
gchar *text;
|
|
|
|
occurrences_count = gtk_source_search_context_get_occurrences_count (search->priv->search_context);
|
|
|
|
gtk_text_buffer_get_selection_bounds (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
&select_start,
|
|
&select_end);
|
|
|
|
occurrence_pos = gtk_source_search_context_get_occurrence_position (search->priv->search_context,
|
|
&select_start,
|
|
&select_end);
|
|
|
|
if (occurrences_count == -1)
|
|
{
|
|
text = g_strdup ("");
|
|
}
|
|
else if (occurrence_pos == -1)
|
|
{
|
|
text = g_strdup_printf ("%u occurrences", occurrences_count);
|
|
}
|
|
else
|
|
{
|
|
text = g_strdup_printf ("%d of %u", occurrence_pos, occurrences_count);
|
|
}
|
|
|
|
gtk_label_set_text (search->priv->label_occurrences, text);
|
|
g_free (text);
|
|
}
|
|
|
|
static void
|
|
update_label_regex_error (TestSearch *search)
|
|
{
|
|
GError *error;
|
|
|
|
error = gtk_source_search_context_get_regex_error (search->priv->search_context);
|
|
|
|
if (error == NULL)
|
|
{
|
|
gtk_label_set_text (search->priv->label_regex_error, "");
|
|
gtk_widget_hide (GTK_WIDGET (search->priv->label_regex_error));
|
|
}
|
|
else
|
|
{
|
|
gtk_label_set_text (search->priv->label_regex_error, error->message);
|
|
gtk_widget_show (GTK_WIDGET (search->priv->label_regex_error));
|
|
g_clear_error (&error);
|
|
}
|
|
}
|
|
|
|
static void
|
|
search_entry_changed_cb (TestSearch *search,
|
|
GtkEntry *entry)
|
|
{
|
|
const gchar *text = gtk_entry_get_text (entry);
|
|
gchar *unescaped_text = gtk_source_utils_unescape_search_text (text);
|
|
|
|
gtk_source_search_settings_set_search_text (search->priv->search_settings, unescaped_text);
|
|
g_free (unescaped_text);
|
|
}
|
|
|
|
static void
|
|
select_search_occurrence (TestSearch *search,
|
|
const GtkTextIter *match_start,
|
|
const GtkTextIter *match_end)
|
|
{
|
|
GtkTextMark *insert;
|
|
|
|
gtk_text_buffer_select_range (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
match_start,
|
|
match_end);
|
|
|
|
insert = gtk_text_buffer_get_insert (GTK_TEXT_BUFFER (search->priv->source_buffer));
|
|
|
|
gtk_text_view_scroll_mark_onscreen (GTK_TEXT_VIEW (search->priv->source_view),
|
|
insert);
|
|
}
|
|
|
|
static void
|
|
backward_search_finished (GtkSourceSearchContext *search_context,
|
|
GAsyncResult *result,
|
|
TestSearch *search)
|
|
{
|
|
GtkTextIter match_start;
|
|
GtkTextIter match_end;
|
|
|
|
if (gtk_source_search_context_backward_finish2 (search_context,
|
|
result,
|
|
&match_start,
|
|
&match_end,
|
|
NULL,
|
|
NULL))
|
|
{
|
|
select_search_occurrence (search, &match_start, &match_end);
|
|
}
|
|
}
|
|
|
|
static void
|
|
button_previous_clicked_cb (TestSearch *search,
|
|
GtkButton *button)
|
|
{
|
|
GtkTextIter start_at;
|
|
|
|
gtk_text_buffer_get_selection_bounds (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
&start_at,
|
|
NULL);
|
|
|
|
gtk_source_search_context_backward_async (search->priv->search_context,
|
|
&start_at,
|
|
NULL,
|
|
(GAsyncReadyCallback)backward_search_finished,
|
|
search);
|
|
}
|
|
|
|
static void
|
|
forward_search_finished (GtkSourceSearchContext *search_context,
|
|
GAsyncResult *result,
|
|
TestSearch *search)
|
|
{
|
|
GtkTextIter match_start;
|
|
GtkTextIter match_end;
|
|
|
|
if (gtk_source_search_context_forward_finish2 (search_context,
|
|
result,
|
|
&match_start,
|
|
&match_end,
|
|
NULL,
|
|
NULL))
|
|
{
|
|
select_search_occurrence (search, &match_start, &match_end);
|
|
}
|
|
}
|
|
|
|
static void
|
|
button_next_clicked_cb (TestSearch *search,
|
|
GtkButton *button)
|
|
{
|
|
GtkTextIter start_at;
|
|
|
|
gtk_text_buffer_get_selection_bounds (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
NULL,
|
|
&start_at);
|
|
|
|
gtk_source_search_context_forward_async (search->priv->search_context,
|
|
&start_at,
|
|
NULL,
|
|
(GAsyncReadyCallback)forward_search_finished,
|
|
search);
|
|
}
|
|
|
|
static void
|
|
button_replace_clicked_cb (TestSearch *search,
|
|
GtkButton *button)
|
|
{
|
|
GtkTextIter match_start;
|
|
GtkTextIter match_end;
|
|
GtkTextIter iter;
|
|
GtkEntryBuffer *entry_buffer;
|
|
gint replace_length;
|
|
|
|
gtk_text_buffer_get_selection_bounds (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
&match_start,
|
|
&match_end);
|
|
|
|
entry_buffer = gtk_entry_get_buffer (search->priv->replace_entry);
|
|
replace_length = gtk_entry_buffer_get_bytes (entry_buffer);
|
|
|
|
gtk_source_search_context_replace2 (search->priv->search_context,
|
|
&match_start,
|
|
&match_end,
|
|
gtk_entry_get_text (search->priv->replace_entry),
|
|
replace_length,
|
|
NULL);
|
|
|
|
gtk_text_buffer_get_selection_bounds (GTK_TEXT_BUFFER (search->priv->source_buffer),
|
|
NULL,
|
|
&iter);
|
|
|
|
gtk_source_search_context_forward_async (search->priv->search_context,
|
|
&iter,
|
|
NULL,
|
|
(GAsyncReadyCallback)forward_search_finished,
|
|
search);
|
|
}
|
|
|
|
static void
|
|
button_replace_all_clicked_cb (TestSearch *search,
|
|
GtkButton *button)
|
|
{
|
|
GtkEntryBuffer *entry_buffer = gtk_entry_get_buffer (search->priv->replace_entry);
|
|
gint replace_length = gtk_entry_buffer_get_bytes (entry_buffer);
|
|
|
|
gtk_source_search_context_replace_all (search->priv->search_context,
|
|
gtk_entry_get_text (search->priv->replace_entry),
|
|
replace_length,
|
|
NULL);
|
|
}
|
|
|
|
static gboolean
|
|
update_label_idle_cb (TestSearch *search)
|
|
{
|
|
search->priv->idle_update_label_id = 0;
|
|
|
|
update_label_occurrences (search);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
static void
|
|
mark_set_cb (GtkTextBuffer *buffer,
|
|
GtkTextIter *location,
|
|
GtkTextMark *mark,
|
|
TestSearch *search)
|
|
{
|
|
GtkTextMark *insert;
|
|
GtkTextMark *selection_bound;
|
|
|
|
insert = gtk_text_buffer_get_insert (buffer);
|
|
selection_bound = gtk_text_buffer_get_selection_bound (buffer);
|
|
|
|
if ((mark == insert || mark == selection_bound) &&
|
|
search->priv->idle_update_label_id == 0)
|
|
{
|
|
search->priv->idle_update_label_id = g_idle_add ((GSourceFunc)update_label_idle_cb,
|
|
search);
|
|
}
|
|
}
|
|
|
|
static void
|
|
highlight_toggled_cb (TestSearch *search,
|
|
GtkToggleButton *button)
|
|
{
|
|
gtk_source_search_context_set_highlight (search->priv->search_context,
|
|
gtk_toggle_button_get_active (button));
|
|
}
|
|
|
|
static void
|
|
match_case_toggled_cb (TestSearch *search,
|
|
GtkToggleButton *button)
|
|
{
|
|
gtk_source_search_settings_set_case_sensitive (search->priv->search_settings,
|
|
gtk_toggle_button_get_active (button));
|
|
}
|
|
|
|
static void
|
|
at_word_boundaries_toggled_cb (TestSearch *search,
|
|
GtkToggleButton *button)
|
|
{
|
|
gtk_source_search_settings_set_at_word_boundaries (search->priv->search_settings,
|
|
gtk_toggle_button_get_active (button));
|
|
}
|
|
|
|
static void
|
|
wrap_around_toggled_cb (TestSearch *search,
|
|
GtkToggleButton *button)
|
|
{
|
|
gtk_source_search_settings_set_wrap_around (search->priv->search_settings,
|
|
gtk_toggle_button_get_active (button));
|
|
}
|
|
|
|
static void
|
|
regex_toggled_cb (TestSearch *search,
|
|
GtkToggleButton *button)
|
|
{
|
|
gtk_source_search_settings_set_regex_enabled (search->priv->search_settings,
|
|
gtk_toggle_button_get_active (button));
|
|
}
|
|
|
|
static void
|
|
test_search_dispose (GObject *object)
|
|
{
|
|
TestSearch *search = TEST_SEARCH (object);
|
|
|
|
g_clear_object (&search->priv->source_buffer);
|
|
g_clear_object (&search->priv->search_context);
|
|
g_clear_object (&search->priv->search_settings);
|
|
|
|
G_OBJECT_CLASS (test_search_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
test_search_class_init (TestSearchClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
object_class->dispose = test_search_dispose;
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class,
|
|
"/org/gnome/gtksourceview/tests/ui/test-search.ui");
|
|
|
|
gtk_widget_class_bind_template_child_private (widget_class, TestSearch, source_view);
|
|
gtk_widget_class_bind_template_child_private (widget_class, TestSearch, replace_entry);
|
|
gtk_widget_class_bind_template_child_private (widget_class, TestSearch, label_occurrences);
|
|
gtk_widget_class_bind_template_child_private (widget_class, TestSearch, label_regex_error);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, search_entry_changed_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, button_previous_clicked_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, button_next_clicked_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, button_replace_clicked_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, button_replace_all_clicked_cb);
|
|
|
|
/* It is also possible to bind the properties with
|
|
* g_object_bind_property(), between the check buttons and the source
|
|
* buffer. But GtkBuilder and Glade don't support that yet.
|
|
*/
|
|
gtk_widget_class_bind_template_callback (widget_class, highlight_toggled_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, match_case_toggled_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, at_word_boundaries_toggled_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, wrap_around_toggled_cb);
|
|
gtk_widget_class_bind_template_callback (widget_class, regex_toggled_cb);
|
|
}
|
|
|
|
static void
|
|
test_search_init (TestSearch *search)
|
|
{
|
|
search->priv = test_search_get_instance_private (search);
|
|
|
|
gtk_widget_init_template (GTK_WIDGET (search));
|
|
|
|
search->priv->source_buffer = GTK_SOURCE_BUFFER (
|
|
gtk_text_view_get_buffer (GTK_TEXT_VIEW (search->priv->source_view)));
|
|
|
|
g_object_ref (search->priv->source_buffer);
|
|
|
|
open_file (search, TOP_SRCDIR "/gtksourceview/gtksourcesearchcontext.c");
|
|
|
|
search->priv->search_settings = gtk_source_search_settings_new ();
|
|
|
|
search->priv->search_context = gtk_source_search_context_new (search->priv->source_buffer,
|
|
search->priv->search_settings);
|
|
|
|
g_signal_connect_swapped (search->priv->search_context,
|
|
"notify::occurrences-count",
|
|
G_CALLBACK (update_label_occurrences),
|
|
search);
|
|
|
|
g_signal_connect (search->priv->source_buffer,
|
|
"mark-set",
|
|
G_CALLBACK (mark_set_cb),
|
|
search);
|
|
|
|
g_signal_connect_swapped (search->priv->search_context,
|
|
"notify::regex-error",
|
|
G_CALLBACK (update_label_regex_error),
|
|
search);
|
|
|
|
update_label_regex_error (search);
|
|
}
|
|
|
|
static TestSearch *
|
|
test_search_new (void)
|
|
{
|
|
return g_object_new (test_search_get_type (), NULL);
|
|
}
|
|
|
|
gint
|
|
main (gint argc, gchar *argv[])
|
|
{
|
|
GtkWidget *window;
|
|
TestSearch *search;
|
|
|
|
gtk_init (&argc, &argv);
|
|
|
|
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
|
|
gtk_window_set_default_size (GTK_WINDOW (window), 700, 500);
|
|
|
|
g_signal_connect (window,
|
|
"destroy",
|
|
G_CALLBACK (gtk_main_quit),
|
|
NULL);
|
|
|
|
search = test_search_new ();
|
|
gtk_container_add (GTK_CONTAINER (window), GTK_WIDGET (search));
|
|
|
|
gtk_widget_show (window);
|
|
|
|
gtk_main ();
|
|
|
|
return 0;
|
|
}
|