commit 1aa6bd5ebe5e58142af641c34dcb2f02420c5dbc
parent 5f9a5d2068f288f4f5067265bd48fe65561248f4
Author: Enno Boland (Gottox) <gottox@s01.de>
Date:   Tue, 15 Jul 2008 22:12:55 +0200
it works :)
Diffstat:
| config.def.h | | | 71 | +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++------ | 
| config.h | | | 72 | +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++------- | 
| config.mk | | | 2 | +- | 
| svkbd.c | | | 199 | ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++------------------- | 
4 files changed, 283 insertions(+), 61 deletions(-)
diff --git a/config.def.h b/config.def.h
@@ -1,13 +1,72 @@
 static const char font[]            = "-*-terminus-medium-r-normal-*-14-*-*-*-*-*-*-*";
 static const char normbgcolor[]     = "#cccccc";
 static const char normfgcolor[]     = "#000000";
-static const char selbgcolor[]      = "#0066ff";
-static const char selfgcolor[]      = "#ffffff";
-static const char hovbgcolor[]      = "#0066ff";
-static const char hovfgcolor[]      = "#cccccc";
+static const char hovbgcolor[]      = "#ffffff";
+static const char hovfgcolor[]      = "#000000";
+static const char pressbgcolor[]    = "#0000cc";
+static const char pressfgcolor[]    = "#ffffff";
 
 static Key keys[] = {
-	{ 1, XK_Return },
+	{ "1!", XK_1, 1 },
+	{ "2@", XK_2, 1 },
+	{ "3#", XK_3, 1 },
+	{ "4$", XK_4, 1 },
+	{ "5%", XK_5, 1 },
+	{ "6^", XK_6, 1 },
+	{ "7&", XK_7, 1 },
+	{ "8*", XK_8, 1 },
+	{ "9(", XK_9, 1 },
+	{ "0)", XK_0, 1 },
+	{ "-_", XK_minus, 1 },
+	{ "=+", XK_plus, 1 },
+	{ "<-", XK_BackSpace, 2 },
 	{ 0 },
-	{ 1, XK_space },
+	{ "->|", XK_Tab, 1 },
+	{ 0, XK_q, 1 },
+	{ 0, XK_w, 1 },
+	{ 0, XK_e, 1 },
+	{ 0, XK_r, 1 },
+	{ 0, XK_t, 1 },
+	{ 0, XK_y, 1 },
+	{ 0, XK_u, 1 },
+	{ 0, XK_i, 1 },
+	{ 0, XK_o, 1 },
+	{ 0, XK_p, 1 },
+	{ "[", XK_bracketleft, 1 },
+	{ "]", XK_bracketright, 1 },
+	{ "Return", XK_Return, 3 },
+	{ 0 },
+	{ 0, XK_Caps_Lock, 2 },
+	{ 0, XK_a, 1 },
+	{ 0, XK_s, 1 },
+	{ 0, XK_d, 1 },
+	{ 0, XK_f, 1 },
+	{ 0, XK_g, 1 },
+	{ 0, XK_h, 1 },
+	{ 0, XK_j, 1 },
+	{ 0, XK_k, 1 },
+	{ 0, XK_l, 1 },
+	{ ":;", XK_semicolon, 1 },
+	{ "'\"", XK_exclam, 1 },
+	{ "\\|", XK_backslash, 1 },
+	{ 0 },
+	{ 0, XK_Shift_L, 2 },
+	{ "<>|", XK_greater, 1 },
+	{ 0, XK_z, 1 },
+	{ 0, XK_x, 1 },
+	{ 0, XK_c, 1 },
+	{ 0, XK_v, 1 },
+	{ 0, XK_b, 1 },
+	{ 0, XK_n, 1 },
+	{ 0, XK_m, 1 },
+	{ ",", XK_colon, 1 },
+	{ ".", XK_period, 1 },
+	{ "/?", XK_slash, 1 },
+	{ 0, XK_Shift_R, 2 },
+	{ 0 },
+	{ "Ctrl", XK_Control_L, 2 },
+	{ "Alt", XK_Alt_L, 2 },
+	{ "", XK_space, 5 },
+	{ "Alt", XK_Alt_R, 2 },
+	{ "Ctrl", XK_Control_R, 2 },
 };
