it works :)

This commit is contained in:
Enno Boland (Gottox) 2008-07-15 22:12:55 +02:00
parent 5f9a5d2068
commit 1aa6bd5ebe
4 changed files with 282 additions and 60 deletions

View File

@ -1,13 +1,72 @@
static const char font[] = "-*-terminus-medium-r-normal-*-14-*-*-*-*-*-*-*"; static const char font[] = "-*-terminus-medium-r-normal-*-14-*-*-*-*-*-*-*";
static const char normbgcolor[] = "#cccccc"; static const char normbgcolor[] = "#cccccc";
static const char normfgcolor[] = "#000000"; static const char normfgcolor[] = "#000000";
static const char selbgcolor[] = "#0066ff"; static const char hovbgcolor[] = "#ffffff";
static const char selfgcolor[] = "#ffffff"; static const char hovfgcolor[] = "#000000";
static const char hovbgcolor[] = "#0066ff"; static const char pressbgcolor[] = "#0000cc";
static const char hovfgcolor[] = "#cccccc"; static const char pressfgcolor[] = "#ffffff";
static Key keys[] = { 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 }, { 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 },
}; };

View File

@ -1,14 +1,72 @@
static const char font[] = "-*-terminus-medium-r-normal-*-14-*-*-*-*-*-*-*"; static const char font[] = "-*-terminus-medium-r-normal-*-14-*-*-*-*-*-*-*";
static const char normbgcolor[] = "#cccccc"; static const char normbgcolor[] = "#cccccc";
static const char normfgcolor[] = "#000000"; static const char normfgcolor[] = "#000000";
static const char selbgcolor[] = "#0066ff"; static const char hovbgcolor[] = "#ffffff";
static const char selfgcolor[] = "#ffffff"; static const char hovfgcolor[] = "#000000";
static const char hovbgcolor[] = "#0066ff"; static const char pressbgcolor[] = "#0000cc";
static const char hovfgcolor[] = "#cccccc"; static const char pressfgcolor[] = "#ffffff";
static Key keys[] = { static Key keys[] = {
{ 2, XK_Return }, { "1!", XK_1, 1 },
{ 1, XK_space }, { "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 }, { 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 },
}; };

View File

@ -12,7 +12,7 @@ X11LIB = /usr/X11R6/lib
# includes and libs # includes and libs
INCS = -I. -I/usr/include -I${X11INC} 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 # flags
CPPFLAGS = -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS} CPPFLAGS = -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS}

197
svkbd.c
View File

