mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-07 19:30:12 +00:00
e13692c52f
This was broken in various ways; Tabbing between pages was causing segfaults, and using the hover state like this does not work anymore, with hover now being completely managed by GTK. Use focus instead, and also fix up the style.
664 lines
18 KiB
C
664 lines
18 KiB
C
/* gtkemojicompletion.c: An Emoji picker widget
|
|
* Copyright 2017, Red Hat, Inc.
|
|
*
|
|
* This library 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 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkemojicompletion.h"
|
|
|
|
#include "gtktextprivate.h"
|
|
#include "gtkeditable.h"
|
|
#include "gtkbox.h"
|
|
#include "gtkcssprovider.h"
|
|
#include "gtklistbox.h"
|
|
#include "gtklabel.h"
|
|
#include "gtkpopover.h"
|
|
#include "gtkintl.h"
|
|
#include "gtkprivate.h"
|
|
#include "gtkgesturelongpress.h"
|
|
#include "gtkeventcontrollerkey.h"
|
|
#include "gtkflowbox.h"
|
|
#include "gtkstack.h"
|
|
#include "gtkstylecontext.h"
|
|
|
|
struct _GtkEmojiCompletion
|
|
{
|
|
GtkPopover parent_instance;
|
|
|
|
GtkText *entry;
|
|
char *text;
|
|
guint length;
|
|
guint offset;
|
|
gulong changed_id;
|
|
guint n_matches;
|
|
|
|
GtkWidget *list;
|
|
GtkWidget *active;
|
|
GtkWidget *active_variation;
|
|
|
|
GVariant *data;
|
|
};
|
|
|
|
struct _GtkEmojiCompletionClass {
|
|
GtkPopoverClass parent_class;
|
|
};
|
|
|
|
static void connect_signals (GtkEmojiCompletion *completion,
|
|
GtkText *text);
|
|
static void disconnect_signals (GtkEmojiCompletion *completion);
|
|
static int populate_completion (GtkEmojiCompletion *completion,
|
|
const char *text,
|
|
guint offset);
|
|
|
|
#define MAX_ROWS 5
|
|
|
|
G_DEFINE_TYPE (GtkEmojiCompletion, gtk_emoji_completion, GTK_TYPE_POPOVER)
|
|
|
|
static void
|
|
gtk_emoji_completion_finalize (GObject *object)
|
|
{
|
|
GtkEmojiCompletion *completion = GTK_EMOJI_COMPLETION (object);
|
|
|
|
disconnect_signals (completion);
|
|
|
|
g_free (completion->text);
|
|
g_variant_unref (completion->data);
|
|
|
|
G_OBJECT_CLASS (gtk_emoji_completion_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
update_completion (GtkEmojiCompletion *completion)
|
|
{
|
|
const char *text;
|
|
guint length;
|
|
guint n_matches;
|
|
|
|
n_matches = 0;
|
|
|
|
text = gtk_editable_get_text (GTK_EDITABLE (completion->entry));
|
|
length = strlen (text);
|
|
|
|
if (length > 0)
|
|
{
|
|
gboolean found_candidate = FALSE;
|
|
const char *p;
|
|
|
|
p = text + length;
|
|
do
|
|
{
|
|
next:
|
|
p = g_utf8_prev_char (p);
|
|
if (*p == ':')
|
|
{
|
|
if (p + 1 == text + length)
|
|
goto next;
|
|
|
|
if (p == text || !g_unichar_isalnum (g_utf8_get_char (p - 1)))
|
|
{
|
|
found_candidate = TRUE;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
while (g_unichar_isalnum (g_utf8_get_char (p)) || *p == '_');
|
|
|
|
if (found_candidate)
|
|
n_matches = populate_completion (completion, p, 0);
|
|
}
|
|
|
|
if (n_matches > 0)
|
|
gtk_popover_popup (GTK_POPOVER (completion));
|
|
else
|
|
gtk_popover_popdown (GTK_POPOVER (completion));
|
|
}
|
|
|
|
static void
|
|
changed_cb (GtkText *text,
|
|
GtkEmojiCompletion *completion)
|
|
{
|
|
update_completion (completion);
|
|
}
|
|
|
|
static void
|
|
emoji_activated (GtkWidget *row,
|
|
GtkEmojiCompletion *completion)
|
|
{
|
|
const char *emoji;
|
|
guint length;
|
|
|
|
gtk_popover_popdown (GTK_POPOVER (completion));
|
|
|
|
emoji = (const char *)g_object_get_data (G_OBJECT (row), "text");
|
|
|
|
g_signal_handler_block (completion->entry, completion->changed_id);
|
|
|
|
length = g_utf8_strlen (gtk_editable_get_text (GTK_EDITABLE (completion->entry)), -1);
|
|
gtk_editable_select_region (GTK_EDITABLE (completion->entry), length - completion->length, length);
|
|
gtk_text_enter_text (completion->entry, emoji);
|
|
|
|
g_signal_handler_unblock (completion->entry, completion->changed_id);
|
|
}
|
|
|
|
static void
|
|
row_activated (GtkListBox *list,
|
|
GtkListBoxRow *row,
|
|
gpointer data)
|
|
{
|
|
GtkEmojiCompletion *completion = data;
|
|
|
|
emoji_activated (GTK_WIDGET (row), completion);
|
|
}
|
|
|
|
static void
|
|
child_activated (GtkFlowBox *box,
|
|
GtkFlowBoxChild *child,
|
|
gpointer data)
|
|
{
|
|
GtkEmojiCompletion *completion = data;
|
|
|
|
emoji_activated (GTK_WIDGET (child), completion);
|
|
}
|
|
|
|
static void
|
|
move_active_row (GtkEmojiCompletion *completion,
|
|
int direction)
|
|
{
|
|
GtkWidget *child;
|
|
|
|
for (child = gtk_widget_get_first_child (completion->list);
|
|
child != NULL;
|
|
child = gtk_widget_get_next_sibling (child))
|
|
gtk_widget_unset_state_flags (child, GTK_STATE_FLAG_FOCUSED);
|
|
|
|
if (completion->active != NULL)
|
|
{
|
|
if (direction == 1)
|
|
completion->active = gtk_widget_get_next_sibling (completion->active);
|
|
else
|
|
completion->active = gtk_widget_get_prev_sibling (completion->active);
|
|
}
|
|
|
|
if (completion->active == NULL)
|
|
{
|
|
if (direction == 1)
|
|
completion->active = gtk_widget_get_first_child (completion->list);
|
|
else
|
|
completion->active = gtk_widget_get_last_child (completion->list);
|
|
}
|
|
|
|
if (completion->active != NULL)
|
|
gtk_widget_set_state_flags (completion->active, GTK_STATE_FLAG_FOCUSED, FALSE);
|
|
|
|
if (completion->active_variation)
|
|
{
|
|
gtk_widget_unset_state_flags (completion->active_variation, GTK_STATE_FLAG_FOCUSED);
|
|
completion->active_variation = NULL;
|
|
}
|
|
}
|
|
|
|
static void
|
|
activate_active_row (GtkEmojiCompletion *completion)
|
|
{
|
|
if (GTK_IS_FLOW_BOX_CHILD (completion->active_variation))
|
|
emoji_activated (completion->active_variation, completion);
|
|
else if (completion->active != NULL)
|
|
emoji_activated (completion->active, completion);
|
|
}
|
|
|
|
static void
|
|
show_variations (GtkEmojiCompletion *completion,
|
|
GtkWidget *row,
|
|
gboolean visible)
|
|
{
|
|
GtkWidget *stack;
|
|
GtkWidget *box;
|
|
GtkWidget *child;
|
|
gboolean is_visible;
|
|
|
|
if (!row)
|
|
return;
|
|
|
|
stack = GTK_WIDGET (g_object_get_data (G_OBJECT (row), "stack"));
|
|
box = gtk_stack_get_child_by_name (GTK_STACK (stack), "variations");
|
|
if (!box)
|
|
return;
|
|
|
|
is_visible = gtk_stack_get_visible_child (GTK_STACK (stack)) == box;
|
|
if (is_visible == visible)
|
|
return;
|
|
|
|
gtk_stack_set_visible_child_name (GTK_STACK (stack), visible ? "variations" : "text");
|
|
for (child = gtk_widget_get_first_child (box); child; child = gtk_widget_get_next_sibling (child))
|
|
gtk_widget_unset_state_flags (child, GTK_STATE_FLAG_FOCUSED);
|
|
completion->active_variation = NULL;
|
|
}
|
|
|
|
static gboolean
|
|
move_active_variation (GtkEmojiCompletion *completion,
|
|
int direction)
|
|
{
|
|
GtkWidget *base;
|
|
GtkWidget *stack;
|
|
GtkWidget *box;
|
|
GtkWidget *next;
|
|
|
|
if (!completion->active)
|
|
return FALSE;
|
|
|
|
base = GTK_WIDGET (g_object_get_data (G_OBJECT (completion->active), "base"));
|
|
stack = GTK_WIDGET (g_object_get_data (G_OBJECT (completion->active), "stack"));
|
|
box = gtk_stack_get_child_by_name (GTK_STACK (stack), "variations");
|
|
|
|
if (gtk_stack_get_visible_child (GTK_STACK (stack)) != box)
|
|
return FALSE;
|
|
|
|
next = NULL;
|
|
|
|
if (!completion->active_variation)
|
|
next = base;
|
|
else if (completion->active_variation == base && direction == 1)
|
|
next = gtk_widget_get_first_child (box);
|
|
else if (completion->active_variation == gtk_widget_get_first_child (box) && direction == -1)
|
|
next = base;
|
|
else if (direction == 1)
|
|
next = gtk_widget_get_next_sibling (completion->active_variation);
|
|
else if (direction == -1)
|
|
next = gtk_widget_get_prev_sibling (completion->active_variation);
|
|
|
|
if (next)
|
|
{
|
|
if (completion->active_variation)
|
|
gtk_widget_unset_state_flags (completion->active_variation, GTK_STATE_FLAG_FOCUSED);
|
|
completion->active_variation = next;
|
|
gtk_widget_set_state_flags (completion->active_variation, GTK_STATE_FLAG_FOCUSED, FALSE);
|
|
}
|
|
|
|
return next != NULL;
|
|
}
|
|
|
|
static gboolean
|
|
key_press_cb (GtkEventControllerKey *key,
|
|
guint keyval,
|
|
guint keycode,
|
|
GdkModifierType modifiers,
|
|
GtkEmojiCompletion *completion)
|
|
{
|
|
if (!gtk_widget_get_visible (GTK_WIDGET (completion)))
|
|
return FALSE;
|
|
|
|
if (keyval == GDK_KEY_Escape)
|
|
{
|
|
gtk_popover_popdown (GTK_POPOVER (completion));
|
|
return TRUE;
|
|
}
|
|
|
|
if (keyval == GDK_KEY_Tab)
|
|
{
|
|
show_variations (completion, completion->active, FALSE);
|
|
|
|
guint offset = completion->offset + MAX_ROWS;
|
|
if (offset >= completion->n_matches)
|
|
offset = 0;
|
|
populate_completion (completion, completion->text, offset);
|
|
return TRUE;
|
|
}
|
|
|
|
if (keyval == GDK_KEY_Up)
|
|
{
|
|
show_variations (completion, completion->active, FALSE);
|
|
|
|
move_active_row (completion, -1);
|
|
return TRUE;
|
|
}
|
|
|
|
if (keyval == GDK_KEY_Down)
|
|
{
|
|
show_variations (completion, completion->active, FALSE);
|
|
|
|
move_active_row (completion, 1);
|
|
return TRUE;
|
|
}
|
|
|
|
if (keyval == GDK_KEY_Return ||
|
|
keyval == GDK_KEY_KP_Enter ||
|
|
keyval == GDK_KEY_ISO_Enter)
|
|
{
|
|
activate_active_row (completion);
|
|
return TRUE;
|
|
}
|
|
|
|
if (keyval == GDK_KEY_Right)
|
|
{
|
|
show_variations (completion, completion->active, TRUE);
|
|
move_active_variation (completion, 1);
|
|
return TRUE;
|
|
}
|
|
|
|
if (keyval == GDK_KEY_Left)
|
|
{
|
|
if (!move_active_variation (completion, -1))
|
|
show_variations (completion, completion->active, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
focus_out_cb (GtkWidget *text,
|
|
GParamSpec *pspec,
|
|
GtkEmojiCompletion *completion)
|
|
{
|
|
if (!gtk_widget_has_focus (text))
|
|
gtk_popover_popdown (GTK_POPOVER (completion));
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
connect_signals (GtkEmojiCompletion *completion,
|
|
GtkText *entry)
|
|
{
|
|
GtkEventController *key_controller;
|
|
|
|
completion->entry = g_object_ref (entry);
|
|
key_controller = gtk_text_get_key_controller (entry);
|
|
|
|
g_signal_connect (key_controller, "key-pressed", G_CALLBACK (key_press_cb), completion);
|
|
completion->changed_id = g_signal_connect (entry, "changed", G_CALLBACK (changed_cb), completion);
|
|
g_signal_connect (entry, "notify::has-focus", G_CALLBACK (focus_out_cb), completion);
|
|
}
|
|
|
|
static void
|
|
disconnect_signals (GtkEmojiCompletion *completion)
|
|
{
|
|
GtkEventController *key_controller;
|
|
|
|
key_controller = gtk_text_get_key_controller (completion->entry);
|
|
|
|
g_signal_handlers_disconnect_by_func (completion->entry, changed_cb, completion);
|
|
g_signal_handlers_disconnect_by_func (key_controller, key_press_cb, completion);
|
|
g_signal_handlers_disconnect_by_func (completion->entry, focus_out_cb, completion);
|
|
|
|
g_clear_object (&completion->entry);
|
|
}
|
|
|
|
static gboolean
|
|
has_variations (GVariant *emoji_data)
|
|
{
|
|
GVariant *codes;
|
|
gsize i;
|
|
gboolean has_variations;
|
|
|
|
has_variations = FALSE;
|
|
codes = g_variant_get_child_value (emoji_data, 0);
|
|
for (i = 0; i < g_variant_n_children (codes); i++)
|
|
{
|
|
gunichar code;
|
|
g_variant_get_child (codes, i, "u", &code);
|
|
if (code == 0)
|
|
{
|
|
has_variations = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
g_variant_unref (codes);
|
|
|
|
return has_variations;
|
|
}
|
|
|
|
static void
|
|
get_text (GVariant *emoji_data,
|
|
gunichar modifier,
|
|
char *text,
|
|
gsize length)
|
|
{
|
|
GVariant *codes;
|
|
gsize i;
|
|
char *p;
|
|
|
|
p = text;
|
|
codes = g_variant_get_child_value (emoji_data, 0);
|
|
for (i = 0; i < g_variant_n_children (codes); i++)
|
|
{
|
|
gunichar code;
|
|
|
|
g_variant_get_child (codes, i, "u", &code);
|
|
if (code == 0)
|
|
code = modifier;
|
|
if (code != 0)
|
|
p += g_unichar_to_utf8 (code, p);
|
|
}
|
|
g_variant_unref (codes);
|
|
p += g_unichar_to_utf8 (0xFE0F, p); /* U+FE0F is the Emoji variation selector */
|
|
p[0] = 0;
|
|
}
|
|
|
|
static void
|
|
add_emoji_variation (GtkWidget *box,
|
|
GVariant *emoji_data,
|
|
gunichar modifier)
|
|
{
|
|
GtkWidget *child;
|
|
GtkWidget *label;
|
|
PangoAttrList *attrs;
|
|
char text[64];
|
|
|
|
get_text (emoji_data, modifier, text, 64);
|
|
|
|
label = gtk_label_new (text);
|
|
attrs = pango_attr_list_new ();
|
|
pango_attr_list_insert (attrs, pango_attr_scale_new (PANGO_SCALE_X_LARGE));
|
|
gtk_label_set_attributes (GTK_LABEL (label), attrs);
|
|
pango_attr_list_unref (attrs);
|
|
|
|
child = g_object_new (GTK_TYPE_FLOW_BOX_CHILD, "css-name", "emoji", NULL);
|
|
g_object_set_data_full (G_OBJECT (child), "text", g_strdup (text), g_free);
|
|
g_object_set_data_full (G_OBJECT (child), "emoji-data",
|
|
g_variant_ref (emoji_data),
|
|
(GDestroyNotify)g_variant_unref);
|
|
if (modifier != 0)
|
|
g_object_set_data (G_OBJECT (child), "modifier", GUINT_TO_POINTER (modifier));
|
|
|
|
gtk_container_add (GTK_CONTAINER (child), label);
|
|
gtk_flow_box_insert (GTK_FLOW_BOX (box), child, -1);
|
|
}
|
|
|
|
static void
|
|
add_emoji (GtkWidget *list,
|
|
GVariant *emoji_data,
|
|
GtkEmojiCompletion *completion)
|
|
{
|
|
GtkWidget *child;
|
|
GtkWidget *label;
|
|
GtkWidget *box;
|
|
PangoAttrList *attrs;
|
|
char text[64];
|
|
const char *shortname;
|
|
GtkWidget *stack;
|
|
gunichar modifier;
|
|
|
|
get_text (emoji_data, 0, text, 64);
|
|
|
|
label = gtk_label_new (text);
|
|
attrs = pango_attr_list_new ();
|
|
pango_attr_list_insert (attrs, pango_attr_scale_new (PANGO_SCALE_X_LARGE));
|
|
gtk_label_set_attributes (GTK_LABEL (label), attrs);
|
|
pango_attr_list_unref (attrs);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (label), "emoji");
|
|
|
|
child = g_object_new (GTK_TYPE_LIST_BOX_ROW, "css-name", "emoji-completion-row", NULL);
|
|
gtk_widget_set_focus_on_click (child, FALSE);
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_container_add (GTK_CONTAINER (child), box);
|
|
gtk_container_add (GTK_CONTAINER (box), label);
|
|
g_object_set_data (G_OBJECT (child), "base", label);
|
|
|
|
stack = gtk_stack_new ();
|
|
gtk_stack_set_homogeneous (GTK_STACK (stack), TRUE);
|
|
gtk_stack_set_transition_type (GTK_STACK (stack), GTK_STACK_TRANSITION_TYPE_OVER_RIGHT_LEFT);
|
|
gtk_container_add (GTK_CONTAINER (box), stack);
|
|
g_object_set_data (G_OBJECT (child), "stack", stack);
|
|
|
|
g_variant_get_child (emoji_data, 2, "&s", &shortname);
|
|
label = gtk_label_new (shortname);
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0);
|
|
|
|
gtk_stack_add_named (GTK_STACK (stack), label, "text");
|
|
|
|
if (has_variations (emoji_data))
|
|
{
|
|
box = gtk_flow_box_new ();
|
|
gtk_flow_box_set_homogeneous (GTK_FLOW_BOX (box), TRUE);
|
|
gtk_flow_box_set_min_children_per_line (GTK_FLOW_BOX (box), 5);
|
|
gtk_flow_box_set_max_children_per_line (GTK_FLOW_BOX (box), 5);
|
|
gtk_flow_box_set_activate_on_single_click (GTK_FLOW_BOX (box), TRUE);
|
|
gtk_flow_box_set_selection_mode (GTK_FLOW_BOX (box), GTK_SELECTION_NONE);
|
|
g_signal_connect (box, "child-activated", G_CALLBACK (child_activated), completion);
|
|
for (modifier = 0x1f3fb; modifier <= 0x1f3ff; modifier++)
|
|
add_emoji_variation (box, emoji_data, modifier);
|
|
|
|
gtk_stack_add_named (GTK_STACK (stack), box, "variations");
|
|
}
|
|
|
|
g_object_set_data_full (G_OBJECT (child), "text", g_strdup (text), g_free);
|
|
g_object_set_data_full (G_OBJECT (child), "emoji-data",
|
|
g_variant_ref (emoji_data), (GDestroyNotify)g_variant_unref);
|
|
|
|
gtk_list_box_insert (GTK_LIST_BOX (list), child, -1);
|
|
}
|
|
|
|
static int
|
|
populate_completion (GtkEmojiCompletion *completion,
|
|
const char *text,
|
|
guint offset)
|
|
{
|
|
GList *children, *l;
|
|
guint n_matches;
|
|
guint n_added;
|
|
GVariantIter iter;
|
|
GVariant *item;
|
|
|
|
if (completion->text != text)
|
|
{
|
|
g_free (completion->text);
|
|
completion->text = g_strdup (text);
|
|
completion->length = g_utf8_strlen (text, -1);
|
|
}
|
|
completion->offset = offset;
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (completion->list));
|
|
for (l = children; l; l = l->next)
|
|
gtk_widget_destroy (GTK_WIDGET (l->data));
|
|
g_list_free (children);
|
|
|
|
completion->active = NULL;
|
|
|
|
n_matches = 0;
|
|
n_added = 0;
|
|
g_variant_iter_init (&iter, completion->data);
|
|
while ((item = g_variant_iter_next_value (&iter)))
|
|
{
|
|
const char *shortname;
|
|
|
|
g_variant_get_child (item, 2, "&s", &shortname);
|
|
if (g_str_has_prefix (shortname, text))
|
|
{
|
|
n_matches++;
|
|
|
|
if (n_matches > offset && n_added < MAX_ROWS)
|
|
{
|
|
add_emoji (completion->list, item, completion);
|
|
n_added++;
|
|
}
|
|
}
|
|
}
|
|
|
|
completion->n_matches = n_matches;
|
|
|
|
if (n_added > 0)
|
|
{
|
|
completion->active = gtk_widget_get_first_child (completion->list);
|
|
gtk_widget_set_state_flags (completion->active, GTK_STATE_FLAG_FOCUSED, FALSE);
|
|
}
|
|
|
|
return n_added;
|
|
}
|
|
|
|
static void
|
|
long_pressed_cb (GtkGesture *gesture,
|
|
double x,
|
|
double y,
|
|
gpointer data)
|
|
{
|
|
GtkEmojiCompletion *completion = data;
|
|
GtkWidget *row;
|
|
|
|
row = GTK_WIDGET (gtk_list_box_get_row_at_y (GTK_LIST_BOX (completion->list), y));
|
|
if (!row)
|
|
return;
|
|
|
|
show_variations (completion, row, TRUE);
|
|
}
|
|
|
|
static void
|
|
gtk_emoji_completion_init (GtkEmojiCompletion *completion)
|
|
{
|
|
GBytes *bytes = NULL;
|
|
GtkGesture *long_press;
|
|
|
|
gtk_widget_init_template (GTK_WIDGET (completion));
|
|
|
|
bytes = g_resources_lookup_data ("/org/gtk/libgtk/emoji/emoji.data", 0, NULL);
|
|
completion->data = g_variant_ref_sink (g_variant_new_from_bytes (G_VARIANT_TYPE ("a(auss)"), bytes, TRUE));
|
|
|
|
g_bytes_unref (bytes);
|
|
|
|
long_press = gtk_gesture_long_press_new ();
|
|
g_signal_connect (long_press, "pressed", G_CALLBACK (long_pressed_cb), completion);
|
|
gtk_widget_add_controller (completion->list, GTK_EVENT_CONTROLLER (long_press));
|
|
}
|
|
|
|
static void
|
|
gtk_emoji_completion_class_init (GtkEmojiCompletionClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
object_class->finalize = gtk_emoji_completion_finalize;
|
|
|
|
gtk_widget_class_set_template_from_resource (widget_class, "/org/gtk/libgtk/ui/gtkemojicompletion.ui");
|
|
|
|
gtk_widget_class_bind_template_child (widget_class, GtkEmojiCompletion, list);
|
|
|
|
gtk_widget_class_bind_template_callback (widget_class, row_activated);
|
|
}
|
|
|
|
GtkWidget *
|
|
gtk_emoji_completion_new (GtkText *text)
|
|
{
|
|
GtkEmojiCompletion *completion;
|
|
|
|
completion = GTK_EMOJI_COMPLETION (g_object_new (GTK_TYPE_EMOJI_COMPLETION,
|
|
"relative-to", text,
|
|
NULL));
|
|
|
|
connect_signals (completion, text);
|
|
|
|
return GTK_WIDGET (completion);
|
|
}
|