Tune input context suspend/resume behavior.

This commit is contained in:
Daiki Ueno
2011-02-22 18:38:11 +09:00
parent d25114b370
commit 50e9f3186a
4 changed files with 41 additions and 113 deletions

View File

@ -415,3 +415,14 @@ eekboard_context_set_enabled (EekboardContext *context,
priv = EEKBOARD_CONTEXT_GET_PRIVATE (context); priv = EEKBOARD_CONTEXT_GET_PRIVATE (context);
priv->enabled = enabled; priv->enabled = enabled;
} }
gboolean
eekboard_context_is_enabled (EekboardContext *context)
{
EekboardContextPrivate *priv;
g_assert (EEKBOARD_IS_CONTEXT(context));
priv = EEKBOARD_CONTEXT_GET_PRIVATE (context);
return priv->enabled;
}

View File

@ -76,6 +76,7 @@ gboolean eekboard_context_is_keyboard_visible
(EekboardContext *context); (EekboardContext *context);
void eekboard_context_set_enabled (EekboardContext *context, void eekboard_context_set_enabled (EekboardContext *context,
gboolean enabled); gboolean enabled);
gboolean eekboard_context_is_enabled (EekboardContext *context);
G_END_DECLS G_END_DECLS
#endif /* EEKBOARD_CONTEXT_H */ #endif /* EEKBOARD_CONTEXT_H */

View File

@ -29,30 +29,18 @@ G_DEFINE_TYPE (EekboardServer, eekboard_server, G_TYPE_DBUS_PROXY);
struct _EekboardServerPrivate struct _EekboardServerPrivate
{ {
GHashTable *context_hash; GHashTable *context_hash;
GSList *context_stack;
/* used in eekboard_server_push_context and
eekboard_server_destroy_context as a callback data */
EekboardContext *context;
}; };
static void static void
eekboard_server_dispose (GObject *object) eekboard_server_dispose (GObject *object)
{ {
EekboardServerPrivate *priv = EEKBOARD_SERVER_GET_PRIVATE(object); EekboardServerPrivate *priv = EEKBOARD_SERVER_GET_PRIVATE(object);
GSList *head;
if (priv->context_hash) { if (priv->context_hash) {
g_hash_table_destroy (priv->context_hash); g_hash_table_destroy (priv->context_hash);
priv->context_hash = NULL; priv->context_hash = NULL;
} }
for (head = priv->context_stack; head; head = priv->context_stack) {
g_object_unref (head->data);
priv->context_stack = g_slist_next (head);
g_slist_free1 (head);
}
G_OBJECT_CLASS (eekboard_server_parent_class)->dispose (object); G_OBJECT_CLASS (eekboard_server_parent_class)->dispose (object);
} }
@ -78,8 +66,6 @@ eekboard_server_init (EekboardServer *self)
g_str_equal, g_str_equal,
(GDestroyNotify)g_free, (GDestroyNotify)g_free,
(GDestroyNotify)g_object_unref); (GDestroyNotify)g_object_unref);
priv->context_stack = NULL;
priv->context = NULL;
} }
EekboardServer * EekboardServer *
@ -141,33 +127,25 @@ eekboard_server_create_context (EekboardServer *server,
} }
priv = EEKBOARD_SERVER_GET_PRIVATE(server); priv = EEKBOARD_SERVER_GET_PRIVATE(server);
g_hash_table_insert (priv->context_hash, g_strdup (object_path), context); g_hash_table_insert (priv->context_hash,
g_strdup (object_path),
g_object_ref (context));
return context; return context;
} }
static void static void
push_context_async_ready_callback (GObject *source_object, server_async_ready_callback (GObject *source_object,
GAsyncResult *res, GAsyncResult *res,
gpointer user_data) gpointer user_data)
{ {
EekboardServer *server = user_data;
EekboardServerPrivate *priv = EEKBOARD_SERVER_GET_PRIVATE(server);
GError *error = NULL; GError *error = NULL;
GVariant *result; GVariant *result;
result = g_dbus_proxy_call_finish (G_DBUS_PROXY(source_object), result = g_dbus_proxy_call_finish (G_DBUS_PROXY(source_object),
res, res,
&error); &error);
if (result) { if (result)
g_variant_unref (result); g_variant_unref (result);
if (priv->context_stack)
eekboard_context_set_enabled (priv->context_stack->data, FALSE);
priv->context_stack = g_slist_prepend (priv->context_stack,
priv->context);
g_object_ref (priv->context);
eekboard_context_set_enabled (priv->context, TRUE);
}
} }
void void
@ -188,43 +166,15 @@ eekboard_server_push_context (EekboardServer *server,
if (!context) if (!context)
return; return;
priv->context = context; eekboard_context_set_enabled (context, TRUE);
g_dbus_proxy_call (G_DBUS_PROXY(server), g_dbus_proxy_call (G_DBUS_PROXY(server),
"PushContext", "PushContext",
g_variant_new ("(s)", object_path), g_variant_new ("(s)", object_path),
G_DBUS_CALL_FLAGS_NONE, G_DBUS_CALL_FLAGS_NONE,
-1, -1,
cancellable, cancellable,
push_context_async_ready_callback, server_async_ready_callback,
server); NULL);
}
static void
pop_context_async_ready_callback (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
EekboardServer *server = user_data;
EekboardServerPrivate *priv = EEKBOARD_SERVER_GET_PRIVATE(server);
GError *error = NULL;
GVariant *result;
result = g_dbus_proxy_call_finish (G_DBUS_PROXY(source_object),
res,
&error);
if (result) {
g_variant_unref (result);
if (priv->context_stack) {
EekboardContext *context = priv->context_stack->data;
eekboard_context_set_enabled (context, FALSE);
priv->context_stack = g_slist_next (priv->context_stack);
g_object_unref (context);
if (priv->context_stack)
eekboard_context_set_enabled (priv->context_stack->data, TRUE);
}
}
} }
void void
@ -239,39 +189,8 @@ eekboard_server_pop_context (EekboardServer *server,
G_DBUS_CALL_FLAGS_NONE, G_DBUS_CALL_FLAGS_NONE,
-1, -1,
cancellable, cancellable,
pop_context_async_ready_callback, server_async_ready_callback,
server); NULL);
}
static void
destroy_context_async_ready_callback (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
EekboardServer *server = user_data;
EekboardServerPrivate *priv = EEKBOARD_SERVER_GET_PRIVATE(server);
GError *error = NULL;
GVariant *result;
const gchar *object_path;
GSList *head;
result = g_dbus_proxy_call_finish (G_DBUS_PROXY(source_object),
res,
&error);
if (result) {
g_variant_unref (result);
head = g_slist_find (priv->context_stack, priv->context);
if (head) {
priv->context_stack = g_slist_remove_link (priv->context_stack,
head);
g_slist_free1 (head);
}
object_path =
g_dbus_proxy_get_object_path (G_DBUS_PROXY(priv->context));
g_hash_table_remove (priv->context_hash, object_path);
}
} }
void void
@ -285,16 +204,17 @@ eekboard_server_destroy_context (EekboardServer *server,
g_return_if_fail (EEKBOARD_IS_SERVER(server)); g_return_if_fail (EEKBOARD_IS_SERVER(server));
g_return_if_fail (EEKBOARD_IS_CONTEXT(context)); g_return_if_fail (EEKBOARD_IS_CONTEXT(context));
object_path = g_dbus_proxy_get_object_path (G_DBUS_PROXY(context));
priv = EEKBOARD_SERVER_GET_PRIVATE(server); priv = EEKBOARD_SERVER_GET_PRIVATE(server);
priv->context = context;
object_path = g_dbus_proxy_get_object_path (G_DBUS_PROXY(context));
g_hash_table_remove (priv->context_hash, object_path);
g_dbus_proxy_call (G_DBUS_PROXY(server), g_dbus_proxy_call (G_DBUS_PROXY(server),
"DestroyContext", "DestroyContext",
g_variant_new ("(s)", object_path), g_variant_new ("(s)", object_path),
G_DBUS_CALL_FLAGS_NONE, G_DBUS_CALL_FLAGS_NONE,
-1, -1,
cancellable, cancellable,
destroy_context_async_ready_callback, server_async_ready_callback,
server); NULL);
} }

