/* vim:set et sts=4: */ /* IBus - The Input Bus * Copyright (C) 2008-2009 Huang Peng * * 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, NULL); 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, (gpointer) hotkey, GUINT_TO_POINTER (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, guint prev_keyval, guint prev_modifiers, gpointer user_data) { IBusHotkeyProfilePrivate *priv; priv = IBUS_HOTKEY_PROFILE_GET_PRIVATE (profile); IBusHotkey hotkey = { .keyval = keyval, .modifiers = modifiers & priv->mask, }; if ((modifiers & IBUS_RELEASE_MASK) && keyval != prev_keyval) { return 0; } GQuark event = (GQuark) GPOINTER_TO_UINT (g_tree_lookup (priv->hotkeys, &hotkey)); if (event != 0) { g_signal_emit (profile, profile_signals[TRIGGER], event, event, user_data); } return event; }