diff --git a/config.h b/config.h
@@ -1,14 +1,72 @@
 static const char font[]            = "-*-terminus-medium-r-normal-*-14-*-*-*-*-*-*-*";
 static const char normbgcolor[]     = "#cccccc";
 static const char normfgcolor[]     = "#000000";
-static const char selbgcolor[]      = "#0066ff";
-static const char selfgcolor[]      = "#ffffff";
-static const char hovbgcolor[]      = "#0066ff";
-static const char hovfgcolor[]      = "#cccccc";
+static const char hovbgcolor[]      = "#ffffff";
+static const char hovfgcolor[]      = "#000000";
+static const char pressbgcolor[]    = "#0000cc";
+static const char pressfgcolor[]    = "#ffffff";
 
 static Key keys[] = {
-	{ 2, XK_Return },
-	{ 1, XK_space },
+	{ "1!", XK_1, 1 },
+	{ "2@", XK_2, 1 },
+	{ "3#", XK_3, 1 },
+	{ "4$", XK_4, 1 },
+	{ "5%", XK_5, 1 },
+	{ "6^", XK_6, 1 },
+	{ "7&", XK_7, 1 },
+	{ "8*", XK_8, 1 },
+	{ "9(", XK_9, 1 },
+	{ "0)", XK_0, 1 },
+	{ "-_", XK_minus, 1 },
+	{ "=+", XK_plus, 1 },
+	{ "<-", XK_BackSpace, 2 },
 	{ 0 },
-	{ 1, XK_space },
+	{ "->|", XK_Tab, 1 },
+	{ 0, XK_q, 1 },
+	{ 0, XK_w, 1 },
+	{ 0, XK_e, 1 },
+	{ 0, XK_r, 1 },
+	{ 0, XK_t, 1 },
+	{ 0, XK_y, 1 },
+	{ 0, XK_u, 1 },
+	{ 0, XK_i, 1 },
+	{ 0, XK_o, 1 },
+	{ 0, XK_p, 1 },
+	{ "[", XK_bracketleft, 1 },
+	{ "]", XK_bracketright, 1 },
+	{ "Return", XK_Return, 3 },
+	{ 0 },
+	{ 0, XK_Caps_Lock, 2 },
+	{ 0, XK_a, 1 },
+	{ 0, XK_s, 1 },
+	{ 0, XK_d, 1 },
+	{ 0, XK_f, 1 },
+	{ 0, XK_g, 1 },
+	{ 0, XK_h, 1 },
+	{ 0, XK_j, 1 },
+	{ 0, XK_k, 1 },
+	{ 0, XK_l, 1 },
+	{ ":;", XK_semicolon, 1 },
+	{ "'\"", XK_exclam, 1 },
+	{ "\\|", XK_backslash, 1 },
+	{ 0 },
+	{ 0, XK_Shift_L, 2 },
+	{ "<>|", XK_greater, 1 },
+	{ 0, XK_z, 1 },
+	{ 0, XK_x, 1 },
+	{ 0, XK_c, 1 },
+	{ 0, XK_v, 1 },
+	{ 0, XK_b, 1 },
+	{ 0, XK_n, 1 },
+	{ 0, XK_m, 1 },
+	{ ",", XK_colon, 1 },
+	{ ".", XK_period, 1 },
+	{ "/?", XK_slash, 1 },
+	{ 0, XK_Shift_R, 2 },
+	{ 0 },
+	{ "Ctrl", XK_Control_L, 2 },
+	{ "Alt", XK_Alt_L, 2 },
+	{ "", XK_space, 5 },
+	{ "Alt", XK_Alt_R, 2 },
+	{ "Ctrl", XK_Control_R, 2 },
 };
diff --git a/config.mk b/config.mk
@@ -12,7 +12,7 @@ X11LIB = /usr/X11R6/lib
 
 # includes and libs
 INCS = -I. -I/usr/include -I${X11INC}
-LIBS = -L/usr/lib -lc -L${X11LIB} -lX11
+LIBS = -L/usr/lib -lc -L${X11LIB} -lX11 -lXtst
 
 # flags
 CPPFLAGS = -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS}
diff --git a/svkbd.c b/svkbd.c
@@ -27,21 +27,14 @@
 #include <locale.h>
 #include <stdarg.h>
 #include <stdio.h>
