Blame tests/test-search.c

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