473 lines
14 KiB
C
473 lines
14 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*-
|
|
* test-completion.c
|
|
* This file is part of GtkSourceView
|
|
*
|
|
* Copyright (C) 2007 - Jesús Barbero Rodríguez <chuchiperriman@gmail.com>
|
|
* 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>
|
|
|
|
typedef struct _TestProvider TestProvider;
|
|
typedef struct _TestProviderClass TestProviderClass;
|
|
|
|
static GtkSourceCompletionWords *word_provider;
|
|
static TestProvider *fixed_provider;
|
|
static TestProvider *random_provider;
|
|
|
|
struct _TestProvider
|
|
{
|
|
GObject parent;
|
|
|
|
GList *proposals;
|
|
gint priority;
|
|
gchar *name;
|
|
|
|
GdkPixbuf *provider_icon;
|
|
|
|
GdkPixbuf *item_icon;
|
|
GIcon *item_gicon;
|
|
|
|
/* If it's a random provider, a subset of 'proposals' are choosen on
|
|
* each populate. Otherwise, all the proposals are shown. */
|
|
guint is_random : 1;
|
|
};
|
|
|
|
struct _TestProviderClass
|
|
{
|
|
GObjectClass parent_class;
|
|
};
|
|
|
|
static void test_provider_iface_init (GtkSourceCompletionProviderIface *iface);
|
|
GType test_provider_get_type (void);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (TestProvider,
|
|
test_provider,
|
|
G_TYPE_OBJECT,
|
|
G_IMPLEMENT_INTERFACE (GTK_SOURCE_TYPE_COMPLETION_PROVIDER,
|
|
test_provider_iface_init))
|
|
|
|
static gchar *
|
|
test_provider_get_name (GtkSourceCompletionProvider *provider)
|
|
{
|
|
return g_strdup (((TestProvider *)provider)->name);
|
|
}
|
|
|
|
static gint
|
|
test_provider_get_priority (GtkSourceCompletionProvider *provider)
|
|
{
|
|
return ((TestProvider *)provider)->priority;
|
|
}
|
|
|
|
static GList *
|
|
select_random_proposals (GList *all_proposals)
|
|
{
|
|
GList *selection = NULL;
|
|
GList *prop;
|
|
|
|
for (prop = all_proposals; prop != NULL; prop = g_list_next (prop))
|
|
{
|
|
if (g_random_boolean ())
|
|
{
|
|
selection = g_list_prepend (selection, prop->data);
|
|
}
|
|
}
|
|
|
|
return selection;
|
|
}
|
|
|
|
static void
|
|
test_provider_populate (GtkSourceCompletionProvider *completion_provider,
|
|
GtkSourceCompletionContext *context)
|
|
{
|
|
TestProvider *provider = (TestProvider *)completion_provider;
|
|
GList *proposals;
|
|
|
|
if (provider->is_random)
|
|
{
|
|
proposals = select_random_proposals (provider->proposals);
|
|
}
|
|
else
|
|
{
|
|
proposals = provider->proposals;
|
|
}
|
|
|
|
gtk_source_completion_context_add_proposals (context,
|
|
completion_provider,
|
|
proposals,
|
|
TRUE);
|
|
}
|
|
|
|
static GdkPixbuf *
|
|
test_provider_get_icon (GtkSourceCompletionProvider *provider)
|
|
{
|
|
TestProvider *tp = (TestProvider *)provider;
|
|
|
|
return tp->is_random ? NULL : tp->provider_icon;
|
|
}
|
|
|
|
static void
|
|
test_provider_iface_init (GtkSourceCompletionProviderIface *iface)
|
|
{
|
|
iface->get_name = test_provider_get_name;
|
|
iface->populate = test_provider_populate;
|
|
iface->get_priority = test_provider_get_priority;
|
|
iface->get_icon = test_provider_get_icon;
|
|
}
|
|
|
|
static void
|
|
test_provider_dispose (GObject *gobject)
|
|
{
|
|
TestProvider *self = (TestProvider *)gobject;
|
|
|
|
g_list_free_full (self->proposals, g_object_unref);
|
|
self->proposals = NULL;
|
|
|
|
g_clear_object (&self->provider_icon);
|
|
g_clear_object (&self->item_icon);
|
|
g_clear_object (&self->item_gicon);
|
|
|
|
G_OBJECT_CLASS (test_provider_parent_class)->dispose (gobject);
|
|
}
|
|
|
|
static void
|
|
test_provider_finalize (GObject *gobject)
|
|
{
|
|
TestProvider *self = (TestProvider *)gobject;
|
|
|
|
g_free (self->name);
|
|
self->name = NULL;
|
|
|
|
G_OBJECT_CLASS (test_provider_parent_class)->finalize (gobject);
|
|
}
|
|
|
|
static void
|
|
test_provider_class_init (TestProviderClass *klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->dispose = test_provider_dispose;
|
|
gobject_class->finalize = test_provider_finalize;
|
|
}
|
|
|
|
static void
|
|
test_provider_init (TestProvider *self)
|
|
{
|
|
GtkIconTheme *theme;
|
|
GIcon *icon;
|
|
GIcon *emblem_icon;
|
|
GEmblem *emblem;
|
|
|
|
theme = gtk_icon_theme_get_default ();
|
|
self->provider_icon = gtk_icon_theme_load_icon (theme, "dialog-information", 16, 0, NULL);
|
|
|
|
self->item_icon = gtk_icon_theme_load_icon (theme, "trophy-gold", 16, 0, NULL);
|
|
|
|
icon = g_themed_icon_new ("trophy-silver");
|
|
emblem_icon = g_themed_icon_new ("emblem-urgent");
|
|
emblem = g_emblem_new (emblem_icon);
|
|
self->item_gicon = g_emblemed_icon_new (icon, emblem);
|
|
g_object_unref (icon);
|
|
g_object_unref (emblem_icon);
|
|
g_object_unref (emblem);
|
|
}
|
|
|
|
static void
|
|
test_provider_set_fixed (TestProvider *provider,
|
|
gint nb_proposals)
|
|
{
|
|
GtkSourceCompletionItem *item;
|
|
GList *proposals = NULL;
|
|
gint i;
|
|
|
|
g_list_free_full (provider->proposals, g_object_unref);
|
|
|
|
item = gtk_source_completion_item_new2 ();
|
|
gtk_source_completion_item_set_markup (item, "A very <b>long</b> proposal. I <i>repeat</i>, a very long proposal!");
|
|
gtk_source_completion_item_set_text (item, "A very long proposal. I repeat, a very long proposal!");
|
|
gtk_source_completion_item_set_icon (item, provider->item_icon);
|
|
gtk_source_completion_item_set_info (item, "To test the horizontal scrollbar and the markup.");
|
|
proposals = g_list_prepend (proposals, item);
|
|
|
|
item = gtk_source_completion_item_new2 ();
|
|
gtk_source_completion_item_set_markup (item, "A proposal with a <b>symbolic</b> icon");
|
|
gtk_source_completion_item_set_text (item, "Test setting the icon-name property");
|
|
gtk_source_completion_item_set_icon_name (item, "face-cool-symbolic");
|
|
proposals = g_list_prepend (proposals, item);
|
|
|
|
item = gtk_source_completion_item_new2 ();
|
|
gtk_source_completion_item_set_markup (item, "A proposal with an emblem <b>GIcon</b>");
|
|
gtk_source_completion_item_set_text (item, "Test setting the GIcon property");
|
|
gtk_source_completion_item_set_gicon (item, provider->item_gicon);
|
|
proposals = g_list_prepend (proposals, item);
|
|
|
|
for (i = nb_proposals - 1; i > 0; i--)
|
|
{
|
|
gchar *name = g_strdup_printf ("Proposal %d", i);
|
|
|
|
item = gtk_source_completion_item_new2 ();
|
|
gtk_source_completion_item_set_label (item, name);
|
|
gtk_source_completion_item_set_text (item, name);
|
|
gtk_source_completion_item_set_icon (item, provider->item_icon);
|
|
gtk_source_completion_item_set_info (item, "The extra info of the proposal.\nA second line.");
|
|
proposals = g_list_prepend (proposals, item);
|
|
|
|
g_free (name);
|
|
}
|
|
|
|
provider->proposals = proposals;
|
|
provider->is_random = 0;
|
|
}
|
|
|
|
static void
|
|
test_provider_set_random (TestProvider *provider,
|
|
gint nb_proposals)
|
|
{
|
|
GList *proposals = NULL;
|
|
gint i;
|
|
|
|
g_list_free_full (provider->proposals, g_object_unref);
|
|
|
|
for (i = 0; i < nb_proposals; i++)
|
|
{
|
|
GtkSourceCompletionItem *item;
|
|
gchar *padding = g_strnfill ((i * 3) % 10, 'o');
|
|
gchar *name = g_strdup_printf ("Propo%ssal %d", padding, i);
|
|
|
|
item = gtk_source_completion_item_new2 ();
|
|
gtk_source_completion_item_set_label (item, name);
|
|
gtk_source_completion_item_set_text (item, name);
|
|
gtk_source_completion_item_set_icon (item, provider->item_icon);
|
|
proposals = g_list_prepend (proposals, item);
|
|
|
|
g_free (padding);
|
|
g_free (name);
|
|
}
|
|
|
|
provider->proposals = proposals;
|
|
provider->is_random = 1;
|
|
}
|
|
|
|
static void
|
|
add_remove_provider (GtkToggleButton *button,
|
|
GtkSourceCompletion *completion,
|
|
GtkSourceCompletionProvider *provider)
|
|
{
|
|
g_return_if_fail (provider != NULL);
|
|
|
|
if (gtk_toggle_button_get_active (button))
|
|
{
|
|
gtk_source_completion_add_provider (completion, provider, NULL);
|
|
}
|
|
else
|
|
{
|
|
gtk_source_completion_remove_provider (completion, provider, NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
enable_word_provider_toggled_cb (GtkToggleButton *button,
|
|
GtkSourceCompletion *completion)
|
|
{
|
|
add_remove_provider (button,
|
|
completion,
|
|
GTK_SOURCE_COMPLETION_PROVIDER (word_provider));
|
|
}
|
|
|
|
static void
|
|
enable_fixed_provider_toggled_cb (GtkToggleButton *button,
|
|
GtkSourceCompletion *completion)
|
|
{
|
|
add_remove_provider (button,
|
|
completion,
|
|
GTK_SOURCE_COMPLETION_PROVIDER (fixed_provider));
|
|
}
|
|
|
|
static void
|
|
enable_random_provider_toggled_cb (GtkToggleButton *button,
|
|
GtkSourceCompletion *completion)
|
|
{
|
|
add_remove_provider (button,
|
|
completion,
|
|
GTK_SOURCE_COMPLETION_PROVIDER (random_provider));
|
|
}
|
|
|
|
static void
|
|
nb_proposals_changed_cb (GtkSpinButton *spin_button,
|
|
TestProvider *provider)
|
|
{
|
|
gint nb_proposals = gtk_spin_button_get_value_as_int (spin_button);
|
|
|
|
if (provider->is_random)
|
|
{
|
|
test_provider_set_random (provider, nb_proposals);
|
|
}
|
|
else
|
|
{
|
|
test_provider_set_fixed (provider, nb_proposals);
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_completion (GtkSourceView *source_view,
|
|
GtkSourceCompletion *completion)
|
|
{
|
|
/* Words completion provider */
|
|
word_provider = gtk_source_completion_words_new (NULL, NULL);
|
|
|
|
gtk_source_completion_words_register (word_provider,
|
|
gtk_text_view_get_buffer (GTK_TEXT_VIEW (source_view)));
|
|
|
|
gtk_source_completion_add_provider (completion,
|
|
GTK_SOURCE_COMPLETION_PROVIDER (word_provider),
|
|
NULL);
|
|
|
|
g_object_set (word_provider, "priority", 10, NULL);
|
|
|
|
/* Fixed provider: the proposals don't change */
|
|
fixed_provider = g_object_new (test_provider_get_type (), NULL);
|
|
test_provider_set_fixed (fixed_provider, 3);
|
|
fixed_provider->priority = 5;
|
|
fixed_provider->name = g_strdup ("Fixed Provider");
|
|
|
|
gtk_source_completion_add_provider (completion,
|
|
GTK_SOURCE_COMPLETION_PROVIDER (fixed_provider),
|
|
NULL);
|
|
|
|
/* Random provider: the proposals vary on each populate */
|
|
random_provider = g_object_new (test_provider_get_type (), NULL);
|
|
test_provider_set_random (random_provider, 10);
|
|
random_provider->priority = 1;
|
|
random_provider->name = g_strdup ("Random Provider");
|
|
|
|
gtk_source_completion_add_provider (completion,
|
|
GTK_SOURCE_COMPLETION_PROVIDER (random_provider),
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
create_window (void)
|
|
{
|
|
GtkBuilder *builder;
|
|
GError *error = NULL;
|
|
GtkWindow *window;
|
|
GtkSourceView *source_view;
|
|
GtkSourceCompletion *completion;
|
|
GtkCheckButton *remember_info_visibility;
|
|
GtkCheckButton *select_on_show;
|
|
GtkCheckButton *show_headers;
|
|
GtkCheckButton *show_icons;
|
|
GtkCheckButton *enable_word_provider;
|
|
GtkCheckButton *enable_fixed_provider;
|
|
GtkCheckButton *enable_random_provider;
|
|
GtkSpinButton *nb_fixed_proposals;
|
|
GtkSpinButton *nb_random_proposals;
|
|
|
|
builder = gtk_builder_new ();
|
|
|
|
gtk_builder_add_from_resource (builder,
|
|
"/org/gnome/gtksourceview/tests/ui/test-completion.ui",
|
|
&error);
|
|
|
|
if (error != NULL)
|
|
{
|
|
g_error ("Impossible to load test-completion.ui: %s", error->message);
|
|
}
|
|
|
|
window = GTK_WINDOW (gtk_builder_get_object (builder, "window"));
|
|
source_view = GTK_SOURCE_VIEW (gtk_builder_get_object (builder, "source_view"));
|
|
remember_info_visibility = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_remember_info_visibility"));
|
|
select_on_show = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_select_on_show"));
|
|
show_headers = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_show_headers"));
|
|
show_icons = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_show_icons"));
|
|
enable_word_provider = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_word_provider"));
|
|
enable_fixed_provider = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_fixed_provider"));
|
|
enable_random_provider = GTK_CHECK_BUTTON (gtk_builder_get_object (builder, "checkbutton_random_provider"));
|
|
nb_fixed_proposals = GTK_SPIN_BUTTON (gtk_builder_get_object (builder, "spinbutton_nb_fixed_proposals"));
|
|
nb_random_proposals = GTK_SPIN_BUTTON (gtk_builder_get_object (builder, "spinbutton_nb_random_proposals"));
|
|
|
|
completion = gtk_source_view_get_completion (source_view);
|
|
|
|
g_signal_connect (window,
|
|
"destroy",
|
|
G_CALLBACK (gtk_main_quit),
|
|
NULL);
|
|
|
|
g_object_bind_property (completion, "remember-info-visibility",
|
|
remember_info_visibility, "active",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_BIDIRECTIONAL);
|
|
|
|
g_object_bind_property (completion, "select-on-show",
|
|
select_on_show, "active",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_BIDIRECTIONAL);
|
|
|
|
g_object_bind_property (completion, "show-headers",
|
|
show_headers, "active",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_BIDIRECTIONAL);
|
|
|
|
g_object_bind_property (completion, "show-icons",
|
|
show_icons, "active",
|
|
G_BINDING_SYNC_CREATE | G_BINDING_BIDIRECTIONAL);
|
|
|
|
create_completion (source_view, completion);
|
|
|
|
g_signal_connect (enable_word_provider,
|
|
"toggled",
|
|
G_CALLBACK (enable_word_provider_toggled_cb),
|
|
completion);
|
|
|
|
g_signal_connect (enable_fixed_provider,
|
|
"toggled",
|
|
G_CALLBACK (enable_fixed_provider_toggled_cb),
|
|
completion);
|
|
|
|
g_signal_connect (enable_random_provider,
|
|
"toggled",
|
|
G_CALLBACK (enable_random_provider_toggled_cb),
|
|
completion);
|
|
|
|
g_signal_connect (nb_fixed_proposals,
|
|
"value-changed",
|
|
G_CALLBACK (nb_proposals_changed_cb),
|
|
fixed_provider);
|
|
|
|
g_signal_connect (nb_random_proposals,
|
|
"value-changed",
|
|
G_CALLBACK (nb_proposals_changed_cb),
|
|
random_provider);
|
|
|
|
g_object_unref (builder);
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
gtk_init (&argc, &argv);
|
|
|
|
create_window ();
|
|
|
|
gtk_main ();
|
|
|
|
/* Not really useful, except for debugging memory leaks. */
|
|
g_object_unref (word_provider);
|
|
g_object_unref (fixed_provider);
|
|
g_object_unref (random_provider);
|
|
|
|
return 0;
|
|
}
|