~ctf/unity-settings-daemon/bug1389099_mic_volume_icons

« back to all changes in this revision

Viewing changes to plugins/common/gsd-keygrab.c

  • Committer: Package Import Robot
  • Author(s): Robert Ancell
  • Date: 2014-02-07 11:44:36 UTC
  • Revision ID: package-import@ubuntu.com-20140207114436-7t5u3yvwc4ul7w3e
Tags: upstream-14.04.0
ImportĀ upstreamĀ versionĀ 14.04.0

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
 
2
 *
 
3
 * Copyright (C) 2001-2003 Bastien Nocera <hadess@hadess.net>
 
4
 * Copyright (C) 2006-2007 William Jon McCann <mccann@jhu.edu>
 
5
 * Copyright (C) 2008 Jens Granseuer <jensgr@gmx.net>
 
6
 *
 
7
 * This program is free software; you can redistribute it and/or modify
 
8
 * it under the terms of the GNU General Public License as published by
 
9
 * the Free Software Foundation; either version 2 of the License, or
 
10
 * (at your option) any later version.
 
11
 *
 
12
 * This program is distributed in the hope that it will be useful,
 
13
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
15
 * GNU General Public License for more details.
 
16
 *
 
17
 * You should have received a copy of the GNU General Public License
 
18
 * along with this program; if not, write to the Free Software
 
19
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 
20
 *
 
21
 */
 
22
 
 
23
#include "config.h"
 
24
 
 
25
#include <string.h>
 
26
#include <gdk/gdk.h>
 
27
#include <gdk/gdkx.h>
 
28
#include <gtk/gtk.h>
 
29
#include <X11/XKBlib.h>
 
30
#include <X11/extensions/XInput2.h>
 
31
#include <X11/extensions/XKB.h>
 
32
#include <gdk/gdkkeysyms.h>
 
33
 
 
34
#include "gsd-keygrab.h"
 
35
 
 
36
/* these are the mods whose combinations are ignored by the keygrabbing code */
 
37
static GdkModifierType gsd_ignored_mods = 0;
 
38
 
 
39
/* these are the ones we actually use for global keys, we always only check
 
40
 * for these set */
 
41
static GdkModifierType gsd_used_mods = 0;
 
42
 
 
43
/* Taken from a comment in XF86keysym.h */
 
44
#define XF86KEYS_RANGE_MIN 0x10080001
 
45
#define XF86KEYS_RANGE_MAX 0x1008FFFF
 
46
 
 
47
#define FKEYS_RANGE_MIN GDK_KEY_F1
 
48
#define FKEYS_RANGE_MAX GDK_KEY_R15
 
49
 
 
50
#define IN_RANGE(x, min, max) (x >= min && x <= max)
 
51
 
 
52
static void
 
53
setup_modifiers (void)
 
54
{
 
55
        if (gsd_used_mods == 0 || gsd_ignored_mods == 0) {
 
56
                GdkModifierType dynmods;
 
57
 
 
58
                /* default modifiers */
 
59
                gsd_ignored_mods = \
 
60
                        0x2000 /*Xkb modifier*/ | GDK_LOCK_MASK | GDK_HYPER_MASK;
 
61
                gsd_used_mods = \
 
62
                        GDK_SHIFT_MASK | GDK_CONTROL_MASK |\
 
63
                        GDK_MOD1_MASK | GDK_MOD2_MASK | GDK_MOD3_MASK | GDK_MOD4_MASK |\
 
64
                        GDK_MOD5_MASK | GDK_SUPER_MASK | GDK_META_MASK;
 
65
 
 
66
                /* NumLock can be assigned to varying keys so we need to
 
67
                 * resolve and ignore it specially */
 
68
                dynmods = XkbKeysymToModifiers (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), GDK_KEY_Num_Lock);
 
69
 
 
70
                gsd_ignored_mods |= dynmods;
 
71
                gsd_used_mods &= ~dynmods;
 
72
        }
 
