ce3fa3938a
This removes the responsibility of tracking these from the backend to the base object. The backends are instead responsible for calling the function to update the values. For the native backend, it's important that this happens on the correct thread, so each time either of these states may change, post a idle callback on the main thread that sets the, at the time of queuing said callback, up to date state. This means that things on the main thread will always be able to get a "new enough but not too new" state when listening on the 'notify::' signals and getting the property value after. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1739>
153 lines
4.4 KiB
C
153 lines
4.4 KiB
C
/*
|
|
* Copyright (C) 2018 Red Hat
|
|
*
|
|
* 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 2 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*
|
|
* Author: Carlos Garnacho <carlosg@gnome.org>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "backends/meta-keymap-utils.h"
|
|
#include "backends/native/meta-input-thread.h"
|
|
#include "backends/native/meta-seat-impl.h"
|
|
#include "backends/native/meta-seat-native.h"
|
|
#include "clutter/clutter-keymap-private.h"
|
|
|
|
static const char *option_xkb_layout = "us";
|
|
static const char *option_xkb_variant = "";
|
|
static const char *option_xkb_options = "";
|
|
|
|
typedef struct _MetaKeymapNative MetaKeymapNative;
|
|
|
|
struct _MetaKeymapNative
|
|
{
|
|
ClutterKeymap parent_instance;
|
|
|
|
struct xkb_keymap *keymap;
|
|
gboolean num_lock;
|
|
gboolean caps_lock;
|
|
};
|
|
|
|
G_DEFINE_TYPE (MetaKeymapNative, meta_keymap_native,
|
|
CLUTTER_TYPE_KEYMAP)
|
|
|
|
static void
|
|
meta_keymap_native_finalize (GObject *object)
|
|
{
|
|
MetaKeymapNative *keymap = META_KEYMAP_NATIVE (object);
|
|
|
|
xkb_keymap_unref (keymap->keymap);
|
|
|
|
G_OBJECT_CLASS (meta_keymap_native_parent_class)->finalize (object);
|
|
}
|
|
|
|
static PangoDirection
|
|
meta_keymap_native_get_direction (ClutterKeymap *keymap)
|
|
{
|
|
return PANGO_DIRECTION_NEUTRAL;
|
|
}
|
|
|
|
static void
|
|
meta_keymap_native_class_init (MetaKeymapNativeClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
ClutterKeymapClass *keymap_class = CLUTTER_KEYMAP_CLASS (klass);
|
|
|
|
object_class->finalize = meta_keymap_native_finalize;
|
|
|
|
keymap_class->get_direction = meta_keymap_native_get_direction;
|
|
}
|
|
|
|
static void
|
|
meta_keymap_native_init (MetaKeymapNative *keymap)
|
|
{
|
|
struct xkb_context *ctx;
|
|
struct xkb_rule_names names;
|
|
|
|
names.rules = "evdev";
|
|
names.model = "pc105";
|
|
names.layout = option_xkb_layout;
|
|
names.variant = option_xkb_variant;
|
|
names.options = option_xkb_options;
|
|
|
|
ctx = meta_create_xkb_context ();
|
|
g_assert (ctx);
|
|
keymap->keymap = xkb_keymap_new_from_names (ctx, &names, 0);
|
|
xkb_context_unref (ctx);
|
|
}
|
|
|
|
void
|
|
meta_keymap_native_set_keyboard_map_in_impl (MetaKeymapNative *keymap,
|
|
struct xkb_keymap *xkb_keymap)
|
|
{
|
|
g_return_if_fail (xkb_keymap != NULL);
|
|
|
|
if (keymap->keymap)
|
|
xkb_keymap_unref (keymap->keymap);
|
|
keymap->keymap = xkb_keymap_ref (xkb_keymap);
|
|
}
|
|
|
|
struct xkb_keymap *
|
|
meta_keymap_native_get_keyboard_map_in_impl (MetaKeymapNative *keymap)
|
|
{
|
|
return keymap->keymap;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
MetaKeymapNative *keymap_native;
|
|
gboolean num_lock_state;
|
|
gboolean caps_lock_state;
|
|
} UpdateLockedModifierStateData;
|
|
|
|
static gboolean
|
|
update_locked_modifier_state_in_main (gpointer user_data)
|
|
{
|
|
UpdateLockedModifierStateData *data = user_data;
|
|
|
|
clutter_keymap_set_lock_modifier_state (CLUTTER_KEYMAP (data->keymap_native),
|
|
data->caps_lock_state,
|
|
data->num_lock_state);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
void
|
|
meta_keymap_native_update_in_impl (MetaKeymapNative *keymap_native,
|
|
MetaSeatImpl *seat_impl,
|
|
struct xkb_state *xkb_state)
|
|
{
|
|
UpdateLockedModifierStateData *data;
|
|
|
|
data = g_new0 (UpdateLockedModifierStateData, 1);
|
|
data->keymap_native = keymap_native;
|
|
data->num_lock_state =
|
|
xkb_state_mod_name_is_active (xkb_state,
|
|
XKB_MOD_NAME_NUM,
|
|
XKB_STATE_MODS_LATCHED |
|
|
XKB_STATE_MODS_LOCKED);
|
|
data->caps_lock_state =
|
|
xkb_state_mod_name_is_active (xkb_state,
|
|
XKB_MOD_NAME_CAPS,
|
|
XKB_STATE_MODS_LATCHED |
|
|
XKB_STATE_MODS_LOCKED);
|
|
|
|
meta_seat_impl_queue_main_thread_idle (seat_impl,
|
|
update_locked_modifier_state_in_main,
|
|
data, g_free);
|
|
}
|