| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384 | /* © 2006-2007 Anselm R. Garbe <garbeam at gmail dot com> * © 2006-2007 Sander van Dijk <a dot h dot vandijk at gmail dot com> * See LICENSE file for license details. */#include "dwm.h"#include <stdio.h>#include <stdlib.h>#include <X11/keysym.h>#include <X11/Xatom.h>/* static */typedef struct {	unsigned long mod;	KeySym keysym;	void (*func)(const char *arg);	const char *arg;} Key;KEYS#define CLEANMASK(mask) (mask & ~(numlockmask | LockMask))#define MOUSEMASK		(BUTTONMASK | PointerMotionMask)static Client *getclient(Window w) {	Client *c;	for(c = clients; c && c->win != w; c = c->next);	return c;}static voidmovemouse(Client *c) {	int x1, y1, ocx, ocy, di, nx, ny;	unsigned int dui;	Window dummy;	XEvent ev;	ocx = nx = c->x;	ocy = ny = c->y;	if(XGrabPointer(dpy, root, False, MOUSEMASK, GrabModeAsync, GrabModeAsync,			None, cursor[CurMove], CurrentTime) != GrabSuccess)		return;	c->ismax = False;	XQueryPointer(dpy, root, &dummy, &dummy, &x1, &y1, &di, &di, &dui);	for(;;) {		XMaskEvent(dpy, MOUSEMASK | ExposureMask | SubstructureRedirectMask, &ev);		switch (ev.type) {		case ButtonRelease:			XUngrabPointer(dpy, CurrentTime);			return;		case ConfigureRequest:		case Expose:		case MapRequest:			handler[ev.type](&ev);			break;		case MotionNotify:			XSync(dpy, False);			nx = ocx + (ev.xmotion.x - x1);			ny = ocy + (ev.xmotion.y - y1);			if(abs(wax + nx) < SNAP)				nx = wax;			else if(abs((wax + waw) - (nx + c->w + 2 * c->border)) < SNAP)				nx = wax + waw - c->w - 2 * c->border;			if(abs(way - ny) < SNAP)				ny = way;			else if(abs((way + wah) - (ny + c->h + 2 * c->border)) < SNAP)				ny = way + wah - c->h - 2 * c->border;			resize(c, nx, ny, c->w, c->h, False);			break;		}	}}static voidresizemouse(Client *c) {	int ocx, ocy;	int nw, nh;	XEvent ev;	ocx = c->x;	ocy = c->y;	if(XGrabPointer(dpy, root, False, MOUSEMASK, GrabModeAsync, GrabModeAsync,			None, cursor[CurResize], CurrentTime) != GrabSuccess)		return;	c->ismax = False;	XWarpPointer(dpy, None, c->win, 0, 0, 0, 0, c->w + c->border - 1, c->h + c->border - 1);	for(;;) {		XMaskEvent(dpy, MOUSEMASK | ExposureMask | SubstructureRedirectMask , &ev);		switch(ev.type) {		case ButtonRelease:			XWarpPointer(dpy, None, c->win, 0, 0, 0, 0,					c->w + c->border - 1, c->h + c->border - 1);			XUngrabPointer(dpy, CurrentTime);			while(XCheckMaskEvent(dpy, EnterWindowMask, &ev));			return;		case ConfigureRequest:		case Expose:		case MapRequest:			handler[ev.type](&ev);			break;		case MotionNotify:			XSync(dpy, False);			if((nw = ev.xmotion.x - ocx - 2 * c->border + 1) <= 0)				nw = 1;			if((nh = ev.xmotion.y - ocy - 2 * c->border + 1) <= 0)				nh = 1;			resize(c, c->x, c->y, nw, nh, True);			break;		}	}}static voidbuttonpress(XEvent *e) {	static char buf[32];	unsigned int i, x;	Client *c;	XButtonPressedEvent *ev = &e->xbutton;	buf[0] = 0;	if(barwin == ev->window) {		x = 0;		for(i = 0; i < ntags; i++) {			x += textw(tags[i]);			if(ev->x < x) {				snprintf(buf, sizeof buf, "%d", i);				if(ev->button == Button1) {					if(ev->state & MODKEY)						tag(buf);					else						view(buf);				}				else if(ev->button == Button3) {					if(ev->state & MODKEY)						toggletag(buf);					else						toggleview(buf);				}				return;			}		}		if(ev->x < x + blw)			switch(ev->button) {			case Button1:				setlayout(NULL);				break;			}	}	else if((c = getclient(ev->window))) {		focus(c);		if(CLEANMASK(ev->state) != MODKEY)			return;		if(ev->button == Button1 && (lt->arrange == floating || c->isfloating)) {			restack();			movemouse(c);		}		else if(ev->button == Button2)			zoom(NULL);		else if(ev->button == Button3		&& (lt->arrange == floating || c->isfloating) && !c->isfixed)		{			restack();			resizemouse(c);		}	}}static voidconfigurerequest(XEvent *e) {	Client *c;	XConfigureRequestEvent *ev = &e->xconfigurerequest;	XWindowChanges wc;	if((c = getclient(ev->window))) {		c->ismax = False;		if(ev->value_mask & CWBorderWidth)			c->border = ev->border_width;		if(c->isfixed || c->isfloating || (lt->arrange == floating)) {			if(ev->value_mask & CWX)				c->x = ev->x;			if(ev->value_mask & CWY)				c->y = ev->y;			if(ev->value_mask & CWWidth)				c->w = ev->width;			if(ev->value_mask & CWHeight)				c->h = ev->height;			if((c->x + c->w) > sw && c->isfloating)				c->x = sw / 2 - c->w / 2; /* center in x direction */			if((c->y + c->h) > sh && c->isfloating)				c->y = sh / 2 - c->h / 2; /* center in y direction */			if((ev->value_mask & (CWX | CWY))			&& !(ev->value_mask & (CWWidth | CWHeight)))				configure(c);			if(isvisible(c))				XMoveResizeWindow(dpy, c->win, c->x, c->y, c->w, c->h);		}		else			configure(c);	}	else {		wc.x = ev->x;		wc.y = ev->y;		wc.width = ev->width;		wc.height = ev->height;		wc.border_width = ev->border_width;		wc.sibling = ev->above;		wc.stack_mode = ev->detail;		XConfigureWindow(dpy, ev->window, ev->value_mask, &wc);	}	XSync(dpy, False);}static voidconfigurenotify(XEvent *e) {	XConfigureEvent *ev = &e->xconfigure;	if (ev->window == root && (ev->width != sw || ev->height != sh)) {		sw = ev->width;		sh = ev->height;		XFreePixmap(dpy, dc.drawable);		dc.drawable = XCreatePixmap(dpy, root, sw, bh, DefaultDepth(dpy, screen));		XResizeWindow(dpy, barwin, sw, bh);		updatebarpos();		lt->arrange();	}}static voiddestroynotify(XEvent *e) {	Client *c;	XDestroyWindowEvent *ev = &e->xdestroywindow;	if((c = getclient(ev->window)))		unmanage(c);}static voidenternotify(XEvent *e) {	Client *c;	XCrossingEvent *ev = &e->xcrossing;	if(ev->mode != NotifyNormal || ev->detail == NotifyInferior)		return;	if(c = getclient(ev->window))		focus(c);	else if(ev->window == root) {		selscreen = True;		focus(NULL);	}}static voidexpose(XEvent *e) {	XExposeEvent *ev = &e->xexpose;	if(ev->count == 0) {		if(barwin == ev->window)			drawstatus();	}}static voidkeypress(XEvent *e) {	static unsigned int len = sizeof key / sizeof key[0];	unsigned int i;	KeySym keysym;	XKeyEvent *ev = &e->xkey;	keysym = XKeycodeToKeysym(dpy, (KeyCode)ev->keycode, 0);	for(i = 0; i < len; i++)		if(keysym == key[i].keysym		&& CLEANMASK(key[i].mod) == CLEANMASK(ev->state))		{			if(key[i].func)				key[i].func(key[i].arg);		}}static voidleavenotify(XEvent *e) {	XCrossingEvent *ev = &e->xcrossing;	if((ev->window == root) && !ev->same_screen) {		selscreen = False;		focus(NULL);	}}static voidmappingnotify(XEvent *e) {	XMappingEvent *ev = &e->xmapping;	XRefreshKeyboardMapping(ev);	if(ev->request == MappingKeyboard)		grabkeys();}static voidmaprequest(XEvent *e) {	static XWindowAttributes wa;	XMapRequestEvent *ev = &e->xmaprequest;	if(!XGetWindowAttributes(dpy, ev->window, &wa))		return;	if(wa.override_redirect)		return;	if(!getclient(ev->window))		manage(ev->window, &wa);}static voidpropertynotify(XEvent *e) {	Client *c;	Window trans;	XPropertyEvent *ev = &e->xproperty;	if(ev->state == PropertyDelete)		return; /* ignore */	if((c = getclient(ev->window))) {		switch (ev->atom) {			default: break;			case XA_WM_TRANSIENT_FOR:				XGetTransientForHint(dpy, c->win, &trans);				if(!c->isfloating && (c->isfloating = (getclient(trans) != NULL)))					lt->arrange();				break;			case XA_WM_NORMAL_HINTS:				updatesizehints(c);				break;		}		if(ev->atom == XA_WM_NAME || ev->atom == netatom[NetWMName]) {			updatetitle(c);			if(c == sel)				drawstatus();		}	}}static voidunmapnotify(XEvent *e) {	Client *c;	XUnmapEvent *ev = &e->xunmap;	if((c = getclient(ev->window)))		unmanage(c);}/* extern */void (*handler[LASTEvent]) (XEvent *) = {	[ButtonPress] = buttonpress,	[ConfigureRequest] = configurerequest,	[ConfigureNotify] = configurenotify,	[DestroyNotify] = destroynotify,	[EnterNotify] = enternotify,	[LeaveNotify] = leavenotify,	[Expose] = expose,	[KeyPress] = keypress,	[MappingNotify] = mappingnotify,	[MapRequest] = maprequest,	[PropertyNotify] = propertynotify,	[UnmapNotify] = unmapnotify};voidgrabkeys(void) {	static unsigned int len = sizeof key / sizeof key[0];	unsigned int i;	KeyCode code;	XUngrabKey(dpy, AnyKey, AnyModifier, root);	for(i = 0; i < len; i++) {		code = XKeysymToKeycode(dpy, key[i].keysym);		XGrabKey(dpy, code, key[i].mod, root, True,				GrabModeAsync, GrabModeAsync);		XGrabKey(dpy, code, key[i].mod | LockMask, root, True,				GrabModeAsync, GrabModeAsync);		XGrabKey(dpy, code, key[i].mod | numlockmask, root, True,				GrabModeAsync, GrabModeAsync);		XGrabKey(dpy, code, key[i].mod | numlockmask | LockMask, root, True,				GrabModeAsync, GrabModeAsync);	}}
 |