View File

@ -84,6 +84,7 @@ struct _ServerContext {
char *object_path; char *object_path;
gboolean enabled; gboolean enabled;
gboolean last_keyboard_visible;
GtkWidget *window; GtkWidget *window;
GtkWidget *widget; GtkWidget *widget;
@ -347,6 +348,9 @@ server_context_init (ServerContext *context)
context->registration_id = 0; context->registration_id = 0;
context->object_path = NULL; context->object_path = NULL;
context->enabled = FALSE;
context->last_keyboard_visible = FALSE;
context->keyboard = NULL; context->keyboard = NULL;
context->widget = NULL; context->widget = NULL;
context->window = NULL; context->window = NULL;
@ -429,11 +433,6 @@ handle_method_call (GDBusConnection *connection,
EekSerializable *serializable; EekSerializable *serializable;
GVariant *variant; GVariant *variant;
if (!context->enabled) {
g_dbus_method_invocation_return_value (invocation, NULL);
return;
}
g_variant_get (parameters, "(v)", &variant); g_variant_get (parameters, "(v)", &variant);
serializable = eek_serializable_deserialize (variant); serializable = eek_serializable_deserialize (variant);
if (!EEK_IS_KEYBOARD(serializable)) { if (!EEK_IS_KEYBOARD(serializable)) {
@ -471,11 +470,6 @@ handle_method_call (GDBusConnection *connection,
if (g_strcmp0 (method_name, "SetGroup") == 0) { if (g_strcmp0 (method_name, "SetGroup") == 0) {
gint group; gint group;
if (!context->enabled) {
g_dbus_method_invocation_return_value (invocation, NULL);
return;
}
if (!context->keyboard) { if (!context->keyboard) {
g_dbus_method_invocation_return_error (invocation, g_dbus_method_invocation_return_error (invocation,
G_IO_ERROR, G_IO_ERROR,
@ -527,11 +521,6 @@ handle_method_call (GDBusConnection *connection,
EekKey *key; EekKey *key;
guint keycode; guint keycode;
if (!context->enabled) {
g_dbus_method_invocation_return_value (invocation, NULL);
return;
}
if (!context->keyboard) { if (!context->keyboard) {
g_dbus_method_invocation_return_error (invocation, g_dbus_method_invocation_return_error (invocation,
G_IO_ERROR, G_IO_ERROR,
@ -604,6 +593,8 @@ server_context_set_enabled (ServerContext *context, gboolean enabled)
NULL, NULL,
&error); &error);
g_assert_no_error (error); g_assert_no_error (error);
if (context->last_keyboard_visible && context->window)
gtk_widget_show_all (context->window);
} else { } else {
error = NULL; error = NULL;
g_dbus_connection_emit_signal (context->connection, g_dbus_connection_emit_signal (context->connection,
@ -614,6 +605,11 @@ server_context_set_enabled (ServerContext *context, gboolean enabled)
NULL, NULL,
&error); &error);
g_assert_no_error (error); g_assert_no_error (error);
if (context->window) {
context->last_keyboard_visible =
gtk_widget_get_visible (context->window);
gtk_widget_hide (context->window);
}
} }
context->enabled = enabled; context->enabled = enabled;
} }