|                                                                                                                                       |  | /*
 * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> * See LICENSE file for license details. */
#include <stdlib.h>
#include <string.h>
#include <X11/Xatom.h>
#include "util.h"
#include "wm.h"
#define CLIENT_MASK		(StructureNotifyMask | PropertyChangeMask | EnterWindowMask)
voidupdate_name(Client *c){	XTextProperty name;	int n;	char **list = NULL;
	name.nitems = 0;	c->name[0] = 0;	XGetTextProperty(dpy, c->win, &name, net_atom[NetWMName]);	if(!name.nitems)		XGetWMName(dpy, c->win, &name);	if(!name.nitems)		return;	if(name.encoding == XA_STRING)		strncpy(c->name, (char *)name.value, sizeof(c->name));	else {		if(XmbTextPropertyToTextList(dpy, &name, &list, &n) >= Success				&& n > 0 && *list)		{			strncpy(c->name, *list, sizeof(c->name));			XFreeStringList(list);		}	}	XFree(name.value);	if(c == stack)		draw_bar();	else		draw_client(c);}
voidfocus(Client *c){	Client **l;	for(l=&stack; *l && *l != c; l=&(*l)->snext);	eassert(*l == c);	*l = c->snext;	c->snext = stack;	stack = c;	XRaiseWindow(dpy, c->win);	XSetInputFocus(dpy, c->win, RevertToPointerRoot, CurrentTime);	XFlush(dpy);}
voidmanage(Window w, XWindowAttributes *wa){	Client *c, **l;	XSetWindowAttributes twa;	long msize;
	c = emallocz(sizeof(Client));	c->win = w;	c->r[RFloat].x = wa->x;	c->r[RFloat].y = wa->y;	c->r[RFloat].width = wa->width;	c->r[RFloat].height = wa->height;	c->border = wa->border_width;	XSetWindowBorderWidth(dpy, c->win, 0);	XSelectInput(dpy, c->win, CLIENT_MASK);	XGetTransientForHint(dpy, c->win, &c->trans);	if(!XGetWMNormalHints(dpy, c->win, &c->size, &msize) || !c->size.flags)		c->size.flags = PSize;	c->fixedsize =		(c->size.flags & PMinSize && c->size.flags & PMaxSize		 && c->size.min_width == c->size.max_width		 && c->size.min_height == c->size.max_height);	update_name(c);	twa.override_redirect = 1;	twa.background_pixmap = ParentRelative;	twa.event_mask = ExposureMask;
	c->title = XCreateWindow(dpy, root, c->r[RFloat].x, c->r[RFloat].y,			c->r[RFloat].width, barrect.height, 0,			DefaultDepth(dpy, screen), CopyFromParent,			DefaultVisual(dpy, screen),			CWOverrideRedirect | CWBackPixmap | CWEventMask, &twa);
	for(l=&clients; *l; l=&(*l)->next);	c->next = *l; /* *l == nil */	*l = c;	c->snext = stack;	stack = c;	XMapWindow(dpy, c->win);	XGrabButton(dpy, AnyButton, Mod1Mask, c->win, False, ButtonPressMask,			GrabModeAsync, GrabModeSync, None, None);	focus(c);}
voidresize(Client *c){	XConfigureEvent e;
	XMoveResizeWindow(dpy, c->win, c->r[RFloat].x, c->r[RFloat].y,			c->r[RFloat].width, c->r[RFloat].height);	e.type = ConfigureNotify;	e.event = c->win;	e.window = c->win;	e.x = c->r[RFloat].x;	e.y = c->r[RFloat].y;	e.width = c->r[RFloat].width;	e.height = c->r[RFloat].height;	e.border_width = c->border;	e.above = None;	e.override_redirect = False;	XSelectInput(dpy, c->win, CLIENT_MASK & ~StructureNotifyMask);	XSendEvent(dpy, c->win, False, StructureNotifyMask, (XEvent *)&e);	XSelectInput(dpy, c->win, CLIENT_MASK);	XFlush(dpy);}
static intdummy_error_handler(Display *dpy, XErrorEvent *error){	return 0;}
voidunmanage(Client *c){	Client **l;
	XGrabServer(dpy);	XSetErrorHandler(dummy_error_handler);
	XUngrabButton(dpy, AnyButton, AnyModifier, c->win);	XUnmapWindow(dpy, c->win);	XDestroyWindow(dpy, c->title);
	for(l=&clients; *l && *l != c; l=&(*l)->next);	eassert(*l == c);	*l = c->next;	for(l=&stack; *l && *l != c; l=&(*l)->snext);	eassert(*l == c);	*l = c->snext;	free(c);
	XFlush(dpy);	XSetErrorHandler(error_handler);	XUngrabServer(dpy);	discard_events(EnterWindowMask);	if(stack)		focus(stack);}
Client *getclient(Window w){	Client *c;	for(c = clients; c; c = c->next)		if(c->win == w)			return c;	return NULL;}
voiddraw_client(Client *c){	
}
 |