73
}
 
74
 
 
75
static void
 
76
grab_key_real (guint      keycode,
 
77
               GdkWindow *root,
 
78
               gboolean   grab,
 
79
               gboolean   synchronous,
 
80
               XIGrabModifiers *mods,
 
81
               int        num_mods)
 
82
{
 
83
        XIEventMask evmask;
 
84
        unsigned char mask[(XI_LASTEVENT + 7)/8];
 
85
 
 
86
        memset (mask, 0, sizeof (mask));
 
87
        XISetMask (mask, XI_KeyPress);
 
88
        XISetMask (mask, XI_KeyRelease);
 
89
 
 
90
        evmask.deviceid = XIAllMasterDevices;
 
91
        evmask.mask_len = sizeof (mask);
 
92
        evmask.mask = mask;
 
93
 
 
94
        if (grab) {
 
95
                XIGrabKeycode (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()),
 
96
                               XIAllMasterDevices,
 
97
                               keycode,
 
98
                               GDK_WINDOW_XID (root),
 
99
                               GrabModeAsync,
 
100
                               synchronous ? GrabModeSync : GrabModeAsync,
 
101
                               False,
 
102
                               &evmask,
 
103
                               num_mods,
 
104
                               mods);
 
105
        } else {
 
106
                XIUngrabKeycode (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()),
 
107
                                 XIAllMasterDevices,
 
108
                                 keycode,
 
109
                                 GDK_WINDOW_XID (root),
 
110
                                 num_mods,
 
111
                                 mods);
 
112
        }
 
113
}
 
114
 
 
115
/* Grab the key. In order to ignore GSD_IGNORED_MODS we need to grab
 
116
 * all combinations of the ignored modifiers and those actually used
 
117
 * for the binding (if any).
 
118
 *
 
119
 * inspired by all_combinations from gnome-panel/gnome-panel/global-keys.c
 
120
 *
 
121
 * This may generate X errors.  The correct way to use this is like:
 
122
 *
 
123
 *        gdk_error_trap_push ();
 
124
 *
 
125
 *        grab_key_unsafe (key, grab, screens);
 
126
 *
 
127
 *        gdk_flush ();
 
128
 *        if (gdk_error_trap_pop ())
 
129
 *                g_warning ("Grab failed, another application may already have access to key '%u'",
 
130
 *                           key->keycode);
 
131
 *
 
132
 * This is not done in the function itself, to allow doing multiple grab_key
 
133
 * operations with one flush only.
 
134
 */
 
135
#define N_BITS 32
 
136
static void
 
137
grab_key_internal (Key             *key,
 
138
                   gboolean         grab,
 
139
                   GsdKeygrabFlags  flags,
 
140
                   GSList          *screens)
 
