d83b160217
If another application has already set a grab on these keys then the call would fail (BadAccess) and wmix stop. With the X error handler, we can display a warning to user and continue anyway. Signed-off-by: Christophe CURIS <christophe.curis@free.fr>
223 lines
6.3 KiB
C
223 lines
6.3 KiB
C
/* WMix -- a mixer using the OSS mixer API.
|
|
* Copyright (C) 2014 Christophe CURIS for the WindowMaker Team
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
/*
|
|
* mmkeys.c: functions related to grabing the Multimedia Keys on keyboard
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
#include <X11/Xproto.h>
|
|
#include <X11/keysym.h>
|
|
#include <X11/XF86keysym.h>
|
|
|
|
#include "include/common.h"
|
|
#include "include/config.h"
|
|
#include "include/mmkeys.h"
|
|
|
|
|
|
/* The global configuration */
|
|
struct multimedia_keys mmkeys;
|
|
|
|
/* The list of keys we're interrested in */
|
|
static const struct {
|
|
KeySym symbol;
|
|
KeyCode *store;
|
|
const char *name;
|
|
} key_list[] = {
|
|
{ XF86XK_AudioRaiseVolume, &mmkeys.raise_volume, "AudioRaiseVolume" },
|
|
{ XF86XK_AudioLowerVolume, &mmkeys.lower_volume, "AudioLowerVolume" },
|
|
{ XF86XK_AudioMute, &mmkeys.mute, "AudioMute" }
|
|
};
|
|
|
|
/* The modifiers that should not have impact on the key grabbed */
|
|
static const struct {
|
|
KeySym symbol;
|
|
const char *name;
|
|
} modifier_symbol[] = {
|
|
{ XK_Caps_Lock, "CapsLock" },
|
|
{ XK_Num_Lock, "NumLock" }
|
|
};
|
|
|
|
typedef struct {
|
|
int count;
|
|
unsigned int list[1 << lengthof(modifier_symbol)];
|
|
} modifier_masks;
|
|
|
|
/* The structure to track grab installation for errors */
|
|
static struct mmkey_track {
|
|
XErrorHandler previous_handler;
|
|
struct {
|
|
const char *key_name;
|
|
unsigned long serial[1 << lengthof(modifier_symbol)];
|
|
Bool displayed;
|
|
} request[lengthof(key_list)];
|
|
} *track_install = NULL;
|
|
|
|
/* Local functions */
|
|
static void mmkey_build_modifier_list(Display *display, modifier_masks *result);
|
|
static int mmkey_catch_grab_error(Display *display, XErrorEvent *event);
|
|
|
|
|
|
/*
|
|
* Grab the multimedia keys on the X server
|
|
*
|
|
* That basically means that whenever these keys are pressed
|
|
* the events will be sent to us instead of the application
|
|
* that has current focus.
|
|
*/
|
|
void mmkey_install(Display *display)
|
|
{
|
|
modifier_masks mod_masks;
|
|
struct mmkey_track install_info;
|
|
Window root_window;
|
|
int i, j;
|
|
|
|
mmkey_build_modifier_list(display, &mod_masks);
|
|
|
|
root_window = DefaultRootWindow(display);
|
|
|
|
memset(&install_info, 0, sizeof(install_info));
|
|
install_info.previous_handler = XSetErrorHandler(mmkey_catch_grab_error);
|
|
track_install = &install_info;
|
|
for (i = 0; i < lengthof(key_list); i++) {
|
|
KeyCode key;
|
|
|
|
key = XKeysymToKeycode(display, key_list[i].symbol);
|
|
*(key_list[i].store) = key;
|
|
|
|
if (key == None)
|
|
continue;
|
|
|
|
install_info.request[i].key_name = key_list[i].name;
|
|
install_info.request[i].displayed = False;
|
|
for (j = 0; j < mod_masks.count; j++) {
|
|
install_info.request[i].serial[j] = NextRequest(display);
|
|
XGrabKey(display, key, mod_masks.list[j], root_window,
|
|
False, GrabModeAsync, GrabModeAsync);
|
|
}
|
|
if (config.verbose)
|
|
printf("Found multimedia key: %s\n", key_list[i].name);
|
|
}
|
|
|
|
/* The grab may fail, so make sure it is reported now */
|
|
XSync(display, False);
|
|
XSetErrorHandler(install_info.previous_handler);
|
|
track_install = NULL;
|
|
}
|
|
|
|
/*
|
|
* Build the list of bit-masks for all the modifiers we want to not have impact on our grab
|
|
*/
|
|
static void mmkey_build_modifier_list(Display *display, modifier_masks *result)
|
|
{
|
|
XModifierKeymap *mods;
|
|
KeyCode mod_code[lengthof(modifier_symbol)];
|
|
unsigned int mod_mask[lengthof(modifier_symbol)];
|
|
char buffer[256];
|
|
int nb_modifiers;
|
|
int i, j, k;
|
|
|
|
/* Get the bitmask associated with the modifiers */
|
|
for (i = 0; i < lengthof(modifier_symbol); i++) {
|
|
mod_code[i] = XKeysymToKeycode(display, modifier_symbol[i].symbol);
|
|
mod_mask[i] = 0L;
|
|
}
|
|
|
|
mods = XGetModifierMapping(display);
|
|
for (i = 0; i < 8; i++) {
|
|
for (j = 0; j < mods->max_keypermod; j++) {
|
|
KeyCode key_mod;
|
|
|
|
key_mod = mods->modifiermap[i * mods->max_keypermod + j];
|
|
for (k = 0; k < lengthof(mod_code); k++) {
|
|
if ((mod_code[k] != None) && (key_mod == mod_code[k]))
|
|
mod_mask[k] |= 1 << i;
|
|
}
|
|
}
|
|
}
|
|
XFreeModifiermap(mods);
|
|
|
|
/* Count the number of modifiers found (and display the list to the user) */
|
|
if (config.verbose)
|
|
strcpy(buffer, "Found key modifiers: ");
|
|
|
|
nb_modifiers = 0;
|
|
for (i = 0; i < lengthof(modifier_symbol); i++) {
|
|
if (mod_mask[i] != 0) {
|
|
if (config.verbose) {
|
|
if (nb_modifiers > 0)
|
|
strcat(buffer, ", ");
|
|
strcat(buffer, modifier_symbol[i].name);
|
|
}
|
|
nb_modifiers++;
|
|
}
|
|
}
|
|
if (config.verbose) {
|
|
if (nb_modifiers == 0)
|
|
strcat(buffer, "None");
|
|
puts(buffer);
|
|
}
|
|
|
|
/* Build the list of possible combinations of modifiers */
|
|
result->count = 1 << nb_modifiers;
|
|
for (i = 0; i < lengthof(result->list); i++)
|
|
result->list[i] = 0L;
|
|
k = 1;
|
|
for (i = 0; i < lengthof(mod_mask); i++) {
|
|
if (mod_mask[i] != 0) {
|
|
for (j = 1; j < result->count; j++)
|
|
if (j & k)
|
|
result->list[j] |= mod_mask[i];
|
|
|
|
k <<= 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Callback when X11 reports an error
|
|
*
|
|
* We only track errors from XGrabKey and display them to user, instead of
|
|
* letting the default error handler exit
|
|
*/
|
|
static int mmkey_catch_grab_error(Display *display, XErrorEvent *event)
|
|
{
|
|
int i, j;
|
|
|
|
if ((event->error_code == BadAccess) && (event->request_code == X_GrabKey)) {
|
|
for (i = 0; i < lengthof(track_install->request); i++) {
|
|
for (j = 0; j < lengthof(track_install->request[i].serial); j++) {
|
|
if (track_install->request[i].serial[j] == 0L)
|
|
break;
|
|
if (event->serial == track_install->request[i].serial[j]) {
|
|
if (!track_install->request[i].displayed) {
|
|
fprintf(stderr, "wmix:warning: could not grab key %s, is another application using it?\n",
|
|
track_install->request[i].key_name);
|
|
track_install->request[i].displayed = True;
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* That's not an XGrabKey known issue, let the default handler manage this */
|
|
return track_install->previous_handler(display, event);
|
|
}
|