-#include <stdlib.h>
 #include <string.h>
-#include <unistd.h>
-#include <sys/select.h>
-#include <sys/types.h>
-#include <sys/wait.h>
-#include <X11/cursorfont.h>
+#include <stdlib.h>
 #include <X11/keysym.h>
 #include <X11/Xatom.h>
 #include <X11/Xlib.h>
 #include <X11/Xproto.h>
 #include <X11/Xutil.h>
-#ifdef XINERAMA
-#include <X11/extensions/Xinerama.h>
-#endif
+#include <X11/extensions/XTest.h>
 
 /* macros */
 #define MAX(a, b)       ((a) > (b) ? (a) : (b))
@@ -53,7 +46,6 @@
 #define MOUSEMASK       (BUTTONMASK|PointerMotionMask)
 #define TAGMASK         ((int)((1LL << LENGTH(tags)) - 1))
 #define TEXTW(x)        (textnw(x, strlen(x)) + dc.font.height)
-#define ISVISIBLE(x)    (x->tags & tagset[seltags])
 
 /* enums */
 enum { ColFG, ColBG, ColLast };
@@ -64,7 +56,7 @@ typedef unsigned long ulong;
 
 typedef struct {
 	ulong norm[ColLast];
-	ulong sel[ColLast];
+	ulong press[ColLast];
 	ulong hover[ColLast];
 	Drawable drawable;
 	GC gc;
@@ -78,21 +70,24 @@ typedef struct {
 } DC; /* draw context */
 
 typedef struct {
-	uint width;
+	char *label;
 	KeySym keysym;
+	uint width;
 	int x, y, w, h;
-	Bool sel;
+	Bool pressed;
 } Key;
 
 /* function declarations */
 static void buttonpress(XEvent *e);
+static void buttonrelease(XEvent *e);
 static void cleanup(void);
 static void configurenotify(XEvent *e);
 static void destroynotify(XEvent *e);
 static void die(const char *errstr, ...);
 static void drawkeyboard(void);
-static void drawkey(Key *k, ulong col[ColLast]);
+static void drawkey(Key *k);
 static void expose(XEvent *e);
+static Key *findkey(int x, int y);
 static ulong getcolor(const char *colstr);
 static void initfont(const char *fontstr);
 static void leavenotify(XEvent *e);
@@ -100,12 +95,14 @@ static void motionnotify(XEvent *e);
 static void run(void);
 static void setup(void);
 static int textnw(const char *text, uint len);
+static void updatekeys();
 
 /* variables */
 static int screen;
 static int wx, wy, ww, wh;
 static void (*handler[LASTEvent]) (XEvent *) = {
 	[ButtonPress] = buttonpress,
+	[ButtonRelease] = buttonrelease,
 	[ConfigureNotify] = configurenotify,
 	[DestroyNotify] = destroynotify,
 	[Expose] = expose,
@@ -116,11 +113,57 @@ static Display *dpy;
 static DC dc;
 static Window root, win;
 static Bool running = True;
+static Key *hover = NULL, *pressed = NULL;
 /* configuration, allows nested code to access above variables */
 #include "config.h"
 
 void
 buttonpress(XEvent *e) {
+	XButtonPressedEvent *ev = &e->xbutton;
+	Key *k;
+
+	if((k = findkey(ev->x, ev->y))) {
+		if(k->pressed && IsModifierKey(k->keysym)) {
+			XTestFakeKeyEvent(dpy, XKeysymToKeycode(dpy, k->keysym), False, 0);
+			k->pressed = 0;
+			pressed = NULL;
+		}
+		else {
+			pressed = k;
+			k->pressed = True;
+			XTestFakeKeyEvent(dpy, XKeysymToKeycode(dpy, k->keysym), True, 0);
+		}
+		drawkey(k);
+		XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, wh, 0, 0);
+	}
+}
+
+void
+buttonrelease(XEvent *e) {
+	int i;
+	XButtonPressedEvent *ev = &e->xbutton;
+	Key *k = findkey(ev->x, ev->y);
+
+	if(pressed && k && !IsModifierKey(k->keysym)) {
+		if(k != pressed) {
+			XTestFakeKeyEvent(dpy, XKeysymToKeycode(dpy, k->keysym), True, 0);
+			k->pressed = 1;
+		}
+		for(i = 0; i < LENGTH(keys); i++) {
+			if(keys[i].pressed && !IsModifierKey(keys[i].keysym)) {
+				XTestFakeKeyEvent(dpy, XKeysymToKeycode(dpy, keys[i].keysym), False, 0);
+				keys[i].pressed = 0;
+			}
+		}
+		for(i = 0; i < LENGTH(keys); i++) {
+			if(keys[i].pressed) {
+				XTestFakeKeyEvent(dpy, XKeysymToKeycode(dpy, keys[i].keysym), False, 0);
+				keys[i].pressed = 0;
+			}
+		}
+		pressed = NULL;
+	}
+	drawkeyboard();
 }
 
 void
@@ -145,7 +188,7 @@ configurenotify(XEvent *e) {
 		wh = ev->height;
 		XFreePixmap(dpy, dc.drawable);
 		dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen));
