mirror of
https://github.com/0intro/wmii
synced 2024-12-26 05:16:59 +03:00
539 lines
12 KiB
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;
|
|
}
|