141
{
 
142
        int     indexes[N_BITS]; /* indexes of bits we need to flip */
 
143
        int     i;
 
144
        int     bit;
 
145
        int     bits_set_cnt;
 
146
        int     uppervalue;
 
147
        guint   mask, modifiers;
 
148
        GArray *all_mods;
 
149
        GSList *l;
 
150
 
 
151
        setup_modifiers ();
 
152
 
 
153
        mask = gsd_ignored_mods & ~key->state & GDK_MODIFIER_MASK;
 
154
 
 
155
        /* XGrabKey requires real modifiers, not virtual ones */
 
156
        modifiers = key->state;
 
157
        gdk_keymap_map_virtual_modifiers (gdk_keymap_get_default (), &modifiers);
 
158
        modifiers &= ~(GDK_META_MASK | GDK_SUPER_MASK | GDK_HYPER_MASK);
 
159
 
 
160
        /* If key doesn't have a usable modifier, we don't want
 
161
         * to grab it, since the user might lose a useful key.
 
162
         *
 
163
         * The exception is the XFree86 keys and the Function keys
 
164
         * (which are useful to grab without a modifier).
 
165
         */
 
166
        if (!(flags & GSD_KEYGRAB_ALLOW_UNMODIFIED) &&
 
167
            (modifiers & gsd_used_mods) == 0 &&
 
168
            !IN_RANGE(key->keysym, XF86KEYS_RANGE_MIN, XF86KEYS_RANGE_MAX) &&
 
169
            !IN_RANGE(key->keysym, FKEYS_RANGE_MIN, FKEYS_RANGE_MAX) &&
 
170
             key->keysym != GDK_KEY_Pause &&
 
171
             key->keysym != GDK_KEY_Print &&
 
172
             key->keysym != GDK_KEY_Scroll_Lock &&
 
173
             key->keysym != GDK_KEY_Caps_Lock &&
 
174
             key->keysym != GDK_KEY_Pause &&
 
175
             key->keysym != GDK_KEY_Break &&
 
176
             key->keysym != GDK_KEY_Menu) {
 
177
                GString *keycodes;
 
178
 
 
179
                keycodes = g_string_new ("");
 
180
                if (key->keycodes != NULL) {
 
181
                        guint *c;
 
182
 
 
183
                        for (c = key->keycodes; *c; ++c) {
 
184
                                g_string_printf (keycodes, " %u", *c);
 
185
                        }
 
186
                }
 
187
                g_warning ("Key 0x%x (keycodes: %s)  with state 0x%x (resolved to 0x%x) "
 
188
                           " has no usable modifiers (usable modifiers are 0x%x)",
 
189
                           key->keysym, keycodes->str, key->state, modifiers, gsd_used_mods);
 
190
                g_string_free (keycodes, TRUE);
 
191
 
 
192
                return;
 
193
        }
 
194
 
 
195
        bit = 0;
 
196
        /* store the indexes of all set bits in mask in the array */
 
197
        for (i = 0; mask; ++i, mask >>= 1) {
 
198
                if (mask & 0x1) {
 
199
                        indexes[bit++] = i;
 
200
                }
 
201
        }
 
202
 
 
203
        bits_set_cnt = bit;
 
204
 
 
205
        all_mods = g_array_new (FALSE, TRUE, sizeof(XIGrabModifiers));
 
206
        uppervalue = 1 << bits_set_cnt;
 
207
        /* store all possible modifier combinations for our mask into all_mods */
 
208
        for (i = 0; i < uppervalue; ++i) {
 
209
                int     j;
 
210
                int     result = 0;
 
211
                XIGrabModifiers *mod;
 
212
 
 
213
                /* map bits in the counter to those in the mask */
 
214
                for (j = 0; j < bits_set_cnt; ++j) {
 
215
                        if (i & (1 << j)) {
 
216
                                result |= (1 << indexes[j]);
 
217
                        }
 
218
                }
 
219
 
 
220
                /* Grow the array by one, to fit our new XIGrabModifiers item */
 
221
                g_array_set_size (all_mods, all_mods->len + 1);
 
222
                mod = &g_array_index (all_mods, XIGrabModifiers, all_mods->len - 1);
 
223
                mod->modifiers = result | modifiers;
 
224
        }
 
225
 
 
226
        /* Capture the actual keycodes with the modifier array */
 
227
        for (l = screens; l; l = l->next) {
 
228
                GdkScreen *screen = l->data;
 
229
                guint *code;
 
230
 
 
231
                for (code = key->keycodes; *code; ++code) {
 
232
                        grab_key_real (*code,
 
233
                                       gdk_screen_get_root_window (screen),
 
234
                                       grab,
 
235
                                       flags & GSD_KEYGRAB_SYNCHRONOUS,
 
236
                                       (XIGrabModifiers *) all_mods->data,
 
237
                                       all_mods->len);
 
238
                }
 
239
        }
 
240
        g_array_free (all_mods, TRUE);
 
241
}
 
242
 
 
243
static void
 
244
get_keys_for_bit (guint  bit,
 
245
                  guint *left,
 
246
                  guint *right)
 
