2010-12-18 20:38:49 +00:00
|
|
|
/* GDK - The GIMP Drawing Kit
|
|
|
|
* Copyright (C) 2000 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* 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
|
2012-02-27 13:01:10 +00:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
2010-12-18 20:38:49 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
|
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
|
|
* files for a list of changes. These files are distributed with
|
|
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <errno.h>
|
2012-07-16 18:02:41 +00:00
|
|
|
#include <sys/mman.h>
|
2010-12-18 20:38:49 +00:00
|
|
|
|
|
|
|
#include "gdk.h"
|
|
|
|
#include "gdkwayland.h"
|
|
|
|
|
|
|
|
#include "gdkprivate-wayland.h"
|
|
|
|
#include "gdkinternals.h"
|
|
|
|
#include "gdkkeysprivate.h"
|
|
|
|
|
2012-06-18 15:02:37 +00:00
|
|
|
#include <xkbcommon/xkbcommon.h>
|
2010-12-18 20:38:49 +00:00
|
|
|
|
|
|
|
typedef struct _GdkWaylandKeymap GdkWaylandKeymap;
|
|
|
|
typedef struct _GdkWaylandKeymapClass GdkWaylandKeymapClass;
|
|
|
|
|
|
|
|
struct _GdkWaylandKeymap
|
|
|
|
{
|
|
|
|
GdkKeymap parent_instance;
|
2012-07-16 18:02:41 +00:00
|
|
|
|
|
|
|
struct xkb_keymap *xkb_keymap;
|
2012-07-16 18:16:48 +00:00
|
|
|
struct xkb_state *xkb_state;
|
2013-04-04 03:45:54 +00:00
|
|
|
|
|
|
|
PangoDirection *direction;
|
|
|
|
gboolean bidi;
|
2010-12-18 20:38:49 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct _GdkWaylandKeymapClass
|
|
|
|
{
|
|
|
|
GdkKeymapClass parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define GDK_TYPE_WAYLAND_KEYMAP (_gdk_wayland_keymap_get_type ())
|
2011-01-07 17:10:41 +00:00
|
|
|
#define GDK_WAYLAND_KEYMAP(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), GDK_TYPE_WAYLAND_KEYMAP, GdkWaylandKeymap))
|
2010-12-18 20:38:49 +00:00
|
|
|
#define GDK_IS_WAYLAND_KEYMAP(object) (G_TYPE_CHECK_INSTANCE_TYPE ((object), GDK_TYPE_WAYLAND_KEYMAP))
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (GdkWaylandKeymap, _gdk_wayland_keymap, GDK_TYPE_KEYMAP)
|
|
|
|
|
|
|
|
static void
|
|
|
|
gdk_wayland_keymap_finalize (GObject *object)
|
|
|
|
{
|
2013-03-23 22:02:28 +00:00
|
|
|
GdkWaylandKeymap *keymap = GDK_WAYLAND_KEYMAP (object);
|
|
|
|
|
|
|
|
xkb_keymap_unref (keymap->xkb_keymap);
|
|
|
|
xkb_state_unref (keymap->xkb_state);
|
2013-04-04 03:45:54 +00:00
|
|
|
g_free (keymap->direction);
|
2013-03-23 22:02:28 +00:00
|
|
|
|
2010-12-18 20:38:49 +00:00
|
|
|
G_OBJECT_CLASS (_gdk_wayland_keymap_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static PangoDirection
|
|
|
|
gdk_wayland_keymap_get_direction (GdkKeymap *keymap)
|
|
|
|
{
|
2013-04-04 03:45:54 +00:00
|
|
|
GdkWaylandKeymap *keymap_wayland = GDK_WAYLAND_KEYMAP (keymap);
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
for (i = 0; i < xkb_keymap_num_layouts (keymap_wayland->xkb_keymap); i++)
|
|
|
|
{
|
|
|
|
if (xkb_state_layout_index_is_active (keymap_wayland->xkb_state, i, XKB_STATE_LAYOUT_EFFECTIVE))
|
|
|
|
return keymap_wayland->direction[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return PANGO_DIRECTION_NEUTRAL;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_have_bidi_layouts (GdkKeymap *keymap)
|
|
|
|
{
|
2013-04-04 03:45:54 +00:00
|
|
|
GdkWaylandKeymap *keymap_wayland = GDK_WAYLAND_KEYMAP (keymap);
|
|
|
|
|
|
|
|
return keymap_wayland->bidi;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_get_caps_lock_state (GdkKeymap *keymap)
|
|
|
|
{
|
2012-07-16 18:51:48 +00:00
|
|
|
return xkb_state_led_name_is_active (GDK_WAYLAND_KEYMAP (keymap)->xkb_state,
|
|
|
|
XKB_LED_NAME_CAPS);
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_get_num_lock_state (GdkKeymap *keymap)
|
|
|
|
{
|
2012-07-16 18:51:48 +00:00
|
|
|
return xkb_state_led_name_is_active (GDK_WAYLAND_KEYMAP (keymap)->xkb_state,
|
|
|
|
XKB_LED_NAME_NUM);
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_get_entries_for_keyval (GdkKeymap *keymap,
|
2011-01-07 17:10:41 +00:00
|
|
|
guint keyval,
|
|
|
|
GdkKeymapKey **keys,
|
|
|
|
gint *n_keys)
|
2010-12-18 20:38:49 +00:00
|
|
|
{
|
2012-07-16 19:08:20 +00:00
|
|
|
if (n_keys)
|
|
|
|
*n_keys = 1;
|
|
|
|
if (keys)
|
2011-01-07 17:10:41 +00:00
|
|
|
{
|
2012-07-16 19:08:20 +00:00
|
|
|
*keys = g_new0 (GdkKeymapKey, 1);
|
|
|
|
(*keys)->keycode = keyval;
|
2011-01-07 17:10:41 +00:00
|
|
|
}
|
|
|
|
|
2012-07-16 19:08:20 +00:00
|
|
|
return TRUE;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_get_entries_for_keycode (GdkKeymap *keymap,
|
2011-01-07 17:10:41 +00:00
|
|
|
guint hardware_keycode,
|
|
|
|
GdkKeymapKey **keys,
|
|
|
|
guint **keyvals,
|
|
|
|
gint *n_entries)
|
2010-12-18 20:38:49 +00:00
|
|
|
{
|
2013-04-05 01:52:12 +00:00
|
|
|
struct xkb_keymap *xkb_keymap = GDK_WAYLAND_KEYMAP (keymap)->xkb_keymap;
|
|
|
|
gint num_layouts, layout;
|
|
|
|
gint num_entries;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
num_layouts = xkb_keymap_num_layouts_for_key (xkb_keymap, hardware_keycode);
|
|
|
|
|
|
|
|
num_entries = 0;
|
|
|
|
for (layout = 0; layout < num_layouts; layout++)
|
|
|
|
num_entries += xkb_keymap_num_levels_for_key (xkb_keymap, hardware_keycode, layout);
|
|
|
|
|
2012-07-16 19:08:20 +00:00
|
|
|
if (n_entries)
|
2013-04-05 01:52:12 +00:00
|
|
|
*n_entries = num_entries;
|
2011-01-07 17:10:41 +00:00
|
|
|
if (keys)
|
|
|
|
{
|
2013-04-05 01:52:12 +00:00
|
|
|
*keys = g_new0 (GdkKeymapKey, num_entries);
|
2012-07-16 19:08:20 +00:00
|
|
|
(*keys)->keycode = hardware_keycode;
|
2011-01-07 17:10:41 +00:00
|
|
|
}
|
|
|
|
if (keyvals)
|
|
|
|
{
|
2013-04-05 01:52:12 +00:00
|
|
|
*keyvals = g_new0 (guint, num_entries);
|
2012-07-16 19:08:20 +00:00
|
|
|
(*keyvals)[0] = hardware_keycode;
|
2011-01-07 17:10:41 +00:00
|
|
|
}
|
2013-04-05 01:52:12 +00:00
|
|
|
|
|
|
|
i = 0;
|
|
|
|
for (layout = 0; layout < num_layouts; layout++)
|
|
|
|
{
|
|
|
|
gint num_levels, level;
|
|
|
|
num_levels = xkb_keymap_num_levels_for_key (xkb_keymap, hardware_keycode, layout);
|
|
|
|
for (level = 0; level < num_levels; level++)
|
|
|
|
{
|
|
|
|
const xkb_keysym_t *syms;
|
|
|
|
int num_syms;
|
|
|
|
num_syms = xkb_keymap_key_get_syms_by_level (xkb_keymap, hardware_keycode, layout, 0, &syms);
|
|
|
|
if (keys)
|
|
|
|
{
|
|
|
|
(*keys)[i].keycode = hardware_keycode;
|
|
|
|
(*keys)[i].group = layout;
|
|
|
|
(*keys)[i].level = level;
|
|
|
|
}
|
|
|
|
if (keyvals && num_syms > 0)
|
|
|
|
(*keyvals)[i] = syms[0];
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-16 19:08:20 +00:00
|
|
|
return TRUE;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static guint
|
|
|
|
gdk_wayland_keymap_lookup_key (GdkKeymap *keymap,
|
2011-01-07 17:10:41 +00:00
|
|
|
const GdkKeymapKey *key)
|
2010-12-18 20:38:49 +00:00
|
|
|
{
|
2012-07-16 19:08:20 +00:00
|
|
|
return key->keycode;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_translate_keyboard_state (GdkKeymap *keymap,
|
2011-01-07 17:10:41 +00:00
|
|
|
guint hardware_keycode,
|
|
|
|
GdkModifierType state,
|
|
|
|
gint group,
|
|
|
|
guint *keyval,
|
|
|
|
gint *effective_group,
|
|
|
|
gint *level,
|
|
|
|
GdkModifierType *consumed_modifiers)
|
2010-12-18 20:38:49 +00:00
|
|
|
{
|
2011-02-10 21:35:38 +00:00
|
|
|
g_return_val_if_fail (keymap == NULL || GDK_IS_KEYMAP (keymap), FALSE);
|
|
|
|
g_return_val_if_fail (group < 4, FALSE);
|
|
|
|
|
|
|
|
if (keyval)
|
2012-07-16 19:08:20 +00:00
|
|
|
*keyval = hardware_keycode;
|
2011-02-10 21:35:38 +00:00
|
|
|
if (effective_group)
|
|
|
|
*effective_group = 0;
|
|
|
|
if (level)
|
|
|
|
*level = 0;
|
|
|
|
if (consumed_modifiers)
|
|
|
|
*consumed_modifiers = 0;
|
|
|
|
|
2012-07-16 19:08:20 +00:00
|
|
|
return TRUE;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
gdk_wayland_keymap_add_virtual_modifiers (GdkKeymap *keymap,
|
2011-01-07 17:10:41 +00:00
|
|
|
GdkModifierType *state)
|
2010-12-18 20:38:49 +00:00
|
|
|
{
|
2012-07-16 19:08:20 +00:00
|
|
|
return;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
gdk_wayland_keymap_map_virtual_modifiers (GdkKeymap *keymap,
|
2011-01-07 17:10:41 +00:00
|
|
|
GdkModifierType *state)
|
2010-12-18 20:38:49 +00:00
|
|
|
{
|
2012-07-16 19:08:20 +00:00
|
|
|
return TRUE;
|
2010-12-18 20:38:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_gdk_wayland_keymap_class_init (GdkWaylandKeymapClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
GdkKeymapClass *keymap_class = GDK_KEYMAP_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = gdk_wayland_keymap_finalize;
|
|
|
|
|
|
|
|
keymap_class->get_direction = gdk_wayland_keymap_get_direction;
|
|
|
|
keymap_class->have_bidi_layouts = gdk_wayland_keymap_have_bidi_layouts;
|
|
|
|
keymap_class->get_caps_lock_state = gdk_wayland_keymap_get_caps_lock_state;
|
|
|
|
keymap_class->get_num_lock_state = gdk_wayland_keymap_get_num_lock_state;
|
|
|
|
keymap_class->get_entries_for_keyval = gdk_wayland_keymap_get_entries_for_keyval;
|
|
|
|
keymap_class->get_entries_for_keycode = gdk_wayland_keymap_get_entries_for_keycode;
|
|
|
|
keymap_class->lookup_key = gdk_wayland_keymap_lookup_key;
|
|
|
|
keymap_class->translate_keyboard_state = gdk_wayland_keymap_translate_keyboard_state;
|
|
|
|
keymap_class->add_virtual_modifiers = gdk_wayland_keymap_add_virtual_modifiers;
|
|
|
|
keymap_class->map_virtual_modifiers = gdk_wayland_keymap_map_virtual_modifiers;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_gdk_wayland_keymap_init (GdkWaylandKeymap *keymap)
|
|
|
|
{
|
|
|
|
}
|
2011-01-07 17:10:41 +00:00
|
|
|
|
2013-04-04 03:45:54 +00:00
|
|
|
static void
|
|
|
|
update_direction (GdkWaylandKeymap *keymap)
|
|
|
|
{
|
|
|
|
gint num_layouts;
|
|
|
|
gint *rtl;
|
|
|
|
gint key;
|
|
|
|
gboolean have_rtl, have_ltr;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
num_layouts = xkb_keymap_num_layouts (keymap->xkb_keymap);
|
|
|
|
|
|
|
|
g_free (keymap->direction);
|
|
|
|
keymap->direction = g_new0 (PangoDirection, num_layouts);
|
|
|
|
|
|
|
|
rtl = g_new0 (gint, num_layouts);
|
|
|
|
|
|
|
|
for (key = 8; key < 255; key++) /* FIXME: min/max keycode */
|
|
|
|
{
|
|
|
|
gint layouts;
|
|
|
|
gint layout;
|
|
|
|
|
|
|
|
layouts = xkb_keymap_num_layouts_for_key (keymap->xkb_keymap, key);
|
|
|
|
for (layout = 0; layout < layouts; layout++)
|
|
|
|
{
|
|
|
|
const xkb_keysym_t *syms;
|
|
|
|
gint num_syms;
|
|
|
|
gint sym;
|
|
|
|
|
|
|
|
num_syms = xkb_keymap_key_get_syms_by_level (keymap->xkb_keymap, key, layout, 0, &syms);
|
|
|
|
for (sym = 0; sym < num_syms; sym++)
|
|
|
|
{
|
|
|
|
PangoDirection dir;
|
|
|
|
dir = pango_unichar_direction (xkb_keysym_to_utf32 (syms[sym]));
|
|
|
|
switch (dir)
|
|
|
|
{
|
|
|
|
case PANGO_DIRECTION_RTL:
|
|
|
|
rtl[layout]++;
|
|
|
|
break;
|
|
|
|
case PANGO_DIRECTION_LTR:
|
|
|
|
rtl[layout]--;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
have_rtl = have_ltr = FALSE;
|
|
|
|
for (i = 0; i < num_layouts; i++)
|
|
|
|
{
|
|
|
|
if (rtl[i] > 0)
|
|
|
|
{
|
|
|
|
keymap->direction[i] = PANGO_DIRECTION_RTL;
|
|
|
|
have_rtl = TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
keymap->direction[i] = PANGO_DIRECTION_LTR;
|
|
|
|
have_ltr = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (have_rtl && have_ltr)
|
|
|
|
keymap->bidi = TRUE;
|
|
|
|
|
|
|
|
g_free (rtl);
|
|
|
|
}
|
|
|
|
|
2013-04-05 01:10:18 +00:00
|
|
|
GdkKeymap *
|
|
|
|
_gdk_wayland_keymap_new ()
|
|
|
|
{
|
|
|
|
GdkWaylandKeymap *keymap;
|
|
|
|
struct xkb_context *context;
|
|
|
|
struct xkb_rule_names names;
|
|
|
|
|
|
|
|
keymap = g_object_new (_gdk_wayland_keymap_get_type(), NULL);
|
|
|
|
|
|
|
|
context = xkb_context_new (0);
|
|
|
|
|
|
|
|
names.rules = "evdev";
|
|
|
|
names.model = "pc105";
|
|
|
|
names.layout = "us";
|
|
|
|
names.variant = "";
|
|
|
|
names.options = "";
|
|
|
|
keymap->xkb_keymap = xkb_keymap_new_from_names (context, &names, 0);
|
|
|
|
keymap->xkb_state = xkb_state_new (keymap->xkb_keymap);
|
|
|
|
xkb_context_unref (context);
|
|
|
|
|
|
|
|
update_direction (keymap);
|
|
|
|
|
|
|
|
return GDK_KEYMAP (keymap);
|
|
|
|
}
|
|
|
|
|
2013-03-23 22:02:28 +00:00
|
|
|
void
|
|
|
|
_gdk_wayland_keymap_update_from_fd (GdkKeymap *keymap,
|
|
|
|
uint32_t format,
|
|
|
|
uint32_t fd,
|
|
|
|
uint32_t size)
|
2012-07-16 11:33:35 +00:00
|
|
|
{
|
2013-03-23 22:02:28 +00:00
|
|
|
GdkWaylandKeymap *keymap_wayland = GDK_WAYLAND_KEYMAP (keymap);
|
2012-07-16 18:02:41 +00:00
|
|
|
struct xkb_context *context;
|
|
|
|
char *map_str;
|
2012-07-16 11:33:35 +00:00
|
|
|
|
2012-07-16 18:02:41 +00:00
|
|
|
context = xkb_context_new (0);
|
2012-07-16 11:33:35 +00:00
|
|
|
|
2012-07-16 18:02:41 +00:00
|
|
|
map_str = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0);
|
2013-03-23 22:02:28 +00:00
|
|
|
if (map_str == MAP_FAILED)
|
|
|
|
{
|
|
|
|
close(fd);
|
|
|
|
return;
|
2012-07-16 18:02:41 +00:00
|
|
|
}
|
2012-07-16 11:33:35 +00:00
|
|
|
|
2013-03-23 22:02:28 +00:00
|
|
|
xkb_keymap_unref (keymap_wayland->xkb_keymap);
|
|
|
|
keymap_wayland->xkb_keymap = xkb_keymap_new_from_string (context, map_str, format, 0);
|
2012-07-16 18:02:41 +00:00
|
|
|
munmap (map_str, size);
|
|
|
|
close (fd);
|
2012-07-16 11:33:35 +00:00
|
|
|
|
2013-03-23 22:02:28 +00:00
|
|
|
xkb_state_unref (keymap_wayland->xkb_state);
|
|
|
|
keymap_wayland->xkb_state = xkb_state_new (keymap_wayland->xkb_keymap);
|
|
|
|
|
|
|
|
xkb_context_unref (context);
|
2013-04-04 03:45:54 +00:00
|
|
|
|
|
|
|
update_direction (keymap_wayland);
|
2012-07-16 18:02:41 +00:00
|
|
|
}
|
2012-07-16 11:33:35 +00:00
|
|
|
|
2012-07-16 18:02:41 +00:00
|
|
|
struct xkb_keymap *_gdk_wayland_keymap_get_xkb_keymap (GdkKeymap *keymap)
|
|
|
|
{
|
|
|
|
return GDK_WAYLAND_KEYMAP (keymap)->xkb_keymap;
|
2012-07-16 11:33:35 +00:00
|
|
|
}
|
|
|
|
|
2012-07-16 18:16:48 +00:00
|
|
|
struct xkb_state *_gdk_wayland_keymap_get_xkb_state (GdkKeymap *keymap)
|
2011-01-07 17:10:41 +00:00
|
|
|
{
|
2012-07-16 18:16:48 +00:00
|
|
|
return GDK_WAYLAND_KEYMAP (keymap)->xkb_state;
|
2011-01-07 17:10:41 +00:00
|
|
|
}
|