summaryrefslogtreecommitdiffstats
path: root/src/ibushotkey.c
diff options
context:
space:
mode:
authorHuang Peng <shawn.p.huang@gmail.com>2009-02-05 10:39:56 +0800
committerHuang Peng <shawn.p.huang@gmail.com>2009-02-05 10:39:56 +0800
commitaedad1ea0a7fef604aa27f4b58433fd8f2ece29e (patch)
treeffcb531d8474bde18b90341bcd4eb639edd74525 /src/ibushotkey.c
parent41ad46305a88637dd99f00a2d2a3f455505d357b (diff)
downloadibus-aedad1ea0a7fef604aa27f4b58433fd8f2ece29e.tar.gz
ibus-aedad1ea0a7fef604aa27f4b58433fd8f2ece29e.tar.xz
ibus-aedad1ea0a7fef604aa27f4b58433fd8f2ece29e.zip
re-implement ibus in c language.
Diffstat (limited to 'src/ibushotkey.c')
-rw-r--r--src/ibushotkey.c479
1 files changed, 479 insertions, 0 deletions
diff --git a/src/ibushotkey.c b/src/ibushotkey.c
new file mode 100644
index 0000000..16aad70
--- /dev/null
+++ b/src/ibushotkey.c
@@ -0,0 +1,479 @@
+/* vim:set et sts=4: */
+/* IBus - The Input Bus
+ * Copyright (C) 2008-2009 Huang Peng <shawn.p.huang@gmail.com>
+ *
+ * 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, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+#include "ibushotkey.h"
+#include "ibuskeysyms.h"
+#include "ibusinternal.h"
+#include "ibusshare.h"
+
+#define IBUS_HOTKEY_PROFILE_GET_PRIVATE(o) \
+ (G_TYPE_INSTANCE_GET_PRIVATE ((o), IBUS_TYPE_HOTKEY_PROFILE, IBusHotkeyProfilePrivate))
+
+enum {
+ TRIGGER,
+ LAST_SIGNAL,
+};
+
+typedef struct _IBusHotkey IBusHotkey;
+typedef struct _IBusHotkeyEvent IBusHotkeyEvent;
+typedef struct _IBusHotkeyProfilePrivate IBusHotkeyProfilePrivate;
+
+struct _IBusHotkey {
+ guint keyval;
+ guint modifiers;
+};
+
+struct _IBusHotkeyEvent {
+ GQuark event;
+ GList *hotkeys;
+};
+
+struct _IBusHotkeyProfilePrivate {
+ GTree *hotkeys;
+ GArray *events;
+ guint mask;
+};
+
+
+
+/* functions prototype */
+static IBusHotkey *ibus_hotkey_new (guint keyval,
+ guint modifiers);
+static IBusHotkey *ibus_hotkey_copy (const IBusHotkey *src);
+static void ibus_hotkey_free (IBusHotkey *hotkey);
+static gboolean ibus_hotkey_serialize (IBusHotkey *hotkey,
+ IBusMessageIter *iter);
+static gboolean ibus_hotkey_deserialize (IBusHotkey *hotkey,
+ IBusMessageIter *iter);
+
+static void ibus_hotkey_profile_class_init (IBusHotkeyProfileClass *klass);
+static void ibus_hotkey_profile_init (IBusHotkeyProfile *profile);
+static void ibus_hotkey_profile_destroy (IBusHotkeyProfile *profile);
+static gboolean ibus_hotkey_profile_serialize (IBusHotkeyProfile *profile,
+ IBusMessageIter *iter);
+static gboolean ibus_hotkey_profile_deserialize(IBusHotkeyProfile *profile,
+ IBusMessageIter *iter);
+static gboolean ibus_hotkey_profile_copy (IBusHotkeyProfile *dest,
+ const IBusHotkeyProfile*src);
+static void ibus_hotkey_profile_trigger (IBusHotkeyProfile *profile,
+ GQuark event);
+
+static IBusSerializableClass *parent_class = NULL;
+
+static guint profile_signals[LAST_SIGNAL] = { 0 };
+
+GType
+ibus_hotkey_get_type (void)
+{
+ static GType type = 0;
+
+ if (type == 0) {
+ type = g_boxed_type_register_static ("IBusHotkey",
+ (GBoxedCopyFunc) ibus_hotkey_copy,
+ (GBoxedFreeFunc) ibus_hotkey_free);
+ }
+
+ return type;
+}
+
+static gboolean
+ibus_hotkey_serialize (IBusHotkey *hotkey,
+ IBusMessageIter *iter)
+{
+ gboolean retval;
+
+ retval = ibus_message_iter_append (iter, G_TYPE_UINT, &hotkey->keyval);
+ g_return_val_if_fail (retval, FALSE);
+
+ retval = ibus_message_iter_append (iter, G_TYPE_UINT, &hotkey->modifiers);
+ g_return_val_if_fail (retval, FALSE);
+
+ return TRUE;
+}
+
+static gboolean
+ibus_hotkey_deserialize (IBusHotkey *hotkey,
+ IBusMessageIter *iter)
+{
+ gboolean retval;
+
+ retval = ibus_message_iter_get (iter, G_TYPE_UINT, &hotkey->keyval);
+ g_return_val_if_fail (retval, FALSE);
+
+ retval = ibus_message_iter_get (iter, G_TYPE_UINT, &hotkey->modifiers);
+ g_return_val_if_fail (retval, FALSE);
+
+ return TRUE;
+}
+
+
+static IBusHotkey *
+ibus_hotkey_new (guint keyval,
+ guint modifiers)
+{
+ IBusHotkey *hotkey = g_slice_new (IBusHotkey);
+
+ hotkey->keyval = keyval;
+ hotkey->modifiers = modifiers;
+
+ return hotkey;
+}
+
+static void
+ibus_hotkey_free (IBusHotkey *hotkey)
+{
+ g_slice_free (IBusHotkey, hotkey);
+}
+
+
+static IBusHotkey *
+ibus_hotkey_copy (const IBusHotkey *src)
+{
+ return ibus_hotkey_new (src->keyval, src->modifiers);
+}
+
+static gint
+ibus_hotkey_cmp_with_data (IBusHotkey *hotkey1,
+ IBusHotkey *hotkey2,
+ gpointer user_data)
+{
+ gint retval;
+
+ if (hotkey1 == hotkey2)
+ return 0;
+
+ retval = hotkey1->keyval - hotkey2->keyval;
+ if (retval == 0)
+ retval = hotkey1->modifiers - hotkey2->modifiers;
+
+ return retval;
+}
+
+
+
+GType
+ibus_hotkey_profile_get_type (void)
+{
+ static GType type = 0;
+
+ static const GTypeInfo type_info = {
+ sizeof (IBusHotkeyProfileClass),
+ (GBaseInitFunc) NULL,
+ (GBaseFinalizeFunc) NULL,
+ (GClassInitFunc) ibus_hotkey_profile_class_init,
+ NULL, /* class finialize */
+ NULL, /* class data */
+ sizeof (IBusHotkeyProfile),
+ 0,
+ (GInstanceInitFunc) ibus_hotkey_profile_init,
+ };
+
+ if (type == 0) {
+ type = g_type_register_static (IBUS_TYPE_SERIALIZABLE,
+ "IBusHotkeyProfile",
+ &type_info,
+ 0);
+ }
+
+ return type;
+}
+
+static void
+ibus_hotkey_profile_class_init (IBusHotkeyProfileClass *klass)
+{
+ IBusObjectClass *object_class = IBUS_OBJECT_CLASS (klass);
+ IBusSerializableClass *serializable_class = IBUS_SERIALIZABLE_CLASS (klass);
+
+ parent_class = (IBusSerializableClass *) g_type_class_peek_parent (klass);
+
+ g_type_class_add_private (klass, sizeof (IBusHotkeyProfilePrivate));
+
+ object_class->destroy = (IBusObjectDestroyFunc) ibus_hotkey_profile_destroy;
+
+ serializable_class->serialize = (IBusSerializableSerializeFunc) ibus_hotkey_profile_serialize;
+ serializable_class->deserialize = (IBusSerializableDeserializeFunc) ibus_hotkey_profile_deserialize;
+ serializable_class->copy = (IBusSerializableCopyFunc) ibus_hotkey_profile_copy;
+
+ klass->trigger = ibus_hotkey_profile_trigger;
+
+ g_string_append (serializable_class->signature, "av");
+
+ /* install signals */
+ profile_signals[TRIGGER] =
+ g_signal_new (I_("trigger"),
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST | G_SIGNAL_DETAILED,
+ G_STRUCT_OFFSET (IBusHotkeyProfileClass, trigger),
+ NULL, NULL,
+ ibus_marshal_VOID__UINT_POINTER,
+ G_TYPE_NONE,
+ 2,
+ G_TYPE_UINT,
+ G_TYPE_POINTER);
+}
+
+static void
+ibus_hotkey_profile_init (IBusHotkeyProfile *profile)
+{
+ IBusHotkeyProfilePrivate *priv;
+ priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile);
+
+ priv->hotkeys = g_tree_new_full ((GCompareDataFunc) ibus_hotkey_cmp_with_data,
+ NULL,
+ (GDestroyNotify) ibus_hotkey_free,
+ NULL);
+ priv->events = g_array_new (TRUE, TRUE, sizeof (IBusHotkeyEvent));
+
+ priv->mask = IBUS_SHIFT_MASK |
+ IBUS_CONTROL_MASK |
+ IBUS_MOD1_MASK |
+ IBUS_RELEASE_MASK;
+}
+
+static void
+ibus_hotkey_profile_destroy (IBusHotkeyProfile *profile)
+{
+ IBusHotkeyProfilePrivate *priv;
+ priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile);
+
+ if (priv->hotkeys) {
+ g_tree_destroy (priv->hotkeys);
+ priv->hotkeys = NULL;
+ }
+
+ if (priv->events) {
+ IBusHotkeyEvent *events, *p;
+ p = events = (IBusHotkeyEvent *)g_array_free (priv->events, FALSE);
+ priv->events = NULL;
+
+ while (p->event != 0) {
+ if (p->hotkeys)
+ g_list_free (p->hotkeys);
+ p ++;
+ }
+ g_free (events);
+ }
+ IBUS_OBJECT_CLASS (parent_class)->destroy ((IBusObject *)profile);
+}
+
+static gboolean
+ibus_hotkey_profile_serialize (IBusHotkeyProfile *profile,
+ IBusMessageIter *iter)
+{
+ gboolean retval;
+
+ retval = parent_class->serialize ((IBusSerializable *) profile, iter);
+ g_return_val_if_fail (retval, FALSE);
+
+ return TRUE;
+}
+
+static gboolean
+ibus_hotkey_profile_deserialize (IBusHotkeyProfile *profile,
+ IBusMessageIter *iter)
+{
+ gboolean retval;
+
+ retval = parent_class->deserialize ((IBusSerializable *) profile, iter);
+ g_return_val_if_fail (retval, FALSE);
+
+ return TRUE;
+}
+
+static gboolean
+ibus_hotkey_profile_copy (IBusHotkeyProfile *dest,
+ const IBusHotkeyProfile *src)
+{
+ gboolean retval;
+
+ retval = parent_class->copy ((IBusSerializable *)dest,
+ (IBusSerializable *)src);
+ g_return_val_if_fail (retval, FALSE);
+
+ g_return_val_if_fail (IBUS_IS_HOTKEY_PROFILE (dest), FALSE);
+ g_return_val_if_fail (IBUS_IS_HOTKEY_PROFILE (src), FALSE);
+
+ return TRUE;
+}
+
+IBusHotkeyProfile *
+ibus_hotkey_profile_new (void)
+{
+ IBusHotkeyProfile *profile = g_object_new (IBUS_TYPE_HOTKEY_PROFILE, 0);
+
+ return profile;
+}
+
+static void
+ibus_hotkey_profile_trigger (IBusHotkeyProfile *profile,
+ GQuark event)
+{
+ // g_debug ("%s is triggerred", g_quark_to_string (event));
+}
+
+gboolean
+ibus_hotkey_profile_add_hotkey (IBusHotkeyProfile *profile,
+ guint keyval,
+ guint modifiers,
+ GQuark event)
+{
+ IBusHotkeyProfilePrivate *priv;
+ priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile);
+
+ IBusHotkey *hotkey = ibus_hotkey_new (keyval, modifiers);
+
+ /* has the same hotkey in profile */
+ if (g_tree_lookup (priv->hotkeys, hotkey) != NULL) {
+ ibus_hotkey_free (hotkey);
+ g_return_val_if_reached (FALSE);
+ }
+
+ g_tree_insert (priv->hotkeys, hotkey, (gpointer)event);
+
+ gint i;
+ IBusHotkeyEvent *p = NULL;
+ for ( i = 0; i < priv->events->len; i++) {
+ p = &g_array_index (priv->events, IBusHotkeyEvent, i);
+ if (p->event == event)
+ break;
+ }
+
+ if (i >= priv->events->len) {
+ g_array_set_size (priv->events, i + 1);
+ p = &g_array_index (priv->events, IBusHotkeyEvent, i);
+ p->event = event;
+ }
+
+ p->hotkeys = g_list_append (p->hotkeys, hotkey);
+
+ return TRUE;
+}
+
+
+gboolean
+ibus_hotkey_profile_add_hotkey_from_string (IBusHotkeyProfile *profile,
+ const gchar *str,
+ GQuark event)
+{
+ guint keyval;
+ guint modifiers;
+
+ if (ibus_key_event_from_string (str, &keyval, &modifiers) == FALSE) {
+ return FALSE;
+ }
+
+ return ibus_hotkey_profile_add_hotkey (profile, keyval, modifiers, event);
+}
+
+gboolean
+ibus_hotkey_profile_remove_hotkey (IBusHotkeyProfile *profile,
+ guint keyval,
+ guint modifiers)
+{
+ IBusHotkeyProfilePrivate *priv;
+ priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile);
+
+ IBusHotkey hotkey = {
+ keyval : keyval,
+ modifiers : modifiers
+ };
+
+ IBusHotkey *p1;
+ GQuark event;
+ gboolean retval;
+
+ retval = g_tree_lookup_extended (priv->hotkeys,
+ &hotkey,
+ (gpointer)&p1,
+ (gpointer)&event);
+
+ if (!retval)
+ return FALSE;
+
+ gint i;
+ IBusHotkeyEvent *p2 = NULL;
+ for ( i = 0; i < priv->events->len; i++) {
+ p2 = &g_array_index (priv->events, IBusHotkeyEvent, i);
+ if (p2->event == event)
+ break;
+ }
+
+ g_assert (p2->event == event);
+
+ p2->hotkeys = g_list_remove (p2->hotkeys, p1);
+ if (p2->hotkeys == NULL) {
+ g_array_remove_index_fast (priv->events, i);
+ }
+
+ g_tree_remove (priv->hotkeys, p1);
+
+ return TRUE;
+}
+
+gboolean
+ibus_hotkey_profile_remove_hotkey_by_event (IBusHotkeyProfile *profile,
+ GQuark event)
+{
+ IBusHotkeyProfilePrivate *priv;
+ priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile);
+
+ gint i;
+ IBusHotkeyEvent *p = NULL;
+ for ( i = 0; i < priv->events->len; i++) {
+ p = &g_array_index (priv->events, IBusHotkeyEvent, i);
+ if (p->event == event)
+ break;
+ }
+
+ if (p == NULL || p->event != event)
+ return FALSE;
+
+ GList *list;
+ for (list = p->hotkeys; list != NULL; list = list->next) {
+ g_tree_remove (priv->hotkeys, (IBusHotkey *)list->data);
+ }
+
+ g_list_free (p->hotkeys);
+ g_array_remove_index_fast (priv->events, i);
+
+ return TRUE;
+}
+
+GQuark
+ibus_hotkey_profile_filter_key_event (IBusHotkeyProfile *profile,
+ guint keyval,
+ guint modifiers,
+ gpointer user_data)
+{
+ IBusHotkeyProfilePrivate *priv;
+ priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile);
+
+ IBusHotkey hotkey = {
+ keyval : keyval,
+ modifiers : modifiers & priv->mask,
+ };
+
+ GQuark event = (GQuark) g_tree_lookup (priv->hotkeys, &hotkey);
+
+ if (event != 0) {
+ g_signal_emit (profile, profile_signals[TRIGGER], event, event, user_data);
+ }
+
+ return event;
+}