@ -27,21 +27,14 @@
#include <locale.h> #include <locale.h>
#include <stdarg.h> #include <stdarg.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h>
#include <string.h> #include <string.h>
#include <unistd.h> #include <stdlib.h>
#include <sys/select.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <X11/cursorfont.h>
#include <X11/keysym.h> #include <X11/keysym.h>
#include <X11/Xatom.h> #include <X11/Xatom.h>
#include <X11/Xlib.h> #include <X11/Xlib.h>
#include <X11/Xproto.h> #include <X11/Xproto.h>
#include <X11/Xutil.h> #include <X11/Xutil.h>
#ifdef XINERAMA #include <X11/extensions/XTest.h>
#include <X11/extensions/Xinerama.h>
#endif
/* macros */ /* macros */
#define MAX(a, b) ((a) > (b) ? (a) : (b)) #define MAX(a, b) ((a) > (b) ? (a) : (b))
@ -53,7 +46,6 @@
#define MOUSEMASK (BUTTONMASK|PointerMotionMask) #define MOUSEMASK (BUTTONMASK|PointerMotionMask)
#define TAGMASK ((int)((1LL << LENGTH(tags)) - 1)) #define TAGMASK ((int)((1LL << LENGTH(tags)) - 1))
#define TEXTW(x) (textnw(x, strlen(x)) + dc.font.height) #define TEXTW(x) (textnw(x, strlen(x)) + dc.font.height)
#define ISVISIBLE(x) (x->tags & tagset[seltags])
/* enums */ /* enums */
enum { ColFG, ColBG, ColLast }; enum { ColFG, ColBG, ColLast };
@ -64,7 +56,7 @@ typedef unsigned long ulong;
typedef struct { typedef struct {
ulong norm[ColLast]; ulong norm[ColLast];
ulong sel[ColLast]; ulong press[ColLast];
ulong hover[ColLast]; ulong hover[ColLast];
Drawable drawable; Drawable drawable;
GC gc; GC gc;
@ -78,21 +70,24 @@ typedef struct {
} DC; /* draw context */ } DC; /* draw context */
typedef struct { typedef struct {
uint width; char *label;
KeySym keysym; KeySym keysym;
uint width;
int x, y, w, h; int x, y, w, h;
Bool sel; Bool pressed;
} Key; } Key;
/* function declarations */ /* function declarations */
static void buttonpress(XEvent *e); static void buttonpress(XEvent *e);
static void buttonrelease(XEvent *e);
static void cleanup(void); static void cleanup(void);
static void configurenotify(XEvent *e); static void configurenotify(XEvent *e);
static void destroynotify(XEvent *e); static void destroynotify(XEvent *e);
static void die(const char *errstr, ...); static void die(const char *errstr, ...);
static void drawkeyboard(void); static void drawkeyboard(void);
static void drawkey(Key *k, ulong col[ColLast]); static void drawkey(Key *k);
static void expose(XEvent *e); static void expose(XEvent *e);
static Key *findkey(int x, int y);
static ulong getcolor(const char *colstr); static ulong getcolor(const char *colstr);
static void initfont(const char *fontstr); static void initfont(const char *fontstr);
static void leavenotify(XEvent *e); static void leavenotify(XEvent *e);
@ -100,12 +95,14 @@ static void motionnotify(XEvent *e);
static void run(void); static void run(void);
static void setup(void); static void setup(void);
static int textnw(const char *text, uint len); static int textnw(const char *text, uint len);
static void updatekeys();
/* variables */ /* variables */
static int screen; static int screen;
static int wx, wy, ww, wh; static int wx, wy, ww, wh;
static void (*handler[LASTEvent]) (XEvent *) = { static void (*handler[LASTEvent]) (XEvent *) = {
[ButtonPress] = buttonpress, [ButtonPress] = buttonpress,
[ButtonRelease] = buttonrelease,
[ConfigureNotify] = configurenotify, [ConfigureNotify] = configurenotify,
[DestroyNotify] = destroynotify, [DestroyNotify] = destroynotify,
[Expose] = expose, [Expose] = expose,
@ -116,11 +113,57 @@ static Display *dpy;
static DC dc; static DC dc;
static Window root, win; static Window root, win;
static Bool running = True; static Bool running = True;
static Key *hover = NULL, *pressed = NULL;
/* configuration, allows nested code to access above variables */ /* configuration, allows nested code to access above variables */
#include "config.h" #include "config.h"
void void
buttonpress(XEvent *e) { 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 void
@ -145,7 +188,7 @@ configurenotify(XEvent *e) {
wh = ev->height; wh = ev->height;
XFreePixmap(dpy, dc.drawable); XFreePixmap(dpy, dc.drawable);
dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen)); dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen));
drawkeyboard(); updatekeys();
} }
} }
@ -161,58 +204,50 @@ die(const char *errstr, ...) {
void void
drawkeyboard(void) { drawkeyboard(void) {
int rows, i, j; int i;
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++) {
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++) { for(i = 0; i < LENGTH(keys); i++) {
if(keys[i].keysym != 0) if(keys[i].keysym != 0)
drawkey(&keys[i], dc.norm); drawkey(&keys[i]);
} }
XSync(dpy, False); XSync(dpy, False);
XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, wh, 0, 0); XCopyArea(dpy, dc.drawable, win, dc.gc, 0, 0, ww, wh, 0, 0);
} }
void void
drawkey(Key *k, ulong col[ColLast]) { drawkey(Key *k) {
int x, y, h, len; int x, y, h, len;
XRectangle r = { k->x, k->y, k->w, k->h}; 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]); XSetForeground(dpy, dc.gc, col[ColBG]);
XFillRectangles(dpy, dc.drawable, dc.gc, &r, 1); XFillRectangles(dpy, dc.drawable, dc.gc, &r, 1);
XSetForeground(dpy, dc.gc, col[ColFG]); XSetForeground(dpy, dc.gc, col[ColFG]);
XDrawRectangles(dpy, dc.drawable, dc.gc, &r, 1); XDrawRectangles(dpy, dc.drawable, dc.gc, &r, 1);
text = XKeysymToString(k->keysym); if(k->label)
len = strlen(text); l = k->label;
else
l = XKeysymToString(k->keysym);
len = strlen(l);
h = dc.font.ascent + dc.font.descent; h = dc.font.ascent + dc.font.descent;
y = k->y + (k->h / 2) - (h / 2) + dc.font.ascent; 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) 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 else
XDrawString(dpy, dc.drawable, dc.gc, x, y, text, len); XDrawString(dpy, dc.drawable, dc.gc, x, y, l, len);
} }
void void
destroynotify(XEvent *e) { destroynotify(XEvent *e) {
running = False;
} }
void void
@ -223,6 +258,18 @@ expose(XEvent *e) {
drawkeyboard(); 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 ulong
getcolor(const char *colstr) { getcolor(const char *colstr) {
Colormap cmap = DefaultColormap(dpy, screen); Colormap cmap = DefaultColormap(dpy, screen);
@ -275,12 +322,29 @@ initfont(const char *fontstr) {
void void
leavenotify(XEvent *e) { 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 void
motionnotify(XEvent *e) { 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 void
@ -298,6 +362,9 @@ run(void) {
void void
setup(void) { setup(void) {
int i;
XWMHints *wmh;
/* init screen */ /* init screen */
screen = DefaultScreen(dpy); screen = DefaultScreen(dpy);
root = RootWindow(dpy, screen); root = RootWindow(dpy, screen);
@ -310,18 +377,28 @@ setup(void) {
wy = DisplayHeight(dpy, screen) - wh; wy = DisplayHeight(dpy, screen) - wh;
dc.norm[ColBG] = getcolor(normbgcolor); dc.norm[ColBG] = getcolor(normbgcolor);
dc.norm[ColFG] = getcolor(normfgcolor); dc.norm[ColFG] = getcolor(normfgcolor);
dc.sel[ColBG] = getcolor(selbgcolor); dc.press[ColBG] = getcolor(pressbgcolor);
dc.sel[ColFG] = getcolor(selfgcolor); dc.press[ColFG] = getcolor(pressfgcolor);
dc.hover[ColBG] = getcolor(hovbgcolor); dc.hover[ColBG] = getcolor(hovbgcolor);
dc.hover[ColFG] = getcolor(hovfgcolor); dc.hover[ColFG] = getcolor(hovfgcolor);
dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen)); dc.drawable = XCreatePixmap(dpy, root, ww, wh, DefaultDepth(dpy, screen));
dc.gc = XCreateGC(dpy, root, 0, 0); dc.gc = XCreateGC(dpy, root, 0, 0);
if(!dc.font.set) if(!dc.font.set)
XSetFont(dpy, dc.gc, dc.font.xfont->fid); 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]); 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); XMapRaised(dpy, win);
updatekeys();
drawkeyboard(); drawkeyboard();
} }
@ -336,6 +413,34 @@ textnw(const char *text, uint len) {
return XTextWidth(dc.font.xfont, text, 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 int
main(int argc, char *argv[]) { main(int argc, char *argv[]) {
if(argc == 2 && !strcmp("-v", argv[1])) if(argc == 2 && !strcmp("-v", argv[1]))