247
{
 
248
        guint left_dummy;
 
249
        guint right_dummy;
 
250
 
 
251
        if (left == NULL)
 
252
                left = &left_dummy;
 
253
        if (right == NULL)
 
254
                right = &right_dummy;
 
255
 
 
256
        *left = 0;
 
257
        *right = 0;
 
258
 
 
259
        switch (1 << bit) {
 
260
        case GDK_SHIFT_MASK:
 
261
                *left = GDK_KEY_Shift_L;
 
262
                *right = GDK_KEY_Shift_R;
 
263
                break;
 
264
        case GDK_CONTROL_MASK:
 
265
                *left = GDK_KEY_Control_L;
 
266
                *right = GDK_KEY_Control_R;
 
267
                break;
 
268
        case GDK_LOCK_MASK:
 
269
                *left = GDK_KEY_Caps_Lock;
 
270
                *right = GDK_KEY_Shift_Lock;
 
271
                break;
 
272
        case GDK_META_MASK:
 
273
        case GDK_MOD1_MASK:
 
274
                *left = GDK_KEY_Alt_L;
 
275
                *right = GDK_KEY_Alt_R;
 
276
                break;
 
277
        case GDK_SUPER_MASK:
 
278
                *left = GDK_KEY_Super_L;
 
279
                *right = GDK_KEY_Super_R;
 
280
                break;
 
281
        }
 
282
}
 
283
 
 
284
static guint
 
285
get_mask_for_key (guint key)
 
286
{
 
287
        switch (key) {
 
288
        case GDK_KEY_Shift_L:
 
289
        case GDK_KEY_Shift_R:
 
290
                return GDK_SHIFT_MASK;
 
291
        case GDK_KEY_Control_L:
 
292
        case GDK_KEY_Control_R:
 
293
                return GDK_CONTROL_MASK;
 
294
        case GDK_KEY_Caps_Lock:
 
295
        case GDK_KEY_Shift_Lock:
 
296
                return GDK_LOCK_MASK;
 
297
        case GDK_KEY_Meta_L:
 
298
        case GDK_KEY_Meta_R:
 
299
        case GDK_KEY_Alt_L:
 
300
        case GDK_KEY_Alt_R:
 
301
                return GDK_MOD1_MASK;
 
302
        case GDK_KEY_Super_L:
 
303
        case GDK_KEY_Super_R:
 
304
                return GDK_SUPER_MASK;
 
305
        }
 
306
 
 
307
        return 0;
 
308
}
 
309
 
 
310
static guint
 
311
get_mirrored_key (guint key)
 
312
{
 
313
        switch (key) {
 
314
        case GDK_KEY_Shift_L:
 
315
                return GDK_KEY_Shift_R;
 
316
        case GDK_KEY_Shift_R:
 
317
                return GDK_KEY_Shift_L;
 
318
        case GDK_KEY_Control_L:
 
319
                return GDK_KEY_Control_R;
 
320
        case GDK_KEY_Control_R:
 
321
                return GDK_KEY_Control_L;
 
322
        case GDK_KEY_Meta_L:
 
323
                return GDK_KEY_Meta_R;
 
324
        case GDK_KEY_Meta_R:
 
325
                return GDK_KEY_Meta_L;
 
326
        case GDK_KEY_Alt_L:
 
327
                return GDK_KEY_Alt_R;
 
328
        case GDK_KEY_Alt_R:
 
329
                return GDK_KEY_Alt_L;
 
330
        case GDK_KEY_Super_L:
 
331
                return GDK_KEY_Super_R;
 
332
        case GDK_KEY_Super_R:
 
333
                return GDK_KEY_Super_L;
 
334
        case GDK_KEY_Hyper_L:
 
335
                return GDK_KEY_Hyper_R;
 
336
        case GDK_KEY_Hyper_R:
 
337
                return GDK_KEY_Hyper_L;
 
338
        }
 
339
 
 
340
        return 0;
 
341
}
 
342
 
 
343
void
 