-		drawkeyboard();
+		updatekeys();
 	}
 }
 
@@ -161,58 +204,50 @@ die(const char *errstr, ...) {
 
 void
 drawkeyboard(void) {
-	int rows, i, j;
-	int x = 0, y = 0, h, base;
+	int i;
 
-	for(i = 0, rows = 1; i < LENGTH(keys); i++)
-		if(keys[i].keysym == 0)
-			rows++;
-	h = wh / rows;
-	for(i = 0; i < LENGTH(keys); i++) {
-		for(j = i, base = 0; j < LENGTH(keys) && keys[j].keysym != 0; j++)
-			base += keys[j].width;
-		for(x = 0; i < LENGTH(keys) && keys[i].keysym != 0; i++) {
-			keys[i].x = x;
-			keys[i].y = y;
-			keys[i].w = keys[i].width * ww / base;
-			keys[i].h = h;
-			x += keys[i].w;
-			printf("%i	%i	%i	%i\n", x, y, keys[i].w, h);
-		}
-		y += h;
-	}
 	for(i = 0; i < LENGTH(keys); i++) {
 		if(keys[i].keysym != 0)
-			drawkey(&keys[i], dc.norm);
+			drawkey(&keys[i]);
 	}
 	XSync(dpy, False);
 	XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, wh, 0, 0);
 }
 
 void
