wmii/cmd/menu/main.c
2009-03-29 23:07:36 -04:00

315 lines
5.5 KiB
C

/* Copyright ©2006-2008 Kris Maglione <fbsdaemon@gmail.com>
* See LICENSE file for license details.
*/
#define IXP_NO_P9_
#define IXP_P9_STRUCTS
#define EXTERN
#include "dat.h"
#include <X11/Xproto.h>
#include <locale.h>
#include <strings.h>
#include <unistd.h>
#include <bio.h>
#include "fns.h"
#define link _link
static const char version[] = "wimenu-"VERSION", ©2008 Kris Maglione\n";
static IxpClient* client;
static IxpCFid* ctlfid;
static Biobuf* inbuf;
static char ctl[1024];
static char* ectl;
static void
usage(void) {
fatal("usage: wimenu -i [-h <history>] [-a <address>] [-p <prompt>]\n");
}
static int
errfmt(Fmt *f) {
return fmtstrcpy(f, ixp_errbuf());
}
/* Stubs. */
void
debug(int flag, const char *fmt, ...) {
va_list ap;
USED(flag);
va_start(ap, fmt);
vfprint(2, fmt, ap);
va_end(ap);
}
void dprint(long, char*, ...);
void dprint(long mask, char *fmt, ...) {
va_list ap;
USED(mask);
va_start(ap, fmt);
vfprint(2, fmt, ap);
va_end(ap);
}
static char*
readctl(char *key) {
char *s, *p;
int nkey, n;
nkey = strlen(key);
p = ctl - 1;
do {
p++;
if(!strncmp(p, key, nkey)) {
p += nkey;
s = strchr(p, '\n');
n = (s ? s : ectl) - p;
s = freelater(emalloc(n + 1));
s[n] = '\0';
return strncpy(s, p, n);
}
} while((p = strchr(p, '\n')));
return "";
}
static void
splice(Item *i) {
i->next->prev = i->prev;
i->prev->next = i->next;
}
static void
link(Item *i, Item *j) {
i->next = j;
j->prev = i;
}
static Item*
populate_list(Biobuf *buf, bool hist) {
Item ret;
Item *i;
char *p;
i = &ret;
while((p = Brdstr(buf, '\n', true))) {
link(i, emallocz(sizeof *i));
i->next_link = i->next;
i = i->next;
i->string = p;
i->retstring = p;
if(!hist) {
i->len = strlen(p);
i->width = textwidth_l(font, p, i->len);
if(i->width > maxwidth)
maxwidth = i->width;
}
}
link(i, &ret);
splice(&ret);
return ret.next != &ret ? ret.next : nil;
}
Item*
filter_list(Item *i, char *filter) {
static Item exact;
Item start, substr;
Item *exactp, *startp, *substrp;
Item **ip;
char *p;
int len;
len = strlen(filter);
exactp = &exact;
startp = &start;
substrp = &substr;
for(; i; i=i->next_link)
if((p = find(i->string, filter))) {
ip = &substrp;
if(p == i->string)
if(strlen(p) == len)
ip = &exactp;
else
ip = &startp;
link(*ip, i);
*ip = i;
}
link(substrp, &exact);
link(startp, &substr);
link(exactp, &start);
splice(&substr);
splice(&start);
splice(&exact);
return exact.next;
}
void
update_filter(void) {
/* TODO: Perhaps filter only previous matches unless filter
* has been truncated.
*/
matchfirst = matchstart = matchidx = filter_list(items, input.string);
}
/*
* There's no way to check accesses to destroyed windows, thus
* those cases are ignored (especially on UnmapNotifies).
* Other types of errors call Xlib's default error handler, which
* calls exit().
*/
ErrorCode ignored_xerrors[] = {
{ 0, BadWindow },
{ X_SetInputFocus, BadMatch },
{ X_PolyText8, BadDrawable },
{ X_PolyFillRectangle, BadDrawable },
{ X_PolySegment, BadDrawable },
{ X_ConfigureWindow, BadMatch },
{ X_GrabKey, BadAccess },
{ X_GetAtomName, BadAtom },
};
static void
end(IxpConn *c) {
USED(c);
srv.running = 0;
}
static void
preselect(IxpServer *s) {
USED(s);
check_x_event(nil);
}
void
init_screens(void) {
Rectangle *rects;
Point p;
int i, n;
/* Pick the screen with the pointer, for now. Later,
* try for the screen with the focused window first.
*/
p = querypointer(&scr.root);
rects = xinerama_screens(&n);
for(i=0; i < n; i++)
if(rect_haspoint_p(p, rects[i]))
break;
if(i == n)
i = 0;
/* Probably not the best route. */
scr.rect = rects[i];
menu_show();
}
int
main(int argc, char *argv[]) {
Item *item;
char *address;
char *histfile;
int i;
long ndump;
quotefmtinstall();
fmtinstall('r', errfmt);
address = getenv("WMII_ADDRESS");
histfile = nil;
prompt = nil;
promptw = 0;
find = strstr;
compare = strncmp;
ndump = -1;
ARGBEGIN{
case 'a':
address = EARGF(usage());
break;
case 'h':
histfile = EARGF(usage());
break;
case 'n':
ndump = strtol(EARGF(usage()), nil, 10);
break;
case 'p':
prompt = EARGF(usage());
break;
case 'i':
find = strcasestr;
compare = strncasecmp;
break;
default:
usage();
}ARGEND;
if(argc)
usage();
setlocale(LC_CTYPE, "");
initdisplay();
xext_init();
if(!isatty(0))
menu_init();
if(address && *address)
client = ixp_mount(address);
else
client = ixp_nsmount("wmii");
if(client == nil)
fatal("can't mount: %r\n");
ctlfid = ixp_open(client, "ctl", OREAD);
i = ixp_read(ctlfid, ctl, 1023);
ectl = ctl + i;
srv.preselect = preselect;
ixp_listen(&srv, ConnectionNumber(display), nil, check_x_event, end);
ontop = !strcmp(readctl("bar on "), "top");
loadcolor(&cnorm, readctl("normcolors "));
loadcolor(&csel, readctl("focuscolors "));
font = loadfont(readctl("font "));
if(!font)
fatal("Can't load font %q", readctl("font "));
inbuf = Bfdopen(0, OREAD);
items = populate_list(inbuf, false);
caret_insert("", true);
update_filter();
Bterm(inbuf);
histidx = &hist;
link(&hist, &hist);
if(histfile) {
inbuf = Bopen(histfile, OREAD);
if(inbuf) {
item = populate_list(inbuf, true);
if(item) {
link(item->prev, &hist);
link(&hist, item);
}
Bterm(inbuf);
}
}
if(barwin == nil)
menu_init();
init_screens();
i = ixp_serverloop(&srv);
if(i)
fprint(2, "%s: error: %r\n", argv0);
XCloseDisplay(display);
if(ndump >= 0 && histfile && result == 0)
history_dump(histfile, ndump);
return result;
}