344
grab_key_unsafe (Key             *key,
 
345
                 GsdKeygrabFlags  flags,
 
346
                 GSList          *screens)
 
347
{
 
348
        guint key_mask = get_mask_for_key (key->keysym);
 
349
 
 
350
        grab_key_internal (key, TRUE, flags, screens);
 
351
 
 
352
        if (key_mask != 0) {
 
353
                Key copy;
 
354
                guint i, j;
 
355
 
 
356
                if ((key->state & key_mask) != 0) {
 
357
                        guint mirror = get_mirrored_key (key->keysym);
 
358
 
 
359
                        if (mirror != 0) {
 
360
                                gint mirror_keys_len;
 
361
                                GdkKeymapKey *mirror_keys;
 
362
 
 
363
                                gdk_keymap_get_entries_for_keyval (gdk_keymap_get_default (),
 
364
                                                                   mirror,
 
365
                                                                   &mirror_keys,
 
366
                                                                   &mirror_keys_len);
 
367
 
 
368
                                copy.keysym = mirror;
 
369
                                copy.state = key->state;
 
370
                                copy.keycodes = g_new0 (guint, mirror_keys_len + 1);
 
371
 
 
372
                                for (j = 0; j < mirror_keys_len; j++)
 
373
                                        copy.keycodes[j] = mirror_keys[j].keycode;
 
374
 
 
375
                                grab_key_internal (&copy, TRUE, flags, screens);
 
376
 
 
377
                                g_free (copy.keycodes);
 
378
                                g_free (mirror_keys);
 
379
                        }
 
380
                }
 
381
 
 
382
                for (i = 0; i < 8 * sizeof (guint); i++) {
 
383
                        guint left, right;
 
384
                        gint left_keys_len, right_keys_len;
 
385
                        GdkKeymapKey *left_keys, *right_keys;
 
386
 
 
387
                        if (1 << i == key_mask || (key->state & 1 << i) == 0)
 
388
                                continue;
 
389
 
 
390
                        get_keys_for_bit (i, &left, &right);
 
391
 
 
392
                        if (left == 0 && right == 0)
 
393
                                continue;
 
394
 
 
395
                        left_keys_len = 0;
 
396
                        right_keys_len = 0;
 
397
                        left_keys = NULL;
 
398
                        right_keys = NULL;
 
399
 
 
400
                        if (left != 0)
 
401
                                gdk_keymap_get_entries_for_keyval (gdk_keymap_get_default (),
 
402
                                                                   left,
 
403
                                                                   &left_keys,
 
404
                                                                   &left_keys_len);
 
405
 
 
406
                        if (right != 0)
 
407
                                gdk_keymap_get_entries_for_keyval (gdk_keymap_get_default (),
 
408
                                                                   right,
 
409
                                                                   &right_keys,
 
410
                                                                   &right_keys_len);
 
411
 
 
412
                        copy.keysym = left != 0 ? left : right;
 
413
                        copy.state = (key->state | key_mask) & ~(1 << i);
 
414
                        copy.keycodes = g_new0 (guint, left_keys_len + right_keys_len + 1);
 
415
 
 
416
                        for (j = 0; j < left_keys_len; j++)
 
417
                                copy.keycodes[j] = left_keys[j].keycode;
 
418
                        for (j = 0; j < right_keys_len; j++)
 
419
                                copy.keycodes[left_keys_len + j] = right_keys[j].keycode;
 
420
 
 
421
                        grab_key_internal (&copy, TRUE, flags, screens);
 
422
 
 
423
                        g_free (copy.keycodes);
 
424
                        g_free (right_keys);
 
425
                        g_free (left_keys);
 
426
                }
 
427
        }
 
428
}
 
429
 
 
430
void
 
431
ungrab_key_unsafe (Key    *key,
 
432
                   GSList *screens)
 
