Blame gtksourceview/completion-providers/words/gtksourcecompletionwords.c

Packit a7d494
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*-
Packit a7d494
 * gtksourcecompletionproviderwords.c
Packit a7d494
 * This file is part of GtkSourceView
Packit a7d494
 *
Packit a7d494
 * Copyright (C) 2009 - Jesse van den Kieboom
Packit a7d494
 * Copyright (C) 2013 - Sébastien Wilmet
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
/**
Packit a7d494
 * SECTION:completionwords
Packit a7d494
 * @title: GtkSourceCompletionWords
Packit a7d494
 * @short_description: A GtkSourceCompletionProvider for the completion of words
Packit a7d494
 *
Packit a7d494
 * The #GtkSourceCompletionWords is an example of an implementation of
Packit a7d494
 * the #GtkSourceCompletionProvider interface. The proposals are words
Packit a7d494
 * appearing in the registered #GtkTextBuffers.
Packit a7d494
 */
Packit a7d494
Packit a7d494
#ifdef HAVE_CONFIG_H
Packit a7d494
#include <config.h>
Packit a7d494
#endif
Packit a7d494
Packit a7d494
#include "gtksourcecompletionwords.h"
Packit a7d494
#include "gtksourcecompletionwordslibrary.h"
Packit a7d494
#include "gtksourcecompletionwordsbuffer.h"
Packit a7d494
#include "gtksourcecompletionwordsutils.h"
Packit a7d494
#include "gtksourceview/gtksource.h"
Packit a7d494
#include "gtksourceview/gtksourceview-enumtypes.h"
Packit a7d494
#include "gtksourceview/gtksourceview-i18n.h"
Packit a7d494
Packit a7d494
#include <string.h>
Packit a7d494
Packit a7d494
#define BUFFER_KEY "GtkSourceCompletionWordsBufferKey"
Packit a7d494
Packit a7d494
enum
Packit a7d494
{
Packit a7d494
	PROP_0,
Packit a7d494
	PROP_NAME,
Packit a7d494
	PROP_ICON,
Packit a7d494
	PROP_PROPOSALS_BATCH_SIZE,
Packit a7d494
	PROP_SCAN_BATCH_SIZE,
Packit a7d494
	PROP_MINIMUM_WORD_SIZE,
Packit a7d494
	PROP_INTERACTIVE_DELAY,
Packit a7d494
	PROP_PRIORITY,
Packit a7d494
	PROP_ACTIVATION,
Packit a7d494
	N_PROPERTIES
Packit a7d494
};
Packit a7d494
Packit a7d494
struct _GtkSourceCompletionWordsPrivate
Packit a7d494
{
Packit a7d494
	gchar *name;
Packit a7d494
	GdkPixbuf *icon;
Packit a7d494
Packit a7d494
	gchar *word;
Packit a7d494
	gint word_len;
Packit a7d494
	guint idle_id;
Packit a7d494
Packit a7d494
	GtkSourceCompletionContext *context;
Packit a7d494
	GSequenceIter *populate_iter;
Packit a7d494
Packit a7d494
	guint cancel_id;
Packit a7d494
Packit a7d494
	guint proposals_batch_size;
Packit a7d494
	guint scan_batch_size;
Packit a7d494
	guint minimum_word_size;
Packit a7d494
Packit a7d494
	GtkSourceCompletionWordsLibrary *library;
Packit a7d494
	GList *buffers;
Packit a7d494
Packit a7d494
	gint interactive_delay;
Packit a7d494
	gint priority;
Packit a7d494
	GtkSourceCompletionActivation activation;
Packit a7d494
};
Packit a7d494
Packit a7d494
typedef struct
Packit a7d494
{
Packit a7d494
	GtkSourceCompletionWords *words;
Packit a7d494
	GtkSourceCompletionWordsBuffer *buffer;
Packit a7d494
} BufferBinding;
Packit a7d494
Packit a7d494
static GParamSpec *properties[N_PROPERTIES];
Packit a7d494
Packit a7d494
static void gtk_source_completion_words_iface_init (GtkSourceCompletionProviderIface *iface);
Packit a7d494
Packit a7d494
G_DEFINE_TYPE_WITH_CODE (GtkSourceCompletionWords,
Packit a7d494
			 gtk_source_completion_words,
Packit a7d494
			 G_TYPE_OBJECT,
Packit a7d494
			 G_ADD_PRIVATE (GtkSourceCompletionWords)
Packit a7d494
			 G_IMPLEMENT_INTERFACE (GTK_SOURCE_TYPE_COMPLETION_PROVIDER,
Packit a7d494
				 		gtk_source_completion_words_iface_init))
