wmii/cmd/wmibar.c

539 lines
12 KiB
C

/*
* (C)opyright MMIV-MMV Anselm R. Garbe <garbeam at gmail dot com>
* See LICENSE file for license details.
*/
#include <assert.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <X11/Xatom.h>
#include <X11/cursorfont.h>
#include <X11/Xproto.h>
#include <X11/Xutil.h>
#include "wmii.h"
/* array indexes for file pointers */
typedef enum {
B_CTL,
B_NEW,
B_EXPANDABLE,
B_GEOMETRY,
B_FONT,
B_FG_COLOR,
B_BORDER_COLOR,
B_BG_COLOR,
B_LAST
} BarIndexes;
typedef struct {
File *root;
Draw d;
} Item;
static IXPServer *ixps = 0;
static Display *dpy;
static GC gc;
static Window win;
static XRectangle rect;
static XRectangle brect;
static int screen_num;
static int displayed = 0;
static char *sockfile = 0;
static File *files[B_LAST];
static Item **items = 0;
static unsigned int id = 0;
static Pixmap pmap;
static Draw zero_draw = { 0 };
static void draw_bar(void *obj, char *arg);
static void quit(void *obj, char *arg);
static void display(void *obj, char *arg);
static void reset(void *obj, char *arg);
static void _destroy(void *obj, char *arg);
static void handle_after_write(IXPServer * s, File * f);
static Action acttbl[] = {
{"quit", quit},
{"display", display},
{"update", draw_bar},
{"reset", reset},
{"destroy", _destroy},
{0, 0}
};
static char *version[] = {
"wmibar - window manager improved bar - " VERSION "\n"
" (C)opyright MMIV-MMV Anselm R. Garbe\n", 0
};
static void usage()
{
fprintf(stderr, "%s",
"usage: wmibar -s <socket file> [-v] [<x>,<y>,<width>,<height>]\n"
" -s socket file\n" " -v version info\n");
exit(1);
}
/**
* <path>/data "<txt value>"
* <path>/fgcolor "#RRGGBBAA"
* <path>/bgcolor "#RRGGBBAA"
* <path>/bordercolor "#RRGGBBAA"
* <path>/b1press "<command>"
* <path>/b2press "<command>"
* <path>/b3press "<command>"
* <path>/b4press "<command>"
* <path>/b5press "<command>"
*/
static void create_label(char *path)
{
File *f;
char file[MAX_BUF];
int i;
snprintf(file, MAX_BUF, "%s/data", path);
f = ixp_create(ixps, file);
f->after_write = handle_after_write;
snprintf(file, MAX_BUF, "%s/fgcolor", path);
wmii_create_ixpfile(ixps, file, files[B_FG_COLOR]->content);
snprintf(file, MAX_BUF, "%s/bgcolor", path);
wmii_create_ixpfile(ixps, file, files[B_BG_COLOR]->content);
snprintf(file, MAX_BUF, "%s/bordercolor", path);
wmii_create_ixpfile(ixps, file, files[B_BORDER_COLOR]->content);
for (i = 1; i < 6; i++) { /* 5 buttons events */
snprintf(file, MAX_BUF, "%s/b%dpress", path, i);
ixp_create(ixps, file);
}
}
static void _destroy(void *obj, char *arg)
{
char buf[512];
if (!arg)
return;
snprintf(buf, sizeof(buf), "/%s", arg);
ixps->remove(ixps, buf);
draw_bar(0, 0);
}
static void reset(void *obj, char *arg)
{
int i;
char buf[512];
for (i = 0; i < id; i++) {
snprintf(buf, sizeof(buf), "/%d", i + 1);
ixps->remove(ixps, buf);
}
id = 0;
draw_bar(0, 0);
}
static void quit(void *obj, char *arg)
{
ixps->runlevel = SHUTDOWN;
}
static void display(void *obj, char *arg)
{
if (!arg)
return;
displayed = _strtonum(arg, 0, 1);
if (displayed) {
XMapRaised(dpy, win);
draw_bar(0, 0);
} else {
XUnmapWindow(dpy, win);
XSync(dpy, False);
}
}
static void init_draw_label(char *path, Draw * d)
{
char buf[MAX_BUF];
File *f;
/* text stuff */
snprintf(buf, MAX_BUF, "%s/data", path);
f = ixp_walk(ixps, buf);
d->data = f->content;
/* style stuff */
snprintf(buf, MAX_BUF, "%s/fgcolor", path);
f = ixp_walk(ixps, buf);
d->fg = blitz_loadcolor(dpy, screen_num, f->content);
snprintf(buf, MAX_BUF, "%s/bgcolor", path);
f = ixp_walk(ixps, buf);
d->bg = blitz_loadcolor(dpy, screen_num, f->content);
snprintf(buf, MAX_BUF, "%s/bordercolor", path);
f = ixp_walk(ixps, buf);
d->border = blitz_loadcolor(dpy, screen_num, f->content);
}
static void init_item(char *path, Item * i)
{
i->d = zero_draw;
i->root = ixp_walk(ixps, path);
i->d.gc = gc;
i->d.drawable = pmap;
i->d.rect = brect;
i->d.rect.y = 0;
init_draw_label(path, &i->d);
}
static int comp_str(const void *s1, const void *s2)
{
return strcmp(*(char **) s1, *(char **) s2);
}
static void draw()
{
unsigned int n = 0, i, w, xoff = 0;
XFontStruct *font;
unsigned expandable = 0;
char buf[32];
if (!items)
return;
n = count_items((void **) items);
font = blitz_getfont(dpy, files[B_FONT]->content);
expandable = _strtonum(files[B_EXPANDABLE]->content, 1, id);
snprintf(buf, sizeof(buf), "/%d", expandable);
if (!ixp_walk(ixps, buf))
expandable = 0;
w = 0;
/* precalc */
for (i = 0; expandable && items[i]; i++)
if (i + 1 != expandable) {
items[i]->d.rect.width = brect.height;
if (items[i]->d.data) {
if (!strncmp(items[i]->d.data, "%m:", 3))
/* meter */
items[i]->d.rect.width = brect.height / 2;
else
items[i]->d.rect.width +=
XTextWidth(font, items[i]->d.data,
strlen(items[i]->d.data));
}
w += items[i]->d.rect.width;
}
if (!expandable || w > brect.width) {
/* failsafe mode, give all labels same width */
w = brect.width / n;
for (i = 0; items[i]; i++)
items[i]->d.rect.width = w;
items[i - 1]->d.rect.width = brect.width - items[i - 1]->d.rect.x;
} else
items[expandable - 1]->d.rect.width = brect.width - w;
for (i = 0; items[i]; i++) {
items[i]->d.font = font;
items[i]->d.rect.x = xoff;
xoff += items[i]->d.rect.width;
if (items[i]->d.data && !strncmp(items[i]->d.data, "%m:", 3))
blitz_drawmeter(dpy, &items[i]->d);
else
blitz_drawlabel(dpy, &items[i]->d);
}
XCopyArea(dpy, pmap, win, gc, 0, 0, brect.width, brect.height, 0, 0);
XSync(dpy, False);
XFreeFont(dpy, font);
}
static void draw_bar(void *obj, char *arg)
{
File *label = 0;
unsigned int i = 0, n = 0;
Item *item;
char buf[512];
if (!displayed)
return;
if (items) {
for (i = 0; items[i]; i++) {
free(items[i]);
}
free(items);
}
items = 0;
snprintf(buf, sizeof(buf), "%s", "/1");
label = ixp_walk(ixps, buf);
if (!label) {
Draw d = { 0 };
/* default stuff */
d.gc = gc;
d.drawable = pmap;
d.rect.width = brect.width;
d.rect.height = brect.height;
d.bg =
blitz_loadcolor(dpy, screen_num, files[B_BG_COLOR]->content);
d.fg =
blitz_loadcolor(dpy, screen_num, files[B_FG_COLOR]->content);
d.border =
blitz_loadcolor(dpy, screen_num,
files[B_BORDER_COLOR]->content);
blitz_drawlabelnoborder(dpy, &d);
} else {
File *f;
char **paths = 0;
/*
* take order into account, directory names are used in
* alphabetical order
*/
n = 0;
for (f = label; f; f = f->next)
n++;
paths = cext_emalloc(sizeof(char *) * n);
i = 0;
for (f = label; f; f = f->next)
paths[i++] = f->name;
qsort(paths, n, sizeof(char *), comp_str);
for (i = 0; i < n; i++) {
snprintf(buf, sizeof(buf), "/%s", paths[i]);
item = cext_emalloc(sizeof(Item));
items = (Item **) attach_item_end((void **) items, item, sizeof(Item *));
init_item(buf, item);
}
draw();
free(paths);
}
}
static Item *get_item_for_file(File * f)
{
int i;
for (i = 0; items && items[i]; i++)
if (items[i]->root == f)
return items[i];
return 0;
}
static void handle_buttonpress(XButtonPressedEvent * e)
{
File *p;
char buf[MAX_BUF];
char path[512];
int i;
for (i = 0; items && items[i]; i++) {
if (blitz_ispointinrect(e->x, e->y, &items[i]->d.rect)) {
path[0] = '\0';
wmii_get_ixppath(items[i]->root, path, sizeof(path));
snprintf(buf, MAX_BUF, "%s/b%upress", path, e->button);
if ((p = ixp_walk(ixps, buf)))
if (p->content)
spawn(dpy, p->content);
return;
}
}
}
static void check_event(Connection * e)
{
XEvent ev;
while (XPending(dpy)) {
XNextEvent(dpy, &ev);
switch (ev.type) {
case ButtonPress:
handle_buttonpress(&ev.xbutton);
break;
case Expose:
if (ev.xexpose.count == 0) {
/* XRaiseWindow(dpy, win); */
draw_bar(0, 0);
}
break;
default:
break;
}
}
}
static void update_geometry(char *size)
{
blitz_strtorect(&rect, &brect, size);
if (!brect.width)
brect.width = DisplayWidth(dpy, screen_num);
if (!brect.height)
brect.height = 20;
}
static void handle_after_write(IXPServer * s, File * f)
{
int i;
size_t len;
Item *item;
char buf[512];
buf[0] = '\0';
if (!strncmp(f->name, "data", 5)) {
if ((item = get_item_for_file(f->parent))) {
wmii_get_ixppath(f->parent, buf, sizeof(buf));
init_draw_label(buf, &item->d);
draw();
}
} else if (files[B_GEOMETRY] == f) {
char *geom = files[B_GEOMETRY]->content;
if (geom && strrchr(geom, ',')) {
update_geometry(geom);
XMoveResizeWindow(dpy, win, brect.x, brect.y,
brect.width, brect.height);
XSync(dpy, False);
pmap = XCreatePixmap(dpy, win, brect.width, brect.height,
DefaultDepth(dpy, screen_num));
XSync(dpy, False);
draw_bar(0, 0);
}
} else if (files[B_CTL] == f) {
for (i = 0; acttbl[i].name; i++) {
len = strlen(acttbl[i].name);
if (!strncmp(acttbl[i].name, (char *) f->content, len)) {
if (strlen(f->content) > len) {
acttbl[i].func(0, &((char *) f->content)[len + 1]);
} else {
acttbl[i].func(0, 0);
}
break;
}
}
}
check_event(0);
}
static void handle_before_read(IXPServer * s, File * f)
{
char buf[64];
if (f == files[B_GEOMETRY]) {
snprintf(buf, sizeof(buf), "%d,%d,%d,%d", brect.x, brect.y,
brect.width, brect.height);
if (f->content)
free(f->content);
f->content = strdup(buf);
f->size = strlen(buf);
} else if (f == files[B_NEW]) {
snprintf(buf, sizeof(buf), "%d", ++id);
if (f->content)
free(f->content);
f->content = strdup(buf);
f->size = strlen(buf);
create_label(buf);
draw_bar(0, 0);
}
}
static void run(char *geom)
{
XSetWindowAttributes wa;
XGCValues gcv;
/* init */
if (!(files[B_CTL] = ixp_create(ixps, "/ctl"))) {
perror("wmibar: cannot connect IXP server");
exit(1);
}
files[B_CTL]->after_write = handle_after_write;
files[B_NEW] = ixp_create(ixps, "/new");
files[B_NEW]->before_read = handle_before_read;
files[B_FONT] = wmii_create_ixpfile(ixps, "/font", BLITZ_FONT);
files[B_BG_COLOR] =
wmii_create_ixpfile(ixps, "/bgcolor", BLITZ_NORM_BG_COLOR);
files[B_FG_COLOR] =
wmii_create_ixpfile(ixps, "/fgcolor", BLITZ_NORM_FG_COLOR);
files[B_BORDER_COLOR] =
wmii_create_ixpfile(ixps, "/bordercolor", BLITZ_NORM_BORDER_COLOR);
files[B_GEOMETRY] = ixp_create(ixps, "/geometry");
files[B_GEOMETRY]->before_read = handle_before_read;
files[B_GEOMETRY]->after_write = handle_after_write;
files[B_EXPANDABLE] = ixp_create(ixps, "/expandable");
wa.override_redirect = 1;
wa.background_pixmap = ParentRelative;
wa.event_mask =
ExposureMask | ButtonPressMask | SubstructureRedirectMask |
SubstructureNotifyMask;
brect.x = brect.y = brect.width = brect.height = 0;
rect.x = rect.y = 0;
rect.width = DisplayWidth(dpy, screen_num);
rect.height = DisplayHeight(dpy, screen_num);
update_geometry(geom);
win = XCreateWindow(dpy, RootWindow(dpy, screen_num), brect.x, brect.y,
brect.width, brect.height, 0, DefaultDepth(dpy,
screen_num),
CopyFromParent, DefaultVisual(dpy, screen_num),
CWOverrideRedirect | CWBackPixmap | CWEventMask,
&wa);
XDefineCursor(dpy, win, XCreateFontCursor(dpy, XC_left_ptr));
XSync(dpy, False);
gcv.function = GXcopy;
gcv.graphics_exposures = False;
gc = XCreateGC(dpy, win, 0, 0);
pmap =
XCreatePixmap(dpy, win, brect.width, brect.height,
DefaultDepth(dpy, screen_num));
/* main event loop */
run_server_with_fd_support(ixps, ConnectionNumber(dpy),
check_event, 0);
deinit_server(ixps);
XFreePixmap(dpy, pmap);
XFreeGC(dpy, gc);
XCloseDisplay(dpy);
}
static int dummy_error_handler(Display * dpy, XErrorEvent * err)
{
return 0;
}
int main(int argc, char *argv[])
{
char geom[64];
int i;
/* command line args */
for (i = 1; (i < argc) && (argv[i][0] == '-'); i++) {
switch (argv[i][1]) {
case 'v':
fprintf(stdout, "%s", version[0]);
exit(0);
break;
case 's':
if (i + 1 < argc)
sockfile = argv[++i];
else
usage();
break;
default:
usage();
break;
}
}
dpy = XOpenDisplay(0);
if (!dpy) {
fprintf(stderr, "%s", "wmibar: cannot open display\n");
exit(1);
}
XSetErrorHandler(dummy_error_handler);
screen_num = DefaultScreen(dpy);
geom[0] = '\0';
if (argc > i)
cext_strlcpy(geom, argv[i], sizeof(geom));
ixps = wmii_setup_server(sockfile);
run(geom);
return 0;
}