it works :)
This commit is contained in:
parent
5f9a5d2068
commit
1aa6bd5ebe
71
config.def.h
71
config.def.h
@ -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 },
|
||||||
};
|
};
|
||||||
|
72
config.h
72
config.h
@ -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 },
|
||||||
};
|
};
|
||||||
|
@ -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
197
svkbd.c
@ -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]))
|
||||||
|
Loading…
Reference in New Issue
Block a user