~ubuntu-branches/debian/sid/openbox/sid

« back to all changes in this revision

Viewing changes to openbox/modkeys.c

  • Committer: Bazaar Package Importer
  • Author(s): Nico Golde, Nico Golde, Eugenio Paolantonio
  • Date: 2011-10-03 22:59:30 UTC
  • mfrom: (1.1.11 upstream)
  • Revision ID: james.westby@ubuntu.com-20111003225930-tdvyax5tx63dyoez
Tags: 3.5.0-1
[Nico Golde]
* New upstream release (Closes: #638783).
  - Fix crashes in the menu code (Closes: #563891).
* Add Brazilian translation to openbox.desktop,
  thanks Sérgio Cipolla (Closes: #627912).
* Remove 06_fix_swap_byte_order.patch, applied upstream.
* Bump debhelper dependency to >= 7.0.50~ due to override.
* Remove CHANGELOG from openbox.docs to prevent double installation.
* Add 02_fix_freedesktop_compliance.dpatch desktop file to
  /usr/share/applications.

[Eugenio Paolantonio]
* debian/patches:
  - Disabled 03_place_windows_in_quadrants.patch
  - Updated 01_rc.xml.patch and 06_fix_swap_byte_order.patch
  - Removed 04_fix_ftbfs_no-add-needed.patch and 20_24bits_support.patch
* debian/control:
  - Added myself to the Uploaders.
  - Build-Depends: removed libxau-dev, libxft-dev and python-xdg;
    added libimlib2-dev
  - openbox Suggests: added python-xdg
  - libobrender21 renamed to libobrender27
  - libobparser21 renamed to libobt0
* debian/rules:
  - Rewrote using a simpler debhelper syntax
  - Moved the install pass to openbox.install
* debian/*.{install,links,dirs}:
  - Updated.
* debian/openbox.xsession:
  - Removed. Openbox now ships it by default.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
/* -*- indent-tabs-mode: nil; tab-width: 4; c-basic-offset: 4; -*-
2
 
 
3
 
   modkeys.c for the Openbox window manager
4
 
   Copyright (c) 2007        Dana Jansens
5
 
 
6
 
   This program is free software; you can redistribute it and/or modify
7
 
   it under the terms of the GNU General Public License as published by
8
 
   the Free Software Foundation; either version 2 of the License, or
9
 
   (at your option) any later version.
10
 
 
11
 
   This program is distributed in the hope that it will be useful,
12
 
   but WITHOUT ANY WARRANTY; without even the implied warranty of
13
 
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
 
   GNU General Public License for more details.
15
 
 
16
 
   See the COPYING file for a copy of the GNU General Public License.
17
 
*/
18
 
 
19
 
#include "modkeys.h"
20
 
#include "openbox.h"
21
 
 
22
 
#include <X11/Xlib.h>
23
 
#include <X11/keysym.h>
24
 
 
25
 
/* These masks are constants and the modifier keys are bound to them as
26
 
   anyone sees fit:
27
 
        ShiftMask (1<<0), LockMask (1<<1), ControlMask (1<<2), Mod1Mask (1<<3),
28
 
        Mod2Mask (1<<4), Mod3Mask (1<<5), Mod4Mask (1<<6), Mod5Mask (1<<7)
29
 
*/
30
 
#define NUM_MASKS 8
31
 
#define ALL_MASKS 0xff /* an or'ing of all 8 keyboard masks */
32
 
 
33
 
/* Get the bitflag for the n'th modifier mask */
34
 
#define nth_mask(n) (1 << n)
35
 
 
36
 
static void set_modkey_mask(guchar mask, KeySym sym);
37
 
 
38
 
static XModifierKeymap *modmap;
39
 
static KeySym *keymap;
40
 
static gint min_keycode, max_keycode, keysyms_per_keycode;
41
 
/* This is a bitmask of the different masks for each modifier key */
42
 
static guchar modkeys_keys[OB_MODKEY_NUM_KEYS];
43
 
 
44
 
static gboolean alt_l = FALSE;
45
 
static gboolean meta_l = FALSE;
46
 
static gboolean super_l = FALSE;
47
 
static gboolean hyper_l = FALSE;
48
 
 
49
 
void modkeys_startup(gboolean reconfigure)
50
 
{
51
 
    gint i, j, k;
52
 
 
53
 
    /* reset the keys to not be bound to any masks */
54
 
    for (i = 0; i < OB_MODKEY_NUM_KEYS; ++i)
55
 
        modkeys_keys[i] = 0;
56
 
 
57
 
    modmap = XGetModifierMapping(ob_display);
58
 
    g_assert(modmap->max_keypermod > 0);
59
 
 
60
 
    XDisplayKeycodes(ob_display, &min_keycode, &max_keycode);
61
 
    keymap = XGetKeyboardMapping(ob_display, min_keycode,
62
 
                                 max_keycode - min_keycode + 1,
63
 
                                 &keysyms_per_keycode);
64
 
 
65
 
    alt_l = meta_l = super_l = hyper_l = FALSE;
66
 
 
67
 
    /* go through each of the modifier masks (eg ShiftMask, CapsMask...) */
68
 
    for (i = 0; i < NUM_MASKS; ++i) {
69
 
        /* go through each keycode that is bound to the mask */
70
 
        for (j = 0; j < modmap->max_keypermod; ++j) {
71
 
            KeySym sym;
72
 
            /* get a keycode that is bound to the mask (i) */
73
 
            KeyCode keycode = modmap->modifiermap[i*modmap->max_keypermod + j];
74
 
            if (keycode) {
75
 
                /* go through each keysym bound to the given keycode */
76
 
                for (k = 0; k < keysyms_per_keycode; ++k) {
77
 
                    sym = keymap[(keycode-min_keycode) * keysyms_per_keycode +
78
 
                                 k];
79
 
                    if (sym != NoSymbol) {
80
 
                        /* bind the key to the mask (e.g. Alt_L => Mod1Mask) */
81
 
                        set_modkey_mask(nth_mask(i), sym);
82
 
                    }
83
 
                }
84
 
            }
85
 
        }
86
 
    }
87
 
 
88
 
    /* CapsLock, Shift, and Control are special and hard-coded */
89
 
    modkeys_keys[OB_MODKEY_KEY_CAPSLOCK] = LockMask;
90
 
    modkeys_keys[OB_MODKEY_KEY_SHIFT] = ShiftMask;
91
 
    modkeys_keys[OB_MODKEY_KEY_CONTROL] = ControlMask;
92
 
}
93
 
 
94
 
void modkeys_shutdown(gboolean reconfigure)
95
 
{
96
 
    XFreeModifiermap(modmap);
97
 
    XFree(keymap);
98
 
}
99
 
 
100
 
guint modkeys_keycode_to_mask(guint keycode)
101
 
{
102
 
    gint i, j;
103
 
    guint mask = 0;
104
 
 
105
 
    if (keycode == NoSymbol) return 0;
106
 
 
107
 
    /* go through each of the modifier masks (eg ShiftMask, CapsMask...) */
108
 
    for (i = 0; i < NUM_MASKS; ++i) {
109
 
        /* go through each keycode that is bound to the mask */
110
 
        for (j = 0; j < modmap->max_keypermod; ++j) {
111
 
            /* compare with a keycode that is bound to the mask (i) */
112
 
            if (modmap->modifiermap[i*modmap->max_keypermod + j] == keycode)
113
 
                mask |= nth_mask(i);
114
 
        }
115
 
    }
116
 
    return mask;
117
 
}
118
 
 
119
 
guint modkeys_only_modifier_masks(guint mask)
120
 
{
121
 
    mask &= ALL_MASKS;
122
 
    /* strip off these lock keys. they shouldn't affect key bindings */
123
 
    mask &= ~LockMask; /* use the LockMask, not what capslock is bound to,
124
 
                          because you could bind it to something else and it
125
 
                          should work as that modifier then. i think capslock
126
 
                          is weird in xkb. */
127
 
    mask &= ~modkeys_key_to_mask(OB_MODKEY_KEY_NUMLOCK);
128
 
    mask &= ~modkeys_key_to_mask(OB_MODKEY_KEY_SCROLLLOCK);
129
 
    return mask;
130
 
}
131
 
 
132
 
guint modkeys_key_to_mask(ObModkeysKey key)
133
 
{
134
 
    return modkeys_keys[key];
135
 
}
136
 
 
137
 
static void set_modkey_mask(guchar mask, KeySym sym)
138
 
{
139
 
    /* find what key this is, and bind it to the mask */
140
 
 
141
 
    if (sym == XK_Num_Lock)
142
 
        modkeys_keys[OB_MODKEY_KEY_NUMLOCK] |= mask;
143
 
    else if (sym == XK_Scroll_Lock)
144
 
        modkeys_keys[OB_MODKEY_KEY_SCROLLLOCK] |= mask;
145
 
 
146
 
    else if (sym == XK_Super_L && super_l)
147
 
        modkeys_keys[OB_MODKEY_KEY_SUPER] |= mask;
148
 
    else if (sym == XK_Super_L && !super_l)
149
 
        /* left takes precident over right, so erase any masks the right
150
 
           key may have set */
151
 
        modkeys_keys[OB_MODKEY_KEY_SUPER] = mask, super_l = TRUE;
152
 
    else if (sym == XK_Super_R && !super_l)
153
 
        modkeys_keys[OB_MODKEY_KEY_SUPER] |= mask;
154
 
 
155
 
    else if (sym == XK_Hyper_L && hyper_l)
156
 
        modkeys_keys[OB_MODKEY_KEY_HYPER] |= mask;
157
 
    else if (sym == XK_Hyper_L && !hyper_l)
158
 
        modkeys_keys[OB_MODKEY_KEY_HYPER] = mask, hyper_l = TRUE;
159
 
    else if (sym == XK_Hyper_R && !hyper_l)
160
 
        modkeys_keys[OB_MODKEY_KEY_HYPER] |= mask;
161
 
 
162
 
    else if (sym == XK_Alt_L && alt_l)
163
 
        modkeys_keys[OB_MODKEY_KEY_ALT] |= mask;
164
 
    else if (sym == XK_Alt_L && !alt_l)
165
 
        modkeys_keys[OB_MODKEY_KEY_ALT] = mask, alt_l = TRUE;
166
 
    else if (sym == XK_Alt_R && !alt_l)
167
 
        modkeys_keys[OB_MODKEY_KEY_ALT] |= mask;
168
 
 
169
 
    else if (sym == XK_Meta_L && meta_l)
170
 
        modkeys_keys[OB_MODKEY_KEY_META] |= mask;
171
 
    else if (sym == XK_Meta_L && !meta_l)
172
 
        modkeys_keys[OB_MODKEY_KEY_META] = mask, meta_l = TRUE;
173
 
    else if (sym == XK_Meta_R && !meta_l)
174
 
        modkeys_keys[OB_MODKEY_KEY_META] |= mask;
175
 
 
176
 
    /* CapsLock, Shift, and Control are special and hard-coded */
177
 
}
178
 
 
179
 
KeyCode* modkeys_sym_to_code(KeySym sym)
180
 
{
181
 
    KeyCode *ret;
182
 
    gint i, j, n;
183
 
 
184
 
    ret = g_new(KeyCode, 1);
185
 
    n = 0;
186
 
    ret[n] = 0;
187
 
 
188
 
    /* go through each keycode and look for the keysym */
189
 
    for (i = min_keycode; i <= max_keycode; ++i)
190
 
        for (j = 0; j < keysyms_per_keycode; ++j)
191
 
            if (sym == keymap[(i-min_keycode) * keysyms_per_keycode + j]) {
192
 
                ret = g_renew(KeyCode, ret, ++n + 1);
193
 
                ret[n-1] = i;
194
 
                ret[n] = 0;
195
 
            }
196
 
    return ret;
197
 
}