-drawkey(Key *k, ulong col[ColLast]) {
+drawkey(Key *k) {
 	int x, y, h, len;
 	XRectangle r = { k->x, k->y, k->w, k->h};
-	const char *text;
+	const char *l;
+	ulong *col;
 
+	if(k->pressed)
+		col = dc.press;
+	else if(hover == k)
+		col = dc.hover;
+	else
+		col = dc.norm;
 	XSetForeground(dpy, dc.gc, col[ColBG]);
 	XFillRectangles(dpy, dc.drawable, dc.gc, &r, 1);
 	XSetForeground(dpy, dc.gc, col[ColFG]);
 	XDrawRectangles(dpy, dc.drawable, dc.gc, &r, 1);
-	text = XKeysymToString(k->keysym);
-	len = strlen(text);
+	if(k->label)
+		l = k->label;
+	else
+		l = XKeysymToString(k->keysym);
+	len = strlen(l);
 	h = dc.font.ascent + dc.font.descent;
 	y = k->y + (k->h / 2) - (h / 2) + dc.font.ascent;
-	x = k->x + (k->w / 2) - (textnw(text, len) / 2);
+	x = k->x + (k->w / 2) - (textnw(l, len) / 2);
 	if(dc.font.set)
-		XmbDrawString(dpy, dc.drawable, dc.font.set, dc.gc, x, y, text, len);
+		XmbDrawString(dpy, dc.drawable, dc.font.set, dc.gc, x, y, l, len);
 	else
-		XDrawString(dpy, dc.drawable, dc.gc, x, y, text, len);
+		XDrawString(dpy, dc.drawable, dc.gc, x, y, l, len);
 }
 
 void
 destroynotify(XEvent *e) {
-
+	running = False;
 }
 
 void
@@ -223,6 +258,18 @@ expose(XEvent *e) {
 		drawkeyboard();
 }
 
+Key *
+findkey(int x, int y) {
+	int i;
+	
+	for(i = 0; i < LENGTH(keys); i++)
+		if(keys[i].keysym && x > keys[i].x &&
+				x < keys[i].x + keys[i].w &&
+				y > keys[i].y && y < keys[i].y + keys[i].h)
+			return &keys[i];
+	return NULL;
+}
+
 ulong
 getcolor(const char *colstr) {
 	Colormap cmap = DefaultColormap(dpy, screen);
@@ -275,12 +322,29 @@ initfont(const char *fontstr) {
 
 void
 leavenotify(XEvent *e) {
+	Key *h = hover;
 
+	if(!hover)
+		return;
+	hover = NULL;
+	drawkey(h);
+	XCopyArea(dpy, dc.drawable, win, dc.gc, h->x, h->y, h->w, h->h, h->x, h->y);
 }
 
 void
 motionnotify(XEvent *e) {
-
+	XMotionEvent *ev = &e->xmotion;
+	Key *h = findkey(ev->x, ev->y), *oh;
+
+	if(h != hover) {
+		oh = hover;;
+		hover = h;
+		if(oh)
+			drawkey(oh);
+		if(hover)
+			drawkey(hover);
+	}
+	XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, wh, 0, 0);
 }
 
 void
@@ -298,6 +362,9 @@ run(void) {
 
 void
 setup(void) {
+	int i;
+	XWMHints *wmh;
+
 	/* init screen */
 	screen = DefaultScreen(dpy);
 	root = RootWindow(dpy, screen);
@@ -310,18 +377,28 @@ setup(void) {
 	wy = DisplayHeight(dpy, screen) - wh;
 	dc.norm[ColBG] = getcolor(normbgcolor);
 	dc.norm[ColFG] = getcolor(normfgcolor);
-	dc.sel[ColBG] = getcolor(selbgcolor);
-	dc.sel[ColFG] = getcolor(selfgcolor);
+	dc.press[ColBG] = getcolor(pressbgcolor);
+	dc.press[ColFG] = getcolor(pressfgcolor);
 	dc.hover[ColBG] = getcolor(hovbgcolor);
 	dc.hover[ColFG] = getcolor(hovfgcolor);
 	dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen));
 	dc.gc = XCreateGC(dpy, root, 0, 0);
 	if(!dc.font.set)
 		XSetFont(dpy, dc.gc, dc.font.xfont->fid);
+	for(i = 0; i < LENGTH(keys); i++)
+		keys[i].pressed = 0;
 
 	win = XCreateSimpleWindow(dpy, root, wx, wy, ww, wh, 0, dc.norm[ColFG], dc.norm[ColBG]);
-	XSelectInput(dpy, win, StructureNotifyMask | PointerMotionMask | ExposureMask);
+	XSelectInput(dpy, win, StructureNotifyMask|PointerMotionMask|
+			ButtonReleaseMask|ButtonPressMask|ExposureMask|
+			LeaveWindowMask);
+	wmh = XAllocWMHints();
+	wmh->input = False;
+	wmh->flags = InputHint;
+	XSetWMHints(dpy, win, wmh);
+	XFree(wmh);
 	XMapRaised(dpy, win);
+	updatekeys();
 	drawkeyboard();
 }
 
@@ -336,6 +413,34 @@ textnw(const char *text, uint len) {
 	return XTextWidth(dc.font.xfont, text, len);
 }
 
+void
+updatekeys() {
+	int rows, i, j;
+	int x = 0, y = 0, h, base;
+
+	for(i = 0, rows = 1; i < LENGTH(keys); i++)
+		if(keys[i].keysym == 0)
+			rows++;
+	h = wh / rows;
+	for(i = 0; i < LENGTH(keys); i++, rows--) {
+		for(j = i, base = 0; j < LENGTH(keys) && keys[j].keysym != 0; j++)
+			base += keys[j].width;
+		for(x = 0; i < LENGTH(keys) && keys[i].keysym != 0; i++) {
+			keys[i].x = x;
+			keys[i].y = y;
+			keys[i].w = keys[i].width * (ww - 1) / base;
+			if(rows)
+				keys[i].h = h;
+			else
+				keys[i].h = (wh - 1) - y;
+			x += keys[i].w;
+		}
+		if(base != 0)
+			keys[i - 1].w = (ww - 1) - keys[i - 1].x;
+		y += h;
+	}
+}
+
 int
 main(int argc, char *argv[]) {
 	if(argc == 2 && !strcmp("-v", argv[1]))