321 lines
9.3 KiB
C
321 lines
9.3 KiB
C
/*
|
|
* Copyright (C) 2010-2011 Daiki Ueno <ueno@unixuser.org>
|
|
* Copyright (C) 2010-2011 Red Hat, Inc.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif /* HAVE_CONFIG_H */
|
|
|
|
#include <stdlib.h>
|
|
#ifdef HAVE_ATSPI
|
|
#include <dbus/dbus.h>
|
|
#include <atspi/atspi.h>
|
|
#endif /* HAVE_ATSPI */
|
|
#ifdef HAVE_IBUS
|
|
#include <ibus.h>
|
|
#endif /* HAVE_IBUS */
|
|
#include <gtk/gtk.h>
|
|
#include <glib/gi18n.h>
|
|
#include "eekboard/eekboard-client.h"
|
|
#include "client.h"
|
|
|
|
static gboolean opt_system = FALSE;
|
|
static gboolean opt_session = FALSE;
|
|
static gchar *opt_address = NULL;
|
|
|
|
static gboolean opt_focus = FALSE;
|
|
#ifdef HAVE_ATSPI
|
|
static gboolean opt_keystroke = FALSE;
|
|
#endif /* HAVE_ATSPI */
|
|
|
|
static gchar *opt_keyboards = NULL;
|
|
|
|
static gboolean opt_fullscreen = FALSE;
|
|
|
|
static const GOptionEntry options[] = {
|
|
{"system", 'y', 0, G_OPTION_ARG_NONE, &opt_system,
|
|
N_("Connect to the system bus")},
|
|
{"session", 'e', 0, G_OPTION_ARG_NONE, &opt_session,
|
|
N_("Connect to the session bus")},
|
|
{"address", 'a', 0, G_OPTION_ARG_STRING, &opt_address,
|
|
N_("Connect to the given D-Bus address")},
|
|
#if ENABLE_FOCUS_LISTENER
|
|
{"listen-focus", 'f', 0, G_OPTION_ARG_NONE, &opt_focus,
|
|
N_("Listen focus change events")},
|
|
#endif /* ENABLE_FOCUS_LISTENER */
|
|
#ifdef HAVE_ATSPI
|
|
{"listen-keystroke", 's', 0, G_OPTION_ARG_NONE, &opt_keystroke,
|
|
N_("Listen keystroke events with AT-SPI")},
|
|
#endif /* HAVE_ATSPI */
|
|
{"keyboards", 'k', 0, G_OPTION_ARG_STRING, &opt_keyboards,
|
|
N_("Specify keyboards (comma separated)")},
|
|
{"fullscreen", 'F', 0, G_OPTION_ARG_NONE, &opt_fullscreen,
|
|
N_("Create window in fullscreen mode")},
|
|
{NULL}
|
|
};
|
|
|
|
static void
|
|
on_context_destroyed (EekboardContext *context,
|
|
gpointer user_data)
|
|
{
|
|
gtk_main_quit ();
|
|
}
|
|
|
|
static void
|
|
on_destroyed (EekboardClient *eekboard,
|
|
gpointer user_data)
|
|
{
|
|
gtk_main_quit ();
|
|
}
|
|
|
|
enum FocusListenerType {
|
|
FOCUS_NONE,
|
|
FOCUS_ATSPI,
|
|
FOCUS_IBUS
|
|
};
|
|
|
|
static gboolean
|
|
set_keyboards (Client *client,
|
|
const gchar * const *keyboards)
|
|
{
|
|
if (g_strv_length ((gchar **)keyboards) == 0) {
|
|
if (!client_enable_xkl (client)) {
|
|
g_printerr ("Can't register xklavier event listeners\n");
|
|
return FALSE;
|
|
}
|
|
} else {
|
|
if (!client_set_keyboards (client, keyboards)) {
|
|
gchar *str = g_strjoinv (", ", (gchar **)keyboards);
|
|
g_printerr ("Can't set keyboards \"%s\"\n", str);
|
|
g_free (str);
|
|
return FALSE;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
Client *client = NULL;
|
|
EekboardClient *eekboard;
|
|
EekboardContext *context;
|
|
GBusType bus_type;
|
|
GDBusConnection *connection;
|
|
GError *error;
|
|
GOptionContext *option_context;
|
|
gint focus;
|
|
GSettings *settings = NULL;
|
|
gchar **keyboards = NULL;
|
|
gint retval = 0;
|
|
|
|
if (!gtk_init_check (&argc, &argv)) {
|
|
g_printerr ("Can't init GTK\n");
|
|
exit (1);
|
|
}
|
|
|
|
eek_init ();
|
|
|
|
option_context = g_option_context_new ("eekboard-desktop-client");
|
|
g_option_context_add_main_entries (option_context, options, NULL);
|
|
g_option_context_parse (option_context, &argc, &argv, NULL);
|
|
g_option_context_free (option_context);
|
|
|
|
if (opt_system)
|
|
bus_type = G_BUS_TYPE_SYSTEM;
|
|
else if (opt_address)
|
|
bus_type = G_BUS_TYPE_NONE;
|
|
else
|
|
bus_type = G_BUS_TYPE_SESSION;
|
|
|
|
switch (bus_type) {
|
|
case G_BUS_TYPE_SYSTEM:
|
|
error = NULL;
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SYSTEM, NULL, &error);
|
|
if (connection == NULL) {
|
|
g_printerr ("Can't connect to the system bus: %s\n",
|
|
error->message);
|
|
g_error_free (error);
|
|
exit (1);
|
|
}
|
|
break;
|
|
case G_BUS_TYPE_SESSION:
|
|
error = NULL;
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
if (connection == NULL) {
|
|
g_printerr ("Can't connect to the session bus: %s\n",
|
|
error->message);
|
|
g_error_free (error);
|
|
exit (1);
|
|
}
|
|
break;
|
|
case G_BUS_TYPE_NONE:
|
|
error = NULL;
|
|
connection = g_dbus_connection_new_for_address_sync (opt_address,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
&error);
|
|
if (connection == NULL) {
|
|
g_printerr ("Can't connect to the bus at %s: %s\n",
|
|
opt_address,
|
|
error->message);
|
|
g_error_free (error);
|
|
exit (1);
|
|
}
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
break;
|
|
}
|
|
|
|
client = client_new (connection);
|
|
g_object_unref (connection);
|
|
|
|
if (client == NULL) {
|
|
g_printerr ("Can't create a client\n");
|
|
exit (1);
|
|
}
|
|
|
|
settings = g_settings_new ("org.fedorahosted.eekboard");
|
|
focus = FOCUS_NONE;
|
|
if (opt_focus) {
|
|
gchar *focus_listener = g_settings_get_string (settings,
|
|
"focus-listener");
|
|
const struct {
|
|
const gchar *name;
|
|
enum FocusListenerType type;
|
|
} focus_listeners[] = {
|
|
#ifdef HAVE_ATSPI
|
|
{ "atspi", FOCUS_ATSPI },
|
|
#endif
|
|
#ifdef HAVE_IBUS
|
|
{ "ibus", FOCUS_IBUS },
|
|
#endif
|
|
{ NULL }
|
|
};
|
|
gint i;
|
|
|
|
focus = FOCUS_NONE;
|
|
for (i = 0; focus_listeners[i].name; i++) {
|
|
if (g_strcmp0 (focus_listener, focus_listeners[i].name) == 0)
|
|
focus = focus_listeners[i].type;
|
|
}
|
|
if (focus == FOCUS_NONE) {
|
|
g_printerr ("Unknown focus listener \"%s\". "
|
|
"Try \"atspi\" or \"ibus\"\n", focus_listener);
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
#ifdef HAVE_ATSPI
|
|
if (focus == FOCUS_ATSPI || opt_keystroke) {
|
|
GSettings *desktop_settings =
|
|
g_settings_new ("org.gnome.desktop.interface");
|
|
gboolean accessibility_enabled =
|
|
g_settings_get_boolean (desktop_settings, "toolkit-accessibility");
|
|
g_object_unref (desktop_settings);
|
|
|
|
if (accessibility_enabled) {
|
|
if (atspi_init () != 0) {
|
|
g_printerr ("Can't init AT-SPI 2\n");
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
|
|
if (focus == FOCUS_ATSPI &&
|
|
!client_enable_atspi_focus (client)) {
|
|
g_printerr ("Can't register AT-SPI focus change event listeners\n");
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
|
|
if (opt_keystroke &&
|
|
!client_enable_atspi_keystroke (client)) {
|
|
g_printerr ("Can't register AT-SPI keystroke event listeners\n");
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
} else {
|
|
g_printerr ("Desktop accessibility support is disabled\n");
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
}
|
|
#endif /* HAVE_ATSPI */
|
|
|
|
#ifdef HAVE_IBUS
|
|
if (focus == FOCUS_IBUS) {
|
|
ibus_init ();
|
|
|
|
if (!client_enable_ibus_focus (client)) {
|
|
g_printerr ("Can't register IBus focus change event listeners\n");
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
}
|
|
#endif /* HAVE_IBUS */
|
|
|
|
#ifdef HAVE_XTEST
|
|
if (!client_enable_xtest (client)) {
|
|
g_printerr ("Can't init xtest\n");
|
|
g_object_unref (client);
|
|
exit (1);
|
|
}
|
|
#endif /* HAVE_XTEST */
|
|
|
|
if (!opt_focus) {
|
|
g_object_get (client, "context", &context, NULL);
|
|
g_signal_connect (context, "destroyed",
|
|
G_CALLBACK(on_context_destroyed), NULL);
|
|
g_object_unref (context);
|
|
}
|
|
|
|
if (opt_fullscreen ||
|
|
g_settings_get_boolean (settings, "start-fullscreen")) {
|
|
g_object_get (client, "context", &context, NULL);
|
|
eekboard_context_set_fullscreen (context, TRUE, NULL);
|
|
g_object_unref (context);
|
|
}
|
|
|
|
g_object_get (client, "eekboard", &eekboard, NULL);
|
|
g_signal_connect (eekboard, "destroyed",
|
|
G_CALLBACK(on_destroyed), NULL);
|
|
g_object_unref (eekboard);
|
|
|
|
if (opt_keyboards != NULL) {
|
|
keyboards = g_strsplit (opt_keyboards, ",", -1);
|
|
|
|
if (!set_keyboards (client, (const gchar * const *)keyboards)) {
|
|
g_strfreev (keyboards);
|
|
retval = 1;
|
|
goto out;
|
|
}
|
|
g_strfreev (keyboards);
|
|
}
|
|
|
|
gtk_main ();
|
|
|
|
out:
|
|
if (client)
|
|
g_object_unref (client);
|
|
if (settings)
|
|
g_object_unref (settings);
|
|
|
|
return retval;
|
|
}
|