Solution 1 I can recommend a piece of code for sending keyboard events which I use myself to simulate key presses. It is based on XSendEvent.
#include <X11/Xlib.h>
#include <X11/keysym.h>
#define MOD_ALT 0x8
#define MOD_CONTROL 0x4
#define MOD_CONTROL_ALT 0xc
#define MOD_SHIFT_CONTROL 0x5
#define MOD_SHIFT_ALT 0x9
XKeyEvent createKeyEvent(Display *display, Window &win,
Window &winRoot, bool press,
int keycode, int modifiers)
{
XKeyEvent event;
event.display = display;
event.window = win;
event.root = winRoot;
event.subwindow = None;
event.time = CurrentTime;
event.x = 1;
event.y = 1;
event.x_root = 1;
event.y_root = 1;
event.same_screen = True;
event.keycode = XKeysymToKeycode(display, keycode);
event.state = modifiers;
if(press)
event.type = KeyPress;
else
event.type = KeyRelease;
return event;
}
void pressKey(Display* display, Window &win_focus, Window &win_root,
KeySym key, int modifiers)
{
XKeyEvent event = createKeyEvent(display, win_focus, win_root, true, key, modifiers);
XSendEvent(event.display, event.window, True, KeyPressMask, (XEvent *)&event);
event = createKeyEvent(display, win_focus, win_root, false, key, modifiers);
XSendEvent(event.display, event.window, True, KeyPressMask, (XEvent *)&event);
}
Then, you can use it to send a key event (in this case Alt+W) as follows:
Display *display = XOpenDisplay(0);
if (display == NULL)
{
printf("Null display!\n");
return 1;
}
Window win_root = XDefaultRootWindow(display);
Window win_focus;
int revert;
XGetInputFocus(display, &win_focus, &revert);
pressKey(display, win_focus, win_root, XK_w, MOD_ALT);
if (display)
XCloseDisplay(display);
Solution 2 A similar behavior can be achieved using the XTest library. It operates on a "higher level" though and for instance the window manager will pick up those keypresses as well. The pressKey function for XTest can look like this (not very clean code):
void pressKey(Display* display, KeySym modifier1, KeySym modifier2, KeySym key)
{
// Release every other modifier
KeyCode keycodec = XKeysymToKeycode(display, XK_Control_L);
KeyCode keycodea = XKeysymToKeycode(display, XK_Alt_L);
KeyCode keycodes = XKeysymToKeycode(display, XK_Shift_L);
XTestFakeKeyEvent(display, keycodec, False, 0); // key release event
XTestFakeKeyEvent(display, keycodea, False, 0); // key release event
XTestFakeKeyEvent(display, keycodes, False, 0); // key release event
XFlush(display);
// Press the actual keys
KeyCode keycode1 = XKeysymToKeycode(display, modifier1);
KeyCode keycode2 = XKeysymToKeycode(display, modifier2);
KeyCode keycode3 = XKeysymToKeycode(display, key);
if (keycode1)
XTestFakeKeyEvent(display, keycode1, True, 0); // key press event
if (keycode2)
XTestFakeKeyEvent(display, keycode2, True, 0); // key press event
if (keycode3)
{
XTestFakeKeyEvent(display, keycode3, True, 0); // key press event
XTestFakeKeyEvent(display, keycode3, False, 0); // key release event
}
if (keycode2)
XTestFakeKeyEvent(display, keycode2, False, 0); // key release event
if (keycode1)
XTestFakeKeyEvent(display, keycode1, False, 0); // key release event
XFlush(display);
}
Solution 3 Another solution is to just script your way out with such tools as xdotool. Basically, most of what you described can be achieved with a bash script. Also xdotool source code is a great source of information on how to achieve all you want.