433
{
 
434
        guint key_mask = get_mask_for_key (key->keysym);
 
435
 
 
436
        grab_key_internal (key, FALSE, 0, screens);
 
437
 
 
438
        if (key_mask != 0) {
 
439
                Key copy;
 
440
                guint i, j;
 
441
 
 
442
                if ((key->state & key_mask) != 0) {
 
443
                        guint mirror = get_mirrored_key (key->keysym);
 
444
 
 
445
                        if (mirror != 0) {
 
446
                                gint mirror_keys_len;
 
447
                                GdkKeymapKey *mirror_keys;
 
448
 
 
449
                                gdk_keymap_get_entries_for_keyval (gdk_keymap_get_default (),
 
450
                                                                   mirror,
 
451
                                                                   &mirror_keys,
 
452
                                                                   &mirror_keys_len);
 
453
 
 
454
                                copy.keysym = mirror;
 
455
                                copy.state = key->state;
 
456
                                copy.keycodes = g_new0 (guint, mirror_keys_len + 1);
 
457
 
 
458
                                for (j = 0; j < mirror_keys_len; j++)
 
459
                                        copy.keycodes[j] = mirror_keys[j].keycode;
 
460
 
 
461
                                grab_key_internal (&copy, FALSE, 0, screens);
 
462
 
 
463
                                g_free (copy.keycodes);
 
464
                                g_free (mirror_keys);
 
465
                        }
 
466
                }
 
467
 
 
468
                for (i = 0; i < 8 * sizeof (guint); i++) {
 
469
                        guint left, right;
 
470
                        gint left_keys_len, right_keys_len;
 
471
                        GdkKeymapKey *left_keys, *right_keys;
 
472
 
 
473
                        if (1 << i == key_mask || (key->state & 1 << i) == 0)
 
474
                                continue;
 
475
 
 
476
                        get_keys_for_bit (i, &left, &right);
 
477
 
 
478
                        if (left == 0 && right == 0)
 
479
                                continue;
 
480
 
 
481
                        left_keys_len = 0;
 
482
                        right_keys_len = 0;
 
483
                        left_keys = NULL;
 
484
                        right_keys = NULL;
 
485
 
 
486
                        if (left != 0)
 
487
                                gdk_keymap_get_entries_for_keyval (gdk_keymap_get_default (),
 
488
                                                                   left,
 
489
                                                                   &left_keys,
 
490
                                                                   &left_keys_len);
 
491
 
 
492
                        if (right != 0)
 
493
                                gdk_keymap_get_entries_for_keyval (gdk_keymap_get_default (),
 
494
                                                                   right,
 
495
                                                                   &right_keys,
 
496
                                                                   &right_keys_len);
 
497
 
 
498
                        copy.keysym = left != 0 ? left : right;
 
499
                        copy.state = (key->state | key_mask) & ~(1 << i);
 
500
                        copy.keycodes = g_new0 (guint, left_keys_len + right_keys_len + 1);
 
501
 
 
502
                        for (j = 0; j < left_keys_len; j++)
 
503
                                copy.keycodes[j] = left_keys[j].keycode;
 
504
                        for (j = 0; j < right_keys_len; j++)
 
505
                                copy.keycodes[left_keys_len + j] = right_keys[j].keycode;
 
506
 
 
507
                        grab_key_internal (&copy, FALSE, 0, screens);
 
508
 
 
509
                        g_free (copy.keycodes);
 
510
                        g_free (right_keys);
 
511
                        g_free (left_keys);
 
512
                }
 
513
        }
 
514
}
 
515
 
 
516
static gboolean
 
517
have_xkb (Display *dpy)
 
518
{
 
519
        static int have_xkb = -1;
 
520
 
 
521
        if (have_xkb == -1) {
 
522
                int opcode, error_base, major, minor, xkb_event_base;
 
523
 
 
524
                have_xkb = XkbQueryExtension (dpy,
 
525
                                              &opcode,
 
526
                                              &xkb_event_base,
 
527
                                              &error_base,
 
528
                                              &major,
 
529
                                              &minor)
 
530
                        && XkbUseExtension (dpy, &major, &minor);
 
531
        }
 
532
 
 
533
        return have_xkb;
 
534
}
 
