/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ /* * st-im-text.c: Text widget with input method support * * Copyright 2009 Red Hat, Inc. * * This started as a copy of ClutterIMText converted to use * GtkIMContext rather than ClutterIMContext. Original code: * * Author: raymond liu * * Copyright (C) 2009, Intel Corporation. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * version 2.1 as published by the Free Software Foundation. * * 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 program. If not, see . */ /** * SECTION:st-im-text * @short_description: Text widget with input method support * @stability: Unstable * @see_also: #ClutterText * @include: st-imtext/st-imtext.h * * #StIMText derives from ClutterText and hooks up better text input * via #GtkIMContext. It is meant to be a drop-in replacement for * ClutterIMText but using GtkIMContext rather than ClutterIMContext. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include "st-im-text.h" /* properties */ enum { PROP_0, PROP_INPUT_PURPOSE, PROP_INPUT_HINTS, }; struct _StIMTextPrivate { GtkIMContext *im_context; GdkWindow *window; guint need_im_reset : 1; guint has_preedit : 1; }; G_DEFINE_TYPE_WITH_PRIVATE (StIMText, st_im_text, CLUTTER_TYPE_TEXT) static void st_im_text_dispose (GObject *object) { StIMTextPrivate *priv = ST_IM_TEXT (object)->priv; G_OBJECT_CLASS (st_im_text_parent_class)->dispose (object); g_clear_object (&priv->im_context); } static void st_im_text_cursor_event (ClutterText *self, const ClutterGeometry *geometry) { StIMTextPrivate *priv = ST_IM_TEXT (self)->priv; gfloat actor_x, actor_y; GdkRectangle area; clutter_actor_get_transformed_position (CLUTTER_ACTOR (self), &actor_x, &actor_y); area.x = (int)(0.5 + geometry->x + actor_x); area.y = (int)(0.5 + geometry->y + actor_y); area.width = geometry->width; area.height = geometry->height; gtk_im_context_set_cursor_location (priv->im_context, &area); if (CLUTTER_TEXT_CLASS (st_im_text_parent_class)->cursor_event) CLUTTER_TEXT_CLASS (st_im_text_parent_class)->cursor_event (self, geometry); } static void st_im_text_commit_cb (GtkIMContext *context, const gchar *str, StIMText *imtext) { ClutterText *clutter_text = CLUTTER_TEXT (imtext); if (clutter_text_get_editable (clutter_text)) { clutter_text_delete_selection (clutter_text); clutter_text_insert_text (clutter_text, str, clutter_text_get_cursor_position (clutter_text)); } } static void st_im_text_preedit_changed_cb (GtkIMContext *context, StIMText *imtext) { ClutterText *clutter_text = CLUTTER_TEXT (imtext); gchar *preedit_str = NULL; PangoAttrList *preedit_attrs = NULL; gint cursor_pos = 0; gtk_im_context_get_preedit_string (context, &preedit_str, &preedit_attrs, &cursor_pos); clutter_text_set_preedit_string (clutter_text, preedit_str, preedit_attrs, cursor_pos); imtext->priv->has_preedit = preedit_str && *preedit_str; g_free (preedit_str); pango_attr_list_unref (preedit_attrs); } static gboolean st_im_text_retrieve_surrounding_cb (GtkIMContext *context, StIMText *imtext) { ClutterText *clutter_text = CLUTTER_TEXT (imtext); ClutterTextBuffer *buffer; const gchar *text; gint cursor_pos; buffer = clutter_text_get_buffer (clutter_text); text = clutter_text_buffer_get_text (buffer); cursor_pos = clutter_text_get_cursor_position (clutter_text); if (cursor_pos < 0) cursor_pos = clutter_text_buffer_get_length (buffer); gtk_im_context_set_surrounding (context, text, /* length and cursor_index are in bytes */ clutter_text_buffer_get_bytes (buffer), g_utf8_offset_to_pointer (text, cursor_pos) - text); return TRUE; } static gboolean st_im_text_delete_surrounding_cb (GtkIMContext *context, gint offset, gint n_chars, StIMText *imtext) { ClutterText *clutter_text = CLUTTER_TEXT (imtext); if (clutter_text_get_editable (clutter_text)) { gint cursor_pos = clutter_text_get_cursor_position (clutter_text); clutter_text_delete_text (clutter_text, cursor_pos + offset, cursor_pos + offset + n_chars); } return TRUE; } static void reset_im_context (StIMText *self) { StIMTextPrivate *priv = self->priv; if (priv->need_im_reset) { gtk_im_context_reset (priv->im_context); priv->need_im_reset = FALSE; } } static gboolean st_im_text_get_paint_volume (ClutterActor *self, ClutterPaintVolume *volume) { return clutter_paint_volume_set_from_allocation (volume, self); } static GdkWindow *event_window; void st_im_text_set_event_window (GdkWindow *window) { g_assert (event_window == NULL); event_window = window; } static void st_im_text_realize (ClutterActor *actor) { StIMTextPrivate *priv = ST_IM_TEXT (actor)->priv; g_assert (event_window != NULL); priv->window = g_object_ref (event_window); gtk_im_context_set_client_window (priv->im_context, priv->window); } static void st_im_text_unrealize (ClutterActor *actor) { StIMText *self = ST_IM_TEXT (actor); StIMTextPrivate *priv = self->priv; reset_im_context (self); gtk_im_context_set_client_window (priv->im_context, NULL); g_object_unref (priv->window); priv->window = NULL; } static gboolean key_is_modifier (guint16 keyval) { /* See gdkkeys-x11.c:_gdk_keymap_key_is_modifier() for how this * really should be implemented */ switch (keyval) { case GDK_KEY_Shift_L: case GDK_KEY_Shift_R: case GDK_KEY_Control_L: case GDK_KEY_Control_R: case GDK_KEY_Caps_Lock: case GDK_KEY_Shift_Lock: case GDK_KEY_Meta_L: case GDK_KEY_Meta_R: case GDK_KEY_Alt_L: case GDK_KEY_Alt_R: case GDK_KEY_Super_L: case GDK_KEY_Super_R: case GDK_KEY_Hyper_L: case GDK_KEY_Hyper_R: case GDK_KEY_ISO_Lock: case GDK_KEY_ISO_Level2_Latch: case GDK_KEY_ISO_Level3_Shift: case GDK_KEY_ISO_Level3_Latch: case GDK_KEY_ISO_Level3_Lock: case GDK_KEY_ISO_Level5_Shift: case GDK_KEY_ISO_Level5_Latch: case GDK_KEY_ISO_Level5_Lock: case GDK_KEY_ISO_Group_Shift: case GDK_KEY_ISO_Group_Latch: case GDK_KEY_ISO_Group_Lock: return TRUE; default: return FALSE; } } static GdkEventKey * key_event_to_gdk (ClutterKeyEvent *event_clutter) { GdkEventKey *event_gdk; event_gdk = (GdkEventKey *)gdk_event_new ((event_clutter->type == CLUTTER_KEY_PRESS) ? GDK_KEY_PRESS : GDK_KEY_RELEASE); g_assert (event_window != NULL); event_gdk->window = g_object_ref (event_window); event_gdk->send_event = FALSE; event_gdk->time = event_clutter->time; /* This depends on ClutterModifierType and GdkModifierType being * identical, which they are currently. (They both match the X * modifier state in the low 16-bits and have the same extensions.) */ event_gdk->state = event_clutter->modifier_state; event_gdk->keyval = event_clutter->keyval; event_gdk->hardware_keycode = event_clutter->hardware_keycode; /* For non-proper non-XKB support, we'd need a huge cut-and-paste * from gdkkeys-x11.c; this is a macro that just shifts a few bits * out of state, so won't make the situation worse if the server * doesn't support XKB; we'll just end up with group == 0 */ event_gdk->group = XkbGroupForCoreState (event_gdk->state); if (event_clutter->unicode_value) { /* This is not particularly close to what GDK does - event_gdk->string * is supposed to be in the locale encoding, and have control keys * as control characters, etc. See gdkevents-x11.c:translate_key_event(). * Hopefully no input method is using event.string. */ char buf[6]; event_gdk->length = g_unichar_to_utf8 (event_clutter->unicode_value, buf); event_gdk->string = g_strndup (buf, event_gdk->length); } event_gdk->is_modifier = key_is_modifier (event_gdk->keyval); return event_gdk; } static gboolean st_im_text_button_press_event (ClutterActor *actor, ClutterButtonEvent *event) { /* The button press indicates the user moving the cursor, or selecting * etc, so we should abort any current preedit operation. ClutterText * treats all buttons identically, so so do we. */ reset_im_context (ST_IM_TEXT (actor)); if (CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->button_press_event) return CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->button_press_event (actor, event); else return FALSE; } static gboolean st_im_text_captured_event (ClutterActor *actor, ClutterEvent *event) { StIMText *self = ST_IM_TEXT (actor); StIMTextPrivate *priv = self->priv; ClutterText *clutter_text = CLUTTER_TEXT (actor); ClutterEventType type = clutter_event_type (event); gboolean result = FALSE; int old_position; if (type != CLUTTER_KEY_PRESS && type != CLUTTER_KEY_RELEASE) return FALSE; if (clutter_text_get_editable (clutter_text)) { GdkEventKey *event_gdk = key_event_to_gdk ((ClutterKeyEvent *)event); if (gtk_im_context_filter_keypress (priv->im_context, event_gdk)) { priv->need_im_reset = TRUE; result = TRUE; } gdk_event_free ((GdkEvent *)event_gdk); } old_position = clutter_text_get_cursor_position (clutter_text); if (!result && CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->captured_event) result = CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->captured_event (actor, event); if (type == CLUTTER_KEY_PRESS && clutter_text_get_cursor_position (clutter_text) != old_position) reset_im_context (self); return result; } static void st_im_text_key_focus_in (ClutterActor *actor) { StIMTextPrivate *priv = ST_IM_TEXT (actor)->priv; ClutterText *clutter_text = CLUTTER_TEXT (actor); if (clutter_text_get_editable (clutter_text)) { priv->need_im_reset = TRUE; gtk_im_context_focus_in (priv->im_context); } if (CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->key_focus_in) CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->key_focus_in (actor); } static void st_im_text_key_focus_out (ClutterActor *actor) { StIMTextPrivate *priv = ST_IM_TEXT (actor)->priv; ClutterText *clutter_text = CLUTTER_TEXT (actor); if (clutter_text_get_editable (clutter_text)) { priv->need_im_reset = TRUE; gtk_im_context_focus_out (priv->im_context); } if (CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->key_focus_out) CLUTTER_ACTOR_CLASS (st_im_text_parent_class)->key_focus_out (actor); } static void st_im_text_set_property (GObject *gobject, guint prop_id, const GValue *value, GParamSpec *pspec) { StIMText *imtext = ST_IM_TEXT (gobject); switch (prop_id) { case PROP_INPUT_PURPOSE: st_im_text_set_input_purpose (imtext, g_value_get_enum (value)); break; case PROP_INPUT_HINTS: st_im_text_set_input_hints (imtext, g_value_get_flags (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); break; } } static void st_im_text_get_property (GObject *gobject, guint prop_id, GValue *value, GParamSpec *pspec) { StIMText *imtext = ST_IM_TEXT (gobject); switch (prop_id) { case PROP_INPUT_PURPOSE: g_value_set_enum (value, st_im_text_get_input_purpose (imtext)); break; case PROP_INPUT_HINTS: g_value_set_flags (value, st_im_text_get_input_hints (imtext)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); break; } } static void st_im_text_class_init (StIMTextClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); ClutterActorClass *actor_class = CLUTTER_ACTOR_CLASS (klass); ClutterTextClass *text_class = CLUTTER_TEXT_CLASS (klass); GParamSpec *pspec; object_class->dispose = st_im_text_dispose; object_class->set_property = st_im_text_set_property; object_class->get_property = st_im_text_get_property; actor_class->get_paint_volume = st_im_text_get_paint_volume; actor_class->realize = st_im_text_realize; actor_class->unrealize = st_im_text_unrealize; actor_class->button_press_event = st_im_text_button_press_event; actor_class->captured_event = st_im_text_captured_event; actor_class->key_focus_in = st_im_text_key_focus_in; actor_class->key_focus_out = st_im_text_key_focus_out; text_class->cursor_event = st_im_text_cursor_event; pspec = g_param_spec_enum ("input-purpose", "Purpose", "Purpose of the text field", GTK_TYPE_INPUT_PURPOSE, GTK_INPUT_PURPOSE_FREE_FORM, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_property (object_class, PROP_INPUT_PURPOSE, pspec); pspec = g_param_spec_flags ("input-hints", "hints", "Hints for the text field behaviour", GTK_TYPE_INPUT_HINTS, GTK_INPUT_HINT_NONE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_property (object_class, PROP_INPUT_HINTS, pspec); } static void st_im_text_init (StIMText *self) { StIMTextPrivate *priv; self->priv = priv = st_im_text_get_instance_private (self); priv->im_context = gtk_im_multicontext_new (); g_signal_connect (priv->im_context, "commit", G_CALLBACK (st_im_text_commit_cb), self); g_signal_connect (priv->im_context, "preedit-changed", G_CALLBACK (st_im_text_preedit_changed_cb), self); g_signal_connect (priv->im_context, "retrieve-surrounding", G_CALLBACK (st_im_text_retrieve_surrounding_cb), self); g_signal_connect (priv->im_context, "delete-surrounding", G_CALLBACK (st_im_text_delete_surrounding_cb), self); } /** * st_im_text_new: * @text: text to set to * * Create a new #StIMText with the specified text * * Returns: a new #ClutterActor */ ClutterActor * st_im_text_new (const gchar *text) { return g_object_new (ST_TYPE_IM_TEXT, "text", text, NULL); } /** * st_im_text_set_input_purpose: * @imtext: a #StIMText * @purpose: the purpose * * Sets the #StIMText:input-purpose property which * can be used by on-screen keyboards and other input * methods to adjust their behaviour. */ void st_im_text_set_input_purpose (StIMText *imtext, GtkInputPurpose purpose) { g_return_if_fail (ST_IS_IM_TEXT (imtext)); if (st_im_text_get_input_purpose (imtext) != purpose) { g_object_set (G_OBJECT (imtext->priv->im_context), "input-purpose", purpose, NULL); g_object_get (G_OBJECT (imtext->priv->im_context), "input-purpose", &purpose, NULL); g_object_notify (G_OBJECT (imtext), "input-purpose"); } } /** * st_im_text_get_input_purpose: * @imtext: a #StIMText * * Gets the value of the #StIMText:input-purpose property. */ GtkInputPurpose st_im_text_get_input_purpose (StIMText *imtext) { GtkInputPurpose purpose; g_return_val_if_fail (ST_IS_IM_TEXT (imtext), GTK_INPUT_PURPOSE_FREE_FORM); g_object_get (G_OBJECT (imtext->priv->im_context), "input-purpose", &purpose, NULL); return purpose; } /** * st_im_text_set_input_hints: * @imtext: a #StIMText * @hints: the hints * * Sets the #StIMText:input-hints property, which * allows input methods to fine-tune their behaviour. */ void st_im_text_set_input_hints (StIMText *imtext, GtkInputHints hints) { g_return_if_fail (ST_IS_IM_TEXT (imtext)); if (st_im_text_get_input_hints (imtext) != hints) { g_object_set (G_OBJECT (imtext->priv->im_context), "input-hints", hints, NULL); g_object_notify (G_OBJECT (imtext), "input-hints"); } } /** * st_im_text_get_input_hints: * @imtext: a #StIMText * * Gets the value of the #StIMText:input-hints property. */ GtkInputHints st_im_text_get_input_hints (StIMText *imtext) { GtkInputHints hints; g_return_val_if_fail (ST_IS_IM_TEXT (imtext), GTK_INPUT_HINT_NONE); g_object_get (G_OBJECT (imtext->priv->im_context), "input-hints", &hints, NULL); return hints; } /** * st_im_text_has_preedit: * @imtext: a #StIMText * * Returns: wether @imtext currently has preedit text */ gboolean st_im_text_has_preedit (StIMText *imtext) { g_return_val_if_fail (ST_IS_IM_TEXT (imtext), FALSE); return imtext->priv->has_preedit; }