dwm-meillo
changeset 1:f10194d4b76d
added gridmenu
author | Anselm R. Garbe <garbeam@wmii.de> |
---|---|
date | Mon, 10 Jul 2006 16:49:43 +0200 (2006-07-10) |
parents | 491f34c11291 |
children | a79188fe4a40 |
files | Makefile gridmenu.1 gridmenu.c |
diffstat | 3 files changed, 617 insertions(+), 3 deletions(-) [+] |
line diff
1.1 --- a/Makefile Mon Jul 10 16:38:18 2006 +0200 1.2 +++ b/Makefile Mon Jul 10 16:49:43 2006 +0200 1.3 @@ -3,12 +3,21 @@ 1.4 1.5 include config.mk 1.6 1.7 -SRC = wm.c 1.8 -OBJ = ${SRC:.c=.o} 1.9 +SRC = wm.c 1.10 +OBJ = ${SRC:.c=.o} 1.11 +MAN = gridwm.1 1.12 +BIN = gridwm gridmenu 1.13 1.14 -all: gridwm 1.15 +all: config gridwm 1.16 @echo finished 1.17 1.18 +config: 1.19 + @echo gridwm build options: 1.20 + @echo "LIBS = ${LIBS}" 1.21 + @echo "CFLAGS = ${CFLAGS}" 1.22 + @echo "LDFLAGS = ${LDFLAGS}" 1.23 + @echo "CC = ${CC}" 1.24 + 1.25 .c.o: 1.26 @echo CC $< 1.27 @${CC} -c ${CFLAGS} $< 1.28 @@ -21,3 +30,26 @@ 1.29 1.30 clean: 1.31 rm -f gridwm *.o 1.32 + 1.33 +dist: clean 1.34 + mkdir -p gridwm-${VERSION} 1.35 + cp -R Makefile README LICENSE config.mk ${SRC} ${MAN} gridwm-${VERSION} 1.36 + tar -cf gridwm-${VERSION}.tar gridwm-${VERSION} 1.37 + gzip gridwm-${VERSION}.tar 1.38 + rm -rf gridwm-${VERSION} 1.39 + 1.40 +install: all 1.41 + @mkdir -p ${DESTDIR}${PREFIX}/bin 1.42 + @cp -f ${BIN} ${DESTDIR}${PREFIX}/bin 1.43 + @echo installed executable files to ${DESTDIR}${PREFIX}/bin 1.44 + @mkdir -p ${DESTDIR}${MANPREFIX}/man1 1.45 + @cp -f ${MAN1} ${DESTDIR}${MANPREFIX}/man1 1.46 + @echo installed manual pages to ${DESTDIR}${MANPREFIX}/man1 1.47 + 1.48 +uninstall: 1.49 + for i in ${BIN}; do \ 1.50 + rm -f ${DESTDIR}${PREFIX}/bin/`basename $$i`; \ 1.51 + done 1.52 + for i in ${MAN1}; do \ 1.53 + rm -f ${DESTDIR}${MANPREFIX}/man1/`basename $$i`; \ 1.54 + done
2.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 2.2 +++ b/gridmenu.1 Mon Jul 10 16:49:43 2006 +0200 2.3 @@ -0,0 +1,84 @@ 2.4 +.TH GRIDMENU 1 grid-0.0 2.5 +.SH NAME 2.6 +gridmenu \- grid window manager menu 2.7 +.SH SYNOPSIS 2.8 +.B gridmenu 2.9 +.RB [ \-v ] 2.10 +.RB [ \-t 2.11 +.IR title ] 2.12 +.SH DESCRIPTION 2.13 +.SS Overview 2.14 +.B gridmenu 2.15 +is a generic, highly customizable, and efficient menu for the X Window System, 2.16 +originally designed for 2.17 +.BR grid (1). 2.18 +It supports arbitrary, user defined menu contents. 2.19 +.SS Options 2.20 +.TP 2.21 +.B \-v 2.22 +prints version information to stdout, then exits. 2.23 +.TP 2.24 +.BI \-t " title" 2.25 +displays 2.26 +.I title 2.27 +above the menu. 2.28 +.SS Usage 2.29 +.B gridmenu 2.30 +reads a list of newline-separated items from stdin and creates a menu. 2.31 +When the user selects an item or enters any text and presses Enter, his choice 2.32 +is printed to stdout and 2.33 +.B gridmenu 2.34 +terminates. 2.35 +.SS Keyboard Control 2.36 +.B gridmenu 2.37 +is completely controlled by the keyboard. The following keys are recognized: 2.38 +.TP 2 2.39 +Any printable character 2.40 +appends the character to the text in the input field. This works as a filter: 2.41 +only items containing this text will be displayed. 2.42 +.TP 2 2.43 +Left/Right (Control-p/Control-n) 2.44 +select the previous/next item. 2.45 +.TP 2 2.46 +Tab (Control-i) 2.47 +copy the selected item to the input field. 2.48 +.TP 2 2.49 +Enter (Control-j) 2.50 +confirm selection and quit (print the selected item to stdout). 2.51 +.TP 2 2.52 +Shift-Enter (Shift-Control-j) 2.53 +confirm selection and quit (print the text in the input field to stdout). 2.54 +.TP 2 2.55 +Escape (Control-[) 2.56 +quit without selecting an item. 2.57 +.TP 2 2.58 +Backspace (Control-h) 2.59 +remove enough characters from the input field to change its filtering effect. 2.60 +.TP 2 2.61 +Control-u 2.62 +remove all characters from the input field. 2.63 +.SS Exit codes 2.64 +.B gridmenu 2.65 +returns 2.66 +.B 0 2.67 +if Enter is pressed on termination, 2.68 +.B 1 2.69 +if Escape is pressed. 2.70 +.SH ENVIRONMENT 2.71 +.TP 2.72 +GRID_FONT 2.73 +The X11 font used to display each item in the menu. 2.74 +.br 2.75 +Default: fixed 2.76 +.TP 2.77 +GRID_NORMCOLORS 2.78 +The foreground, background, and border colors of a label. Syntactically, three blank-separated color values of the form #RRGGBB are expected. 2.79 +.br 2.80 +Default: #222222 #eeeeee #666666 2.81 +.TP 2.82 +GRID_SELCOLORS 2.83 +Like GRID_NORMCOLORS, but for the selected label. 2.84 +.br 2.85 +Default: #ffffff #335577 #447799 2.86 +.SH SEE ALSO 2.87 +.BR gridwm (1)
3.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 3.2 +++ b/gridmenu.c Mon Jul 10 16:49:43 2006 +0200 3.3 @@ -0,0 +1,498 @@ 3.4 +/* 3.5 + * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> 3.6 + * (C)opyright MMVI Sander van Dijk <a dot h dot vandijk at gmail dot com> 3.7 + * See LICENSE file for license details. 3.8 + */ 3.9 + 3.10 +#include <ctype.h> 3.11 +#include <stdlib.h> 3.12 +#include <stdio.h> 3.13 +#include <string.h> 3.14 +#include <sys/stat.h> 3.15 +#include <sys/wait.h> 3.16 +#include <time.h> 3.17 +#include <unistd.h> 3.18 +#include <X11/Xlib.h> 3.19 +#include <X11/cursorfont.h> 3.20 +#include <X11/Xutil.h> 3.21 +#include <X11/keysym.h> 3.22 + 3.23 +#include <blitz.h> 3.24 +#include <cext.h> 3.25 + 3.26 +typedef struct Item Item; 3.27 + 3.28 +struct Item { 3.29 + Item *next; /* traverses all items */ 3.30 + Item *left, *right; /* traverses items matching current search pattern */ 3.31 + char *text; 3.32 +}; 3.33 + 3.34 +static char *title = nil; 3.35 +static Bool done = False; 3.36 +static int ret = 0; 3.37 +static char text[4096]; 3.38 +static BlitzColor selcolor; 3.39 +static BlitzColor normcolor; 3.40 +static Window win; 3.41 +static XRectangle mrect; 3.42 +static Item *allitem = nil; /* first of all items */ 3.43 +static Item *item = nil; /* first of pattern matching items */ 3.44 +static Item *sel = nil; 3.45 +static Item *nextoff = nil; 3.46 +static Item *prevoff = nil; 3.47 +static Item *curroff = nil; 3.48 +static int nitem = 0; 3.49 +static unsigned int cmdw = 0; 3.50 +static unsigned int twidth = 0; 3.51 +static unsigned int cwidth = 0; 3.52 +static Blitz blz = {0}; 3.53 +static BlitzBrush brush = {0}; 3.54 +static const int seek = 30; /* 30px */ 3.55 + 3.56 +static void draw_menu(void); 3.57 +static void handle_kpress(XKeyEvent * e); 3.58 + 3.59 +static char version[] = "wmiimenu - " VERSION ", (C)opyright MMIV-MMVI Anselm R. Garbe\n"; 3.60 + 3.61 +static void 3.62 +usage() 3.63 +{ 3.64 + fprintf(stderr, "%s", "usage: wmiimenu [-v] [-t <title>]\n"); 3.65 + exit(1); 3.66 +} 3.67 + 3.68 +static void 3.69 +update_offsets() 3.70 +{ 3.71 + unsigned int tw, w = cmdw + 2 * seek; 3.72 + 3.73 + if(!curroff) 3.74 + return; 3.75 + 3.76 + for(nextoff = curroff; nextoff; nextoff=nextoff->right) { 3.77 + tw = blitz_textwidth(brush.font, nextoff->text); 3.78 + if(tw > mrect.width / 3) 3.79 + tw = mrect.width / 3; 3.80 + w += tw + mrect.height; 3.81 + if(w > mrect.width) 3.82 + break; 3.83 + } 3.84 + 3.85 + w = cmdw + 2 * seek; 3.86 + for(prevoff = curroff; prevoff && prevoff->left; prevoff=prevoff->left) { 3.87 + tw = blitz_textwidth(brush.font, prevoff->left->text); 3.88 + if(tw > mrect.width / 3) 3.89 + tw = mrect.width / 3; 3.90 + w += tw + mrect.height; 3.91 + if(w > mrect.width) 3.92 + break; 3.93 + } 3.94 +} 3.95 + 3.96 +static void 3.97 +update_items(char *pattern) 3.98 +{ 3.99 + unsigned int plen = strlen(pattern); 3.100 + Item *i, *j; 3.101 + 3.102 + if(!pattern) 3.103 + return; 3.104 + 3.105 + if(!title || *pattern) 3.106 + cmdw = cwidth; 3.107 + else 3.108 + cmdw = twidth; 3.109 + 3.110 + item = j = nil; 3.111 + nitem = 0; 3.112 + 3.113 + for(i = allitem; i; i=i->next) 3.114 + if(!plen || !strncmp(pattern, i->text, plen)) { 3.115 + if(!j) 3.116 + item = i; 3.117 + else 3.118 + j->right = i; 3.119 + i->left = j; 3.120 + i->right = nil; 3.121 + j = i; 3.122 + nitem++; 3.123 + } 3.124 + for(i = allitem; i; i=i->next) 3.125 + if(plen && strncmp(pattern, i->text, plen) 3.126 + && strstr(i->text, pattern)) { 3.127 + if(!j) 3.128 + item = i; 3.129 + else 3.130 + j->right = i; 3.131 + i->left = j; 3.132 + i->right = nil; 3.133 + j = i; 3.134 + nitem++; 3.135 + } 3.136 + 3.137 + curroff = prevoff = nextoff = sel = item; 3.138 + 3.139 + update_offsets(); 3.140 +} 3.141 + 3.142 +/* creates brush structs for brush mode drawing */ 3.143 +static void 3.144 +draw_menu() 3.145 +{ 3.146 + unsigned int offx = 0; 3.147 + 3.148 + Item *i; 3.149 + 3.150 + brush.align = WEST; 3.151 + 3.152 + brush.rect = mrect; 3.153 + brush.rect.x = 0; 3.154 + brush.rect.y = 0; 3.155 + brush.color = normcolor; 3.156 + brush.border = False; 3.157 + blitz_draw_tile(&brush); 3.158 + 3.159 + /* print command */ 3.160 + if(!title || text[0]) { 3.161 + brush.color = normcolor; 3.162 + cmdw = cwidth; 3.163 + if(cmdw && item) 3.164 + brush.rect.width = cmdw; 3.165 + blitz_draw_label(&brush, text); 3.166 + } 3.167 + else { 3.168 + cmdw = twidth; 3.169 + brush.color = selcolor; 3.170 + brush.rect.width = cmdw; 3.171 + blitz_draw_label(&brush, title); 3.172 + } 3.173 + offx += brush.rect.width; 3.174 + 3.175 + brush.align = CENTER; 3.176 + if(curroff) { 3.177 + brush.color = normcolor; 3.178 + brush.rect.x = offx; 3.179 + brush.rect.width = seek; 3.180 + offx += brush.rect.width; 3.181 + blitz_draw_label(&brush, (curroff && curroff->left) ? "<" : nil); 3.182 + 3.183 + /* determine maximum items */ 3.184 + for(i = curroff; i != nextoff; i=i->right) { 3.185 + brush.color = normcolor; 3.186 + brush.border = False; 3.187 + brush.rect.x = offx; 3.188 + brush.rect.width = blitz_textwidth(brush.font, i->text); 3.189 + if(brush.rect.width > mrect.width / 3) 3.190 + brush.rect.width = mrect.width / 3; 3.191 + brush.rect.width += mrect.height; 3.192 + if(sel == i) { 3.193 + brush.color = selcolor; 3.194 + brush.border = True; 3.195 + } 3.196 + blitz_draw_label(&brush, i->text); 3.197 + offx += brush.rect.width; 3.198 + } 3.199 + 3.200 + brush.color = normcolor; 3.201 + brush.border = False; 3.202 + brush.rect.x = mrect.width - seek; 3.203 + brush.rect.width = seek; 3.204 + blitz_draw_label(&brush, nextoff ? ">" : nil); 3.205 + } 3.206 + XCopyArea(blz.dpy, brush.drawable, win, brush.gc, 0, 0, mrect.width, 3.207 + mrect.height, 0, 0); 3.208 + XSync(blz.dpy, False); 3.209 +} 3.210 + 3.211 +static void 3.212 +handle_kpress(XKeyEvent * e) 3.213 +{ 3.214 + KeySym ksym; 3.215 + char buf[32]; 3.216 + int num, prev_nitem; 3.217 + unsigned int i, len = strlen(text); 3.218 + 3.219 + buf[0] = 0; 3.220 + num = XLookupString(e, buf, sizeof(buf), &ksym, 0); 3.221 + 3.222 + if(IsFunctionKey(ksym) || IsKeypadKey(ksym) 3.223 + || IsMiscFunctionKey(ksym) || IsPFKey(ksym) 3.224 + || IsPrivateKeypadKey(ksym)) 3.225 + return; 3.226 + 3.227 + /* first check if a control mask is omitted */ 3.228 + if(e->state & ControlMask) { 3.229 + switch (ksym) { 3.230 + case XK_H: 3.231 + case XK_h: 3.232 + ksym = XK_BackSpace; 3.233 + break; 3.234 + case XK_I: 3.235 + case XK_i: 3.236 + ksym = XK_Tab; 3.237 + break; 3.238 + case XK_J: 3.239 + case XK_j: 3.240 + ksym = XK_Return; 3.241 + break; 3.242 + case XK_N: 3.243 + case XK_n: 3.244 + ksym = XK_Right; 3.245 + break; 3.246 + case XK_P: 3.247 + case XK_p: 3.248 + ksym = XK_Left; 3.249 + break; 3.250 + case XK_U: 3.251 + case XK_u: 3.252 + text[0] = 0; 3.253 + update_items(text); 3.254 + draw_menu(); 3.255 + return; 3.256 + break; 3.257 + case XK_bracketleft: 3.258 + ksym = XK_Escape; 3.259 + break; 3.260 + default: /* ignore other control sequences */ 3.261 + return; 3.262 + break; 3.263 + } 3.264 + } 3.265 + switch (ksym) { 3.266 + case XK_Left: 3.267 + if(!(sel && sel->left)) 3.268 + return; 3.269 + sel=sel->left; 3.270 + if(sel->right == curroff) { 3.271 + curroff = prevoff; 3.272 + update_offsets(); 3.273 + } 3.274 + break; 3.275 + case XK_Tab: 3.276 + if(!sel) 3.277 + return; 3.278 + cext_strlcpy(text, sel->text, sizeof(text)); 3.279 + update_items(text); 3.280 + break; 3.281 + case XK_Right: 3.282 + if(!(sel && sel->right)) 3.283 + return; 3.284 + sel=sel->right; 3.285 + if(sel == nextoff) { 3.286 + curroff = nextoff; 3.287 + update_offsets(); 3.288 + } 3.289 + break; 3.290 + case XK_Return: 3.291 + if(e->state & ShiftMask) { 3.292 + if(text) 3.293 + fprintf(stdout, "%s", text); 3.294 + } 3.295 + else if(sel) 3.296 + fprintf(stdout, "%s", sel->text); 3.297 + else if(text) 3.298 + fprintf(stdout, "%s", text); 3.299 + fflush(stdout); 3.300 + done = True; 3.301 + break; 3.302 + case XK_Escape: 3.303 + ret = 1; 3.304 + done = True; 3.305 + break; 3.306 + case XK_BackSpace: 3.307 + if((i = len)) { 3.308 + prev_nitem = nitem; 3.309 + do { 3.310 + text[--i] = 0; 3.311 + update_items(text); 3.312 + } while(i && nitem && prev_nitem == nitem); 3.313 + update_items(text); 3.314 + } 3.315 + break; 3.316 + default: 3.317 + if((num == 1) && !iscntrl((int) buf[0])) { 3.318 + buf[num] = 0; 3.319 + if(len > 0) 3.320 + cext_strlcat(text, buf, sizeof(text)); 3.321 + else 3.322 + cext_strlcpy(text, buf, sizeof(text)); 3.323 + update_items(text); 3.324 + } 3.325 + } 3.326 + draw_menu(); 3.327 +} 3.328 + 3.329 +static char * 3.330 +read_allitems() 3.331 +{ 3.332 + static char *maxname = nil; 3.333 + char *p, buf[1024]; 3.334 + unsigned int len = 0, max = 0; 3.335 + Item *i, *new; 3.336 + 3.337 + i = nil; 3.338 + while(fgets(buf, sizeof(buf), stdin)) { 3.339 + len = strlen(buf); 3.340 + if (buf[len - 1] == '\n') 3.341 + buf[len - 1] = 0; 3.342 + p = cext_estrdup(buf); 3.343 + if(max < len) { 3.344 + maxname = p; 3.345 + max = len; 3.346 + } 3.347 + 3.348 + new = cext_emalloc(sizeof(Item)); 3.349 + new->next = new->left = new->right = nil; 3.350 + new->text = p; 3.351 + if(!i) 3.352 + allitem = new; 3.353 + else 3.354 + i->next = new; 3.355 + i = new; 3.356 + } 3.357 + 3.358 + return maxname; 3.359 +} 3.360 + 3.361 +int 3.362 +main(int argc, char *argv[]) 3.363 +{ 3.364 + int i; 3.365 + XSetWindowAttributes wa; 3.366 + char *maxname, *p; 3.367 + BlitzFont font = {0}; 3.368 + GC gc; 3.369 + Drawable pmap; 3.370 + XEvent ev; 3.371 + 3.372 + /* command line args */ 3.373 + for(i = 1; i < argc; i++) { 3.374 + if (argv[i][0] == '-') 3.375 + switch (argv[i][1]) { 3.376 + case 'v': 3.377 + fprintf(stdout, "%s", version); 3.378 + exit(0); 3.379 + break; 3.380 + case 't': 3.381 + if(++i < argc) 3.382 + title = argv[i]; 3.383 + else 3.384 + usage(); 3.385 + break; 3.386 + default: 3.387 + usage(); 3.388 + break; 3.389 + } 3.390 + else 3.391 + usage(); 3.392 + } 3.393 + 3.394 + blz.dpy = XOpenDisplay(0); 3.395 + if(!blz.dpy) { 3.396 + fprintf(stderr, "%s", "wmiimenu: cannot open dpy\n"); 3.397 + exit(1); 3.398 + } 3.399 + blz.screen = DefaultScreen(blz.dpy); 3.400 + blz.root = RootWindow(blz.dpy, blz.screen); 3.401 + 3.402 + maxname = read_allitems(); 3.403 + 3.404 + /* grab as early as possible, but after reading all items!!! */ 3.405 + while(XGrabKeyboard 3.406 + (blz.dpy, blz.root, True, GrabModeAsync, 3.407 + GrabModeAsync, CurrentTime) != GrabSuccess) 3.408 + usleep(1000); 3.409 + 3.410 + font.fontstr = getenv("WMII_FONT"); 3.411 + if (!font.fontstr) 3.412 + font.fontstr = cext_estrdup(BLITZ_FONT); 3.413 + blitz_loadfont(&blz, &font); 3.414 + 3.415 + if((p = getenv("WMII_NORMCOLORS"))) 3.416 + cext_strlcpy(normcolor.colstr, p, sizeof(normcolor.colstr)); 3.417 + if(strlen(normcolor.colstr) != 23) 3.418 + cext_strlcpy(normcolor.colstr, BLITZ_NORMCOLORS, sizeof(normcolor.colstr)); 3.419 + blitz_loadcolor(&blz, &normcolor); 3.420 + 3.421 + if((p = getenv("WMII_SELCOLORS"))) 3.422 + cext_strlcpy(selcolor.colstr, p, sizeof(selcolor.colstr)); 3.423 + if(strlen(selcolor.colstr) != 23) 3.424 + cext_strlcpy(selcolor.colstr, BLITZ_SELCOLORS, sizeof(selcolor.colstr)); 3.425 + blitz_loadcolor(&blz, &selcolor); 3.426 + 3.427 + wa.override_redirect = 1; 3.428 + wa.background_pixmap = ParentRelative; 3.429 + wa.event_mask = ExposureMask | ButtonPressMask | KeyPressMask 3.430 + | SubstructureRedirectMask | SubstructureNotifyMask; 3.431 + 3.432 + mrect.width = DisplayWidth(blz.dpy, blz.screen); 3.433 + mrect.height = font.ascent + font.descent + 4; 3.434 + mrect.y = DisplayHeight(blz.dpy, blz.screen) - mrect.height; 3.435 + mrect.x = 0; 3.436 + 3.437 + win = XCreateWindow(blz.dpy, blz.root, mrect.x, mrect.y, 3.438 + mrect.width, mrect.height, 0, DefaultDepth(blz.dpy, blz.screen), 3.439 + CopyFromParent, DefaultVisual(blz.dpy, blz.screen), 3.440 + CWOverrideRedirect | CWBackPixmap | CWEventMask, &wa); 3.441 + XDefineCursor(blz.dpy, win, XCreateFontCursor(blz.dpy, XC_xterm)); 3.442 + XSync(blz.dpy, False); 3.443 + 3.444 + /* pixmap */ 3.445 + gc = XCreateGC(blz.dpy, win, 0, 0); 3.446 + pmap = XCreatePixmap(blz.dpy, win, mrect.width, mrect.height, 3.447 + DefaultDepth(blz.dpy, blz.screen)); 3.448 + 3.449 + XSync(blz.dpy, False); 3.450 + 3.451 + brush.blitz = &blz; 3.452 + brush.color = normcolor; 3.453 + brush.drawable = pmap; 3.454 + brush.gc = gc; 3.455 + brush.font = &font; 3.456 + 3.457 + if(maxname) 3.458 + cwidth = blitz_textwidth(brush.font, maxname) + mrect.height; 3.459 + if(cwidth > mrect.width / 3) 3.460 + cwidth = mrect.width / 3; 3.461 + 3.462 + if(title) { 3.463 + twidth = blitz_textwidth(brush.font, title) + mrect.height; 3.464 + if(twidth > mrect.width / 3) 3.465 + twidth = mrect.width / 3; 3.466 + } 3.467 + 3.468 + cmdw = title ? twidth : cwidth; 3.469 + 3.470 + text[0] = 0; 3.471 + update_items(text); 3.472 + XMapRaised(blz.dpy, win); 3.473 + draw_menu(); 3.474 + XSync(blz.dpy, False); 3.475 + 3.476 + /* main event loop */ 3.477 + while(!XNextEvent(blz.dpy, &ev)) { 3.478 + switch (ev.type) { 3.479 + case KeyPress: 3.480 + handle_kpress(&ev.xkey); 3.481 + break; 3.482 + case Expose: 3.483 + if(ev.xexpose.count == 0) { 3.484 + draw_menu(); 3.485 + } 3.486 + break; 3.487 + default: 3.488 + break; 3.489 + } 3.490 + if(done) 3.491 + break; 3.492 + } 3.493 + 3.494 + XUngrabKeyboard(blz.dpy, CurrentTime); 3.495 + XFreePixmap(blz.dpy, pmap); 3.496 + XFreeGC(blz.dpy, gc); 3.497 + XDestroyWindow(blz.dpy, win); 3.498 + XCloseDisplay(blz.dpy); 3.499 + 3.500 + return ret; 3.501 +}