535
 
 
536
gboolean
 
537
key_uses_keycode (const Key *key, guint keycode)
 
538
{
 
539
        if (key->keycodes != NULL) {
 
540
                guint *c;
 
541
 
 
542
                for (c = key->keycodes; *c; ++c) {
 
543
                        if (*c == keycode)
 
544
                                return TRUE;
 
545
                }
 
546
        }
 
547
        return FALSE;
 
548
}
 
549
 
 
550
/* Adapted from _gdk_x11_device_xi2_translate_state()
 
551
 * in gtk+/gdk/x11/gdkdevice-xi2.c */
 
552
static guint
 
553
device_xi2_translate_state (XIModifierState *mods_state,
 
554
                            XIGroupState    *group_state)
 
555
{
 
556
        guint state;
 
557
        gint group;
 
558
 
 
559
        state = (guint) mods_state->base | mods_state->latched | mods_state->locked;
 
560
 
 
561
        group = group_state->base | group_state->latched | group_state->locked;
 
562
        /* FIXME: do we need the XKB complications for this ? */
 
563
        group = CLAMP(group, 0, 3);
 
564
        state |= group << 13;
 
565
 
 
566
        return state;
 
567
}
 
568
 
 
569
gboolean
 
570
match_xi2_key (Key *key, XIDeviceEvent *event)
 
571
{
 
572
        guint keyval;
 
573
        GdkModifierType consumed;
 
574
        gint group;
 
575
        guint keycode, state;
 
576
 
 
577
        if (key == NULL)
 
578
                return FALSE;
 
579
 
 
580
        setup_modifiers ();
 
581
 
 
582
        state = device_xi2_translate_state (&event->mods, &event->group);
 
583
 
 
584
        if (have_xkb (event->display))
 
585
                group = XkbGroupForCoreState (state);
 
586
        else
 
587
                group = (state & GDK_KEY_Mode_switch) ? 1 : 0;
 
588
 
 
589
        keycode = event->detail;
 
590
 
 
591
        /* Check if we find a keysym that matches our current state */
 
592
        if (gdk_keymap_translate_keyboard_state (gdk_keymap_get_default (), keycode,
 
593
                                                 state, group,
 
594
                                                 &keyval, NULL, NULL, &consumed)) {
 
595
                guint key_bit, event_bit;
 
596
                guint lower, upper;
 
597
                guint mask, full_mask;
 
598
 
 
599
                /* HACK: we don't want to use SysRq as a keybinding, so we avoid
 
600
                 * its translation from Alt+Print. */
 
601
                if (keyval == GDK_KEY_Sys_Req &&
 
602
                    (state & GDK_MOD1_MASK) != 0) {
 
603
                        consumed = 0;
 
604
                        keyval = GDK_KEY_Print;
 
605
                }
 
606
 
 
607
                /* The Key structure contains virtual modifiers, whereas
 
608
                 * the XEvent will be using the real modifier, so translate those */
 
609
                key_bit = get_mask_for_key (key->keysym);
 
610
                event_bit = get_mask_for_key (keyval);
 
611
                mask = key->state;
 
612
                full_mask = mask | key_bit;
 
613
                gdk_keymap_map_virtual_modifiers (gdk_keymap_get_default (), &mask);
 
614
                gdk_keymap_map_virtual_modifiers (gdk_keymap_get_default (), &full_mask);
 
615
                mask &= ~(GDK_META_MASK | GDK_SUPER_MASK | GDK_HYPER_MASK);
 
616
                full_mask &= ~(GDK_META_MASK | GDK_SUPER_MASK | GDK_HYPER_MASK);
 
617
 
 
618
                gdk_keyval_convert_case (keyval, &lower, &upper);
 
619
 
 
620
                /* If we are checking against the lower version of the
 
621
                 * keysym, we might need the Shift state for matching,
 
622
                 * so remove it from the consumed modifiers */
 
623
                if (lower == key->keysym || event_bit != 0)
 
624
                        consumed &= ~GDK_SHIFT_MASK;
 
625
 
 
626
                state &= ~consumed & gsd_used_mods;
 
627
 
 
628
                if (key_bit != 0 && event_bit != 0) {
 
629
                        state |= event_bit;
 
630
                        gdk_keymap_map_virtual_modifiers (gdk_keymap_get_default (), &state);
 
631
                        state &= ~(GDK_META_MASK | GDK_SUPER_MASK | GDK_HYPER_MASK);
 
632
                        return state == full_mask;
 
633
                }
 
634
 
 
635
                return (lower == key->keysym || upper == key->keysym) && state == mask;
 
636
        }
 
637
 
 
638
        /* The key we passed doesn't have a keysym, so try with just the keycode */
 
639
        return (key != NULL
 
640
                && key->state == (state & gsd_used_mods)
 
641
                && key_uses_keycode (key, keycode));
 
642
}
 