Packit a7d494
Packit a7d494
static gchar *
Packit a7d494
gtk_source_completion_words_get_name (GtkSourceCompletionProvider *self)
Packit a7d494
{
Packit a7d494
	return g_strdup (GTK_SOURCE_COMPLETION_WORDS (self)->priv->name);
Packit a7d494
}
Packit a7d494
Packit a7d494
static GdkPixbuf *
Packit a7d494
gtk_source_completion_words_get_icon (GtkSourceCompletionProvider *self)
Packit a7d494
{
Packit a7d494
	return GTK_SOURCE_COMPLETION_WORDS (self)->priv->icon;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
population_finished (GtkSourceCompletionWords *words)
Packit a7d494
{
Packit a7d494
	if (words->priv->idle_id != 0)
Packit a7d494
	{
Packit a7d494
		g_source_remove (words->priv->idle_id);
Packit a7d494
		words->priv->idle_id = 0;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_free (words->priv->word);
Packit a7d494
	words->priv->word = NULL;
Packit a7d494
Packit a7d494
	if (words->priv->context != NULL)
Packit a7d494
	{
Packit a7d494
		if (words->priv->cancel_id)
Packit a7d494
		{
Packit a7d494
			g_signal_handler_disconnect (words->priv->context,
Packit a7d494
			                             words->priv->cancel_id);
Packit a7d494
			words->priv->cancel_id = 0;
Packit a7d494
		}
Packit a7d494
Packit a7d494
		g_clear_object (&words->priv->context);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static gboolean
Packit a7d494
add_in_idle (GtkSourceCompletionWords *words)
Packit a7d494
{
Packit a7d494
	guint idx = 0;
Packit a7d494
	GList *ret = NULL;
Packit a7d494
	gboolean finished;
Packit a7d494
Packit a7d494
	if (words->priv->populate_iter == NULL)
Packit a7d494
	{
Packit a7d494
		words->priv->populate_iter =
Packit a7d494
			gtk_source_completion_words_library_find_first (words->priv->library,
Packit a7d494
			                                                words->priv->word,
Packit a7d494
			                                                words->priv->word_len);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	while (idx < words->priv->proposals_batch_size &&
Packit a7d494
	       words->priv->populate_iter)
Packit a7d494
	{
Packit a7d494
		GtkSourceCompletionWordsProposal *proposal =
Packit a7d494
				gtk_source_completion_words_library_get_proposal (words->priv->populate_iter);
Packit a7d494
Packit a7d494
		/* Only add non-exact matches */
Packit a7d494
		if (strcmp (gtk_source_completion_words_proposal_get_word (proposal),
Packit a7d494
		            words->priv->word) != 0)
Packit a7d494
		{
Packit a7d494
			ret = g_list_prepend (ret, proposal);
Packit a7d494
		}
Packit a7d494
Packit a7d494
		words->priv->populate_iter =
Packit a7d494
				gtk_source_completion_words_library_find_next (words->priv->populate_iter,
Packit a7d494
		                                                               words->priv->word,
Packit a7d494
		                                                               words->priv->word_len);
Packit a7d494
		++idx;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	ret = g_list_reverse (ret);
Packit a7d494
	finished = words->priv->populate_iter == NULL;
Packit a7d494
Packit a7d494
	gtk_source_completion_context_add_proposals (words->priv->context,
Packit a7d494
	                                             GTK_SOURCE_COMPLETION_PROVIDER (words),
Packit a7d494
	                                             ret,
Packit a7d494
	                                             finished);
Packit a7d494
Packit a7d494
	g_list_free (ret);
Packit a7d494
Packit a7d494
	if (finished)
Packit a7d494
	{
Packit a7d494
		gtk_source_completion_words_library_unlock (words->priv->library);
Packit a7d494
		population_finished (words);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return !finished;
Packit a7d494
}
Packit a7d494
Packit a7d494
static gchar *
Packit a7d494
get_word_at_iter (GtkTextIter *iter)
Packit a7d494
{
Packit a7d494
	GtkTextBuffer *buffer;
Packit a7d494
	GtkTextIter start_line;
Packit a7d494
	gchar *line_text;
Packit a7d494
	gchar *word;
Packit a7d494
Packit a7d494
	buffer = gtk_text_iter_get_buffer (iter);
Packit a7d494
	start_line = *iter;
Packit a7d494
	gtk_text_iter_set_line_offset (&start_line, 0);
Packit a7d494
Packit a7d494
	line_text = gtk_text_buffer_get_text (buffer, &start_line, iter, FALSE);
Packit a7d494
Packit a7d494
	word = _gtk_source_completion_words_utils_get_end_word (line_text);
Packit a7d494
Packit a7d494
	g_free (line_text);
Packit a7d494
	return word;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_populate (GtkSourceCompletionProvider *provider,
Packit a7d494
                                      GtkSourceCompletionContext  *context)
Packit a7d494
{
Packit a7d494
	GtkSourceCompletionWords *words = GTK_SOURCE_COMPLETION_WORDS (provider);
Packit a7d494
	GtkSourceCompletionActivation activation;
Packit a7d494
	GtkTextIter iter;
Packit a7d494
	gchar *word;
Packit a7d494
Packit a7d494
	if (!gtk_source_completion_context_get_iter (context, &iter))
Packit a7d494
	{
Packit a7d494
		gtk_source_completion_context_add_proposals (context, provider, NULL, TRUE);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_free (words->priv->word);
Packit a7d494
	words->priv->word = NULL;
Packit a7d494
Packit a7d494
	word = get_word_at_iter (&iter);
Packit a7d494
Packit a7d494
	activation = gtk_source_completion_context_get_activation (context);
Packit a7d494
Packit a7d494
	if (word == NULL ||
Packit a7d494
	    (activation == GTK_SOURCE_COMPLETION_ACTIVATION_INTERACTIVE &&
Packit a7d494
	     g_utf8_strlen (word, -1) < (glong)words->priv->minimum_word_size))
Packit a7d494
	{
Packit a7d494
		g_free (word);
Packit a7d494
		gtk_source_completion_context_add_proposals (context, provider, NULL, TRUE);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	words->priv->cancel_id =
Packit a7d494
		g_signal_connect_swapped (context,
Packit a7d494
			                  "cancelled",
Packit a7d494
			                  G_CALLBACK (population_finished),
Packit a7d494
			                  provider);
Packit a7d494
Packit a7d494
	words->priv->context = g_object_ref (context);
Packit a7d494
Packit a7d494
	words->priv->word = word;
Packit a7d494
	words->priv->word_len = strlen (word);
Packit a7d494
Packit a7d494
	/* Do first right now */
Packit a7d494
	if (add_in_idle (words))
Packit a7d494
	{
Packit a7d494
		gtk_source_completion_words_library_lock (words->priv->library);
Packit a7d494
		words->priv->idle_id = gdk_threads_add_idle ((GSourceFunc)add_in_idle,
Packit a7d494
		                                             words);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_dispose (GObject *object)
Packit a7d494
{
Packit a7d494
	GtkSourceCompletionWords *provider = GTK_SOURCE_COMPLETION_WORDS (object);
Packit a7d494
Packit a7d494
	population_finished (provider);
Packit a7d494
Packit a7d494
	while (provider->priv->buffers != NULL)
Packit a7d494
	{
Packit a7d494
		BufferBinding *binding = provider->priv->buffers->data;
Packit a7d494
		GtkTextBuffer *buffer = gtk_source_completion_words_buffer_get_buffer (binding->buffer);
Packit a7d494
Packit a7d494
		gtk_source_completion_words_unregister (provider, buffer);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_free (provider->priv->name);
Packit a7d494
	provider->priv->name = NULL;
Packit a7d494
Packit a7d494
	g_clear_object (&provider->priv->icon);
Packit a7d494
	g_clear_object (&provider->priv->library);
Packit a7d494
Packit a7d494
	G_OBJECT_CLASS (gtk_source_completion_words_parent_class)->dispose (object);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
update_buffers_batch_size (GtkSourceCompletionWords *words)
Packit a7d494
{
Packit a7d494
	GList *item;
Packit a7d494
Packit a7d494
	for (item = words->priv->buffers; item != NULL; item = g_list_next (item))
Packit a7d494
	{
Packit a7d494
		BufferBinding *binding = item->data;
Packit a7d494
		gtk_source_completion_words_buffer_set_scan_batch_size (binding->buffer,
Packit a7d494
		                                                        words->priv->scan_batch_size);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
update_buffers_minimum_word_size (GtkSourceCompletionWords *words)
Packit a7d494
{
Packit a7d494
	GList *item;
Packit a7d494
Packit a7d494
	for (item = words->priv->buffers; item != NULL; item = g_list_next (item))
Packit a7d494
	{
Packit a7d494
		BufferBinding *binding = (BufferBinding *)item->data;
Packit a7d494
		gtk_source_completion_words_buffer_set_minimum_word_size (binding->buffer,
Packit a7d494
		                                                          words->priv->minimum_word_size);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_set_property (GObject      *object,
Packit a7d494
                                          guint         prop_id,
Packit a7d494
                                          const GValue *value,
Packit a7d494
                                          GParamSpec   *pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceCompletionWords *self = GTK_SOURCE_COMPLETION_WORDS (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_NAME:
Packit a7d494
			g_free (self->priv->name);
Packit a7d494
			self->priv->name = g_value_dup_string (value);
Packit a7d494
Packit a7d494
			if (self->priv->name == NULL)
Packit a7d494
			{
Packit a7d494
				self->priv->name = g_strdup (_("Document Words"));
Packit a7d494
			}
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_ICON:
Packit a7d494
			g_clear_object (&self->priv->icon);
Packit a7d494
			self->priv->icon = g_value_dup_object (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_PROPOSALS_BATCH_SIZE:
Packit a7d494
			self->priv->proposals_batch_size = g_value_get_uint (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_SCAN_BATCH_SIZE:
Packit a7d494
			self->priv->scan_batch_size = g_value_get_uint (value);
Packit a7d494
			update_buffers_batch_size (self);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_MINIMUM_WORD_SIZE:
Packit a7d494
			self->priv->minimum_word_size = g_value_get_uint (value);
Packit a7d494
			update_buffers_minimum_word_size (self);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_INTERACTIVE_DELAY:
Packit a7d494
			self->priv->interactive_delay = g_value_get_int (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_PRIORITY:
Packit a7d494
			self->priv->priority = g_value_get_int (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_ACTIVATION:
Packit a7d494
			self->priv->activation = g_value_get_flags (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		default:
Packit a7d494
			G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit a7d494
			break;
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_get_property (GObject    *object,
Packit a7d494
                                          guint       prop_id,
Packit a7d494
                                          GValue     *value,
Packit a7d494
                                          GParamSpec *pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceCompletionWords *self = GTK_SOURCE_COMPLETION_WORDS (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_NAME:
Packit a7d494
			g_value_set_string (value, self->priv->name);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_ICON:
Packit a7d494
			g_value_set_object (value, self->priv->icon);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_PROPOSALS_BATCH_SIZE:
Packit a7d494
			g_value_set_uint (value, self->priv->proposals_batch_size);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_SCAN_BATCH_SIZE:
Packit a7d494
			g_value_set_uint (value, self->priv->scan_batch_size);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_MINIMUM_WORD_SIZE:
Packit a7d494
			g_value_set_uint (value, self->priv->minimum_word_size);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_INTERACTIVE_DELAY:
Packit a7d494
			g_value_set_int (value, self->priv->interactive_delay);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_PRIORITY:
Packit a7d494
			g_value_set_int (value, self->priv->priority);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_ACTIVATION:
Packit a7d494
			g_value_set_flags (value, self->priv->activation);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		default:
Packit a7d494
			G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit a7d494
			break;
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_class_init (GtkSourceCompletionWordsClass *klass)
Packit a7d494
{
Packit a7d494
	GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit a7d494
Packit a7d494
	object_class->get_property = gtk_source_completion_words_get_property;
Packit a7d494
	object_class->set_property = gtk_source_completion_words_set_property;
Packit a7d494
	object_class->dispose = gtk_source_completion_words_dispose;
Packit a7d494
Packit a7d494
	properties[PROP_NAME] =
Packit a7d494
		g_param_spec_string ("name",
Packit a7d494
				     "Name",
Packit a7d494
				     "The provider name",
Packit a7d494
				     NULL,
Packit a7d494
				     G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	properties[PROP_ICON] =
Packit a7d494
		g_param_spec_object ("icon",
Packit a7d494
				     "Icon",
Packit a7d494
				     "The provider icon",
Packit a7d494
				     GDK_TYPE_PIXBUF,
Packit a7d494
				     G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	properties[PROP_PROPOSALS_BATCH_SIZE] =
Packit a7d494
		g_param_spec_uint ("proposals-batch-size",
Packit a7d494
				   "Proposals Batch Size",
Packit a7d494
				   "Number of proposals added in one batch",
Packit a7d494
				   1,
Packit a7d494
				   G_MAXUINT,
Packit a7d494
				   300,
Packit a7d494
				   G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	properties[PROP_SCAN_BATCH_SIZE] =
Packit a7d494
		g_param_spec_uint ("scan-batch-size",
Packit a7d494
				   "Scan Batch Size",
Packit a7d494
				   "Number of lines scanned in one batch",
Packit a7d494
				   1,
Packit a7d494
				   G_MAXUINT,
Packit a7d494
				   50,
Packit a7d494
				   G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	properties[PROP_MINIMUM_WORD_SIZE] =
Packit a7d494
		g_param_spec_uint ("minimum-word-size",
Packit a7d494
				   "Minimum Word Size",
Packit a7d494
				   "The minimum word size to complete",
Packit a7d494
				   2,
Packit a7d494
				   G_MAXUINT,
Packit a7d494
				   2,
Packit a7d494
				   G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	properties[PROP_INTERACTIVE_DELAY] =
Packit a7d494
		g_param_spec_int ("interactive-delay",
Packit a7d494
				  "Interactive Delay",
Packit a7d494
				  "The delay before initiating interactive completion",
Packit a7d494
				  -1,
Packit a7d494
				  G_MAXINT,
Packit a7d494
				  50,
Packit a7d494
				  G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	properties[PROP_PRIORITY] =
Packit a7d494
		g_param_spec_int ("priority",
Packit a7d494
				  "Priority",
Packit a7d494
				  "Provider priority",
Packit a7d494
				  G_MININT,
Packit a7d494
				  G_MAXINT,
Packit a7d494
				  0,
Packit a7d494
				  G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceCompletionWords:activation:
Packit a7d494
	 *
Packit a7d494
	 * The type of activation.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.10
Packit a7d494
	 */
Packit a7d494
	properties[PROP_ACTIVATION] =
Packit a7d494
		g_param_spec_flags ("activation",
Packit a7d494
				    "Activation",
Packit a7d494
				    "The type of activation",
Packit a7d494
				    GTK_SOURCE_TYPE_COMPLETION_ACTIVATION,
Packit a7d494
				    GTK_SOURCE_COMPLETION_ACTIVATION_INTERACTIVE |
Packit a7d494
				    GTK_SOURCE_COMPLETION_ACTIVATION_USER_REQUESTED,
Packit a7d494
				    G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
Packit a7d494
Packit a7d494
	g_object_class_install_properties (object_class, N_PROPERTIES, properties);
Packit a7d494
}
Packit a7d494
Packit a7d494
static gboolean
Packit a7d494
gtk_source_completion_words_get_start_iter (GtkSourceCompletionProvider *provider,
Packit a7d494
                                            GtkSourceCompletionContext  *context,
Packit a7d494
                                            GtkSourceCompletionProposal *proposal,
Packit a7d494
                                            GtkTextIter                 *iter)
Packit a7d494
{
Packit a7d494
	gchar *word;
Packit a7d494
	glong nb_chars;
Packit a7d494
Packit a7d494
	if (!gtk_source_completion_context_get_iter (context, iter))
Packit a7d494
	{
Packit a7d494
		return FALSE;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	word = get_word_at_iter (iter);
Packit a7d494
	g_return_val_if_fail (word != NULL, FALSE);
Packit a7d494
Packit a7d494
	nb_chars = g_utf8_strlen (word, -1);
Packit a7d494
	gtk_text_iter_backward_chars (iter, nb_chars);
Packit a7d494
Packit a7d494
	g_free (word);
Packit a7d494
	return TRUE;
Packit a7d494
}
Packit a7d494
Packit a7d494
static gint
Packit a7d494
gtk_source_completion_words_get_interactive_delay (GtkSourceCompletionProvider *provider)
Packit a7d494
{
Packit a7d494
	return GTK_SOURCE_COMPLETION_WORDS (provider)->priv->interactive_delay;
Packit a7d494
}
Packit a7d494
Packit a7d494
static gint
Packit a7d494
gtk_source_completion_words_get_priority (GtkSourceCompletionProvider *provider)
Packit a7d494
{
Packit a7d494
	return GTK_SOURCE_COMPLETION_WORDS (provider)->priv->priority;
Packit a7d494
}
Packit a7d494
Packit a7d494
static GtkSourceCompletionActivation
Packit a7d494
gtk_source_completion_words_get_activation (GtkSourceCompletionProvider *provider)
Packit a7d494
{
Packit a7d494
	return GTK_SOURCE_COMPLETION_WORDS (provider)->priv->activation;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_iface_init (GtkSourceCompletionProviderIface *iface)
Packit a7d494
{
Packit a7d494
	iface->get_name = gtk_source_completion_words_get_name;
Packit a7d494
	iface->get_icon = gtk_source_completion_words_get_icon;
Packit a7d494
	iface->populate = gtk_source_completion_words_populate;
Packit a7d494
	iface->get_start_iter = gtk_source_completion_words_get_start_iter;
Packit a7d494
	iface->get_interactive_delay = gtk_source_completion_words_get_interactive_delay;
Packit a7d494
	iface->get_priority = gtk_source_completion_words_get_priority;
Packit a7d494
	iface->get_activation = gtk_source_completion_words_get_activation;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_completion_words_init (GtkSourceCompletionWords *self)
Packit a7d494
{
Packit a7d494
	self->priv = gtk_source_completion_words_get_instance_private (self);
Packit a7d494
Packit a7d494
	self->priv->library = gtk_source_completion_words_library_new ();
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_completion_words_new:
Packit a7d494
 * @name: (nullable): The name for the provider, or %NULL.
Packit a7d494
 * @icon: (nullable): A specific icon for the provider, or %NULL.
Packit a7d494
 *
Packit a7d494
 * Returns: a new #GtkSourceCompletionWords provider
Packit a7d494
 */
Packit a7d494
GtkSourceCompletionWords *
Packit a7d494
gtk_source_completion_words_new (const gchar *name,
Packit a7d494
                                 GdkPixbuf   *icon)
Packit a7d494
{
Packit a7d494
	return g_object_new (GTK_SOURCE_TYPE_COMPLETION_WORDS,
Packit a7d494
	                     "name", name,
Packit a7d494
	                     "icon", icon,
Packit a7d494
	                     NULL);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
buffer_destroyed (BufferBinding *binding)
Packit a7d494
{
Packit a7d494
	binding->words->priv->buffers = g_list_remove (binding->words->priv->buffers,
Packit a7d494
	                                               binding);
Packit a7d494
	g_object_unref (binding->buffer);
Packit a7d494
	g_slice_free (BufferBinding, binding);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_completion_words_register:
Packit a7d494
 * @words: a #GtkSourceCompletionWords
Packit a7d494
 * @buffer: a #GtkTextBuffer
Packit a7d494
 *
Packit a7d494
 * Registers @buffer in the @words provider.
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_completion_words_register (GtkSourceCompletionWords *words,
Packit a7d494
                                      GtkTextBuffer            *buffer)
Packit a7d494
{
Packit a7d494
	GtkSourceCompletionWordsBuffer *buf;
Packit a7d494
	BufferBinding *binding;
Packit a7d494
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_COMPLETION_WORDS (words));
Packit a7d494
	g_return_if_fail (GTK_IS_TEXT_BUFFER (buffer));
Packit a7d494
Packit a7d494
	binding = g_object_get_data (G_OBJECT (buffer), BUFFER_KEY);
Packit a7d494
Packit a7d494
	if (binding != NULL)
Packit a7d494
	{
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	buf = gtk_source_completion_words_buffer_new (words->priv->library,
Packit a7d494
	                                              buffer);
Packit a7d494
Packit a7d494
	gtk_source_completion_words_buffer_set_scan_batch_size (buf,
Packit a7d494
	                                                        words->priv->scan_batch_size);
Packit a7d494
Packit a7d494
	gtk_source_completion_words_buffer_set_minimum_word_size (buf,
Packit a7d494
	                                                          words->priv->minimum_word_size);
Packit a7d494
Packit a7d494
	binding = g_slice_new (BufferBinding);
Packit a7d494
	binding->words = words;
Packit a7d494
	binding->buffer = buf;
Packit a7d494
Packit a7d494
	g_object_set_data_full (G_OBJECT (buffer),
Packit a7d494
	                        BUFFER_KEY,
Packit a7d494
	                        binding,
Packit a7d494
	                        (GDestroyNotify)buffer_destroyed);
Packit a7d494
Packit a7d494
	words->priv->buffers = g_list_prepend (words->priv->buffers,
Packit a7d494
	                                       binding);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_completion_words_unregister:
Packit a7d494
 * @words: a #GtkSourceCompletionWords
Packit a7d494
 * @buffer: a #GtkTextBuffer
Packit a7d494
 *
Packit a7d494
 * Unregisters @buffer from the @words provider.
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_completion_words_unregister (GtkSourceCompletionWords *words,
Packit a7d494
                                        GtkTextBuffer            *buffer)
Packit a7d494
{
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_COMPLETION_WORDS (words));
Packit a7d494
	g_return_if_fail (GTK_IS_TEXT_BUFFER (buffer));
Packit a7d494
Packit a7d494
	g_object_set_data (G_OBJECT (buffer), BUFFER_KEY, NULL);
Packit a7d494
}