643
 
 
644
Key *
 
645
parse_key (const char *str)
 
646
{
 
647
        Key *key;
 
648
 
 
649
        if (str == NULL ||
 
650
            *str == '\0' ||
 
651
            g_str_equal (str, "disabled")) {
 
652
                return NULL;
 
653
        }
 
654
 
 
655
        key = g_new0 (Key, 1);
 
656
        gtk_accelerator_parse_with_keycode (str, &key->keysym, &key->keycodes, &key->state);
 
657
        if (key->keysym == 0 &&
 
658
            key->keycodes == NULL &&
 
659
            key->state == 0) {
 
660
                g_free (key);
 
661
                return NULL;
 
662
        }
 
663
 
 
664
        return key;
 
665
}
 
666
 
 
667
void
 
668
free_key (Key *key)
 
669
{
 
670
        if (key == NULL)
 
671
                return;
 
672
        g_free (key->keycodes);
 
673
        g_free (key);
 
674
}
 
675
 
 
676
static void
 
677
grab_button_real (int        deviceid,
 
678
                  gboolean   grab,
 
679
                  GdkWindow *root)
 
680
{
 
681
        XIGrabModifiers mods;
 
682
 
 
683
        mods.modifiers = XIAnyModifier;
 
684
 
 
685
        if (grab) {
 
686
                XIEventMask evmask;
 
687
                unsigned char mask[(XI_LASTEVENT + 7)/8];
 
688
 
 
689
                memset (mask, 0, sizeof (mask));
 
690
                XISetMask (mask, XI_ButtonRelease);
 
691
                XISetMask (mask, XI_ButtonPress);
 
692
 
 
693
                evmask.deviceid = deviceid;
 
694
                evmask.mask_len = sizeof (mask);
 
695
                evmask.mask = mask;
 
696
 
 
697
                XIGrabButton (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()),
 
698
                              deviceid,
 
699
                              XIAnyButton,
 
700
                              GDK_WINDOW_XID (root),
 
701
                              None,
 
702
                              GrabModeAsync,
 
703
                              GrabModeAsync,
 
704
                              False,
 
705
                              &evmask,
 
706
                              1,
 
707
                              &mods);
 
708
        } else {
 
709
                XIUngrabButton (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()),
 
710
                                deviceid,
 
711
                                XIAnyButton,
 
712
                                GDK_WINDOW_XID (root),
 
713
                                1, &mods);
 
714
        }
 
715
}
 
716
 
 
717
void
 
718
grab_button (int      deviceid,
 
719
             gboolean grab,
 
720
             GSList  *screens)
 
721
{
 
722
        GSList *l;
 
723
 
 
724
        for (l = screens; l; l = l->next) {
 
725
                GdkScreen *screen = l->data;
 
726
 
 
727
                grab_button_real (deviceid,
 
728
                                  grab,
 
729
                                  gdk_screen_get_root_window (screen));
 
730
        }
 
731
}