Mercurial > hg > index.fcgi > dwm > dwm-3.6.1-11pba
diff event.c @ 0:7024076fa948
initial add
author | paulo@localhost |
---|---|
date | Sun, 22 Mar 2009 23:26:35 -0700 |
parents | |
children | de6bb7885c97 |
line diff
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/event.c Sun Mar 22 23:26:35 2009 -0700 1.3 @@ -0,0 +1,380 @@ 1.4 +/* (C)opyright MMVI-MMVII Anselm R. Garbe <garbeam at gmail dot com> 1.5 + * See LICENSE file for license details. 1.6 + */ 1.7 +#include "dwm.h" 1.8 +#include <stdlib.h> 1.9 +#include <X11/keysym.h> 1.10 +#include <X11/Xatom.h> 1.11 + 1.12 +/* static */ 1.13 + 1.14 +typedef struct { 1.15 + unsigned long mod; 1.16 + KeySym keysym; 1.17 + void (*func)(Arg *arg); 1.18 + Arg arg; 1.19 +} Key; 1.20 + 1.21 +KEYS 1.22 + 1.23 +#define CLEANMASK(mask) (mask & ~(numlockmask | LockMask)) 1.24 +#define MOUSEMASK (BUTTONMASK | PointerMotionMask) 1.25 + 1.26 +static Client * 1.27 +getclient(Window w) { 1.28 + Client *c; 1.29 + 1.30 + for(c = clients; c && c->win != w; c = c->next); 1.31 + return c; 1.32 +} 1.33 + 1.34 +static void 1.35 +movemouse(Client *c) { 1.36 + int x1, y1, ocx, ocy, di, nx, ny; 1.37 + unsigned int dui; 1.38 + Window dummy; 1.39 + XEvent ev; 1.40 + 1.41 + ocx = nx = c->x; 1.42 + ocy = ny = c->y; 1.43 + if(XGrabPointer(dpy, root, False, MOUSEMASK, GrabModeAsync, GrabModeAsync, 1.44 + None, cursor[CurMove], CurrentTime) != GrabSuccess) 1.45 + return; 1.46 + c->ismax = False; 1.47 + XQueryPointer(dpy, root, &dummy, &dummy, &x1, &y1, &di, &di, &dui); 1.48 + for(;;) { 1.49 + XMaskEvent(dpy, MOUSEMASK | ExposureMask | SubstructureRedirectMask, &ev); 1.50 + switch (ev.type) { 1.51 + case ButtonRelease: 1.52 + XUngrabPointer(dpy, CurrentTime); 1.53 + return; 1.54 + case ConfigureRequest: 1.55 + case Expose: 1.56 + case MapRequest: 1.57 + handler[ev.type](&ev); 1.58 + break; 1.59 + case MotionNotify: 1.60 + XSync(dpy, False); 1.61 + nx = ocx + (ev.xmotion.x - x1); 1.62 + ny = ocy + (ev.xmotion.y - y1); 1.63 + if(abs(wax + nx) < SNAP) 1.64 + nx = wax; 1.65 + else if(abs((wax + waw) - (nx + c->w + 2 * c->border)) < SNAP) 1.66 + nx = wax + waw - c->w - 2 * c->border; 1.67 + if(abs(way - ny) < SNAP) 1.68 + ny = way; 1.69 + else if(abs((way + wah) - (ny + c->h + 2 * c->border)) < SNAP) 1.70 + ny = way + wah - c->h - 2 * c->border; 1.71 + resize(c, nx, ny, c->w, c->h, False); 1.72 + break; 1.73 + } 1.74 + } 1.75 +} 1.76 + 1.77 +static void 1.78 +resizemouse(Client *c) { 1.79 + int ocx, ocy; 1.80 + int nw, nh; 1.81 + XEvent ev; 1.82 + 1.83 + ocx = c->x; 1.84 + ocy = c->y; 1.85 + if(XGrabPointer(dpy, root, False, MOUSEMASK, GrabModeAsync, GrabModeAsync, 1.86 + None, cursor[CurResize], CurrentTime) != GrabSuccess) 1.87 + return; 1.88 + c->ismax = False; 1.89 + XWarpPointer(dpy, None, c->win, 0, 0, 0, 0, c->w + c->border - 1, c->h + c->border - 1); 1.90 + for(;;) { 1.91 + XMaskEvent(dpy, MOUSEMASK | ExposureMask | SubstructureRedirectMask , &ev); 1.92 + switch(ev.type) { 1.93 + case ButtonRelease: 1.94 + XWarpPointer(dpy, None, c->win, 0, 0, 0, 0, 1.95 + c->w + c->border - 1, c->h + c->border - 1); 1.96 + XUngrabPointer(dpy, CurrentTime); 1.97 + while(XCheckMaskEvent(dpy, EnterWindowMask, &ev)); 1.98 + return; 1.99 + case ConfigureRequest: 1.100 + case Expose: 1.101 + case MapRequest: 1.102 + handler[ev.type](&ev); 1.103 + break; 1.104 + case MotionNotify: 1.105 + XSync(dpy, False); 1.106 + if((nw = ev.xmotion.x - ocx - 2 * c->border + 1) <= 0) 1.107 + nw = 1; 1.108 + if((nh = ev.xmotion.y - ocy - 2 * c->border + 1) <= 0) 1.109 + nh = 1; 1.110 + resize(c, c->x, c->y, nw, nh, True); 1.111 + break; 1.112 + } 1.113 + } 1.114 +} 1.115 + 1.116 +static void 1.117 +buttonpress(XEvent *e) { 1.118 + int x; 1.119 + Arg a; 1.120 + Client *c; 1.121 + XButtonPressedEvent *ev = &e->xbutton; 1.122 + 1.123 + if(barwin == ev->window) { 1.124 + x = 0; 1.125 + for(a.i = 0; a.i < ntags; a.i++) { 1.126 + x += textw(tags[a.i]); 1.127 + if(ev->x < x) { 1.128 + if(ev->button == Button1) { 1.129 + if(ev->state & MODKEY) 1.130 + tag(&a); 1.131 + else 1.132 + view(&a); 1.133 + } 1.134 + else if(ev->button == Button3) { 1.135 + if(ev->state & MODKEY) 1.136 + toggletag(&a); 1.137 + else 1.138 + toggleview(&a); 1.139 + } 1.140 + return; 1.141 + } 1.142 + } 1.143 + if(ev->x < x + blw) 1.144 + switch(ev->button) { 1.145 + case Button1: 1.146 + a.i = -1; 1.147 + setlayout(&a); 1.148 + break; 1.149 + case Button4: 1.150 + a.i = 1; 1.151 + incnmaster(&a); 1.152 + break; 1.153 + case Button5: 1.154 + a.i = -1; 1.155 + incnmaster(&a); 1.156 + break; 1.157 + } 1.158 + if(ev->x > x + blw) 1.159 + switch(ev->button) { 1.160 + case Button1: zoom(NULL); break; 1.161 + case Button2: toggleversatile(NULL); break; 1.162 + case Button3: killclient(NULL); break; 1.163 + case Button4: focusprev(NULL); break; 1.164 + case Button5: focusnext(NULL); break; 1.165 + } 1.166 + } 1.167 + else if((c = getclient(ev->window))) { 1.168 + focus(c); 1.169 + if(CLEANMASK(ev->state) != MODKEY) 1.170 + return; 1.171 + if(ev->button == Button1 && (lt->arrange == versatile || c->isversatile)) { 1.172 + restack(); 1.173 + movemouse(c); 1.174 + } 1.175 + else if(ev->button == Button2) 1.176 + zoom(NULL); 1.177 + else if(ev->button == Button3 1.178 + && (lt->arrange == versatile || c->isversatile) && !c->isfixed) 1.179 + { 1.180 + restack(); 1.181 + resizemouse(c); 1.182 + } 1.183 + } 1.184 +} 1.185 + 1.186 +static void 1.187 +configurerequest(XEvent *e) { 1.188 + Client *c; 1.189 + XConfigureRequestEvent *ev = &e->xconfigurerequest; 1.190 + XWindowChanges wc; 1.191 + 1.192 + if((c = getclient(ev->window))) { 1.193 + c->ismax = False; 1.194 + if(ev->value_mask & CWBorderWidth) 1.195 + c->border = ev->border_width; 1.196 + if(c->isfixed || c->isversatile || (lt->arrange == versatile)) { 1.197 + if(ev->value_mask & CWX) 1.198 + c->x = ev->x; 1.199 + if(ev->value_mask & CWY) 1.200 + c->y = ev->y; 1.201 + if(ev->value_mask & CWWidth) 1.202 + c->w = ev->width; 1.203 + if(ev->value_mask & CWHeight) 1.204 + c->h = ev->height; 1.205 + if((ev->value_mask & (CWX | CWY)) 1.206 + && !(ev->value_mask & (CWWidth | CWHeight))) 1.207 + configure(c); 1.208 + if(isvisible(c)) 1.209 + /* XMoveResizeWindow(dpy, c->win, c->x, c->y, c->w, c->h); */ 1.210 + resize(c, c->x, c->y, c->w, c->h, False); 1.211 + } 1.212 + else 1.213 + configure(c); 1.214 + } 1.215 + else { 1.216 + wc.x = ev->x; 1.217 + wc.y = ev->y; 1.218 + wc.width = ev->width; 1.219 + wc.height = ev->height; 1.220 + wc.border_width = ev->border_width; 1.221 + wc.sibling = ev->above; 1.222 + wc.stack_mode = ev->detail; 1.223 + XConfigureWindow(dpy, ev->window, ev->value_mask, &wc); 1.224 + } 1.225 + XSync(dpy, False); 1.226 +} 1.227 + 1.228 +static void 1.229 +destroynotify(XEvent *e) { 1.230 + Client *c; 1.231 + XDestroyWindowEvent *ev = &e->xdestroywindow; 1.232 + 1.233 + if((c = getclient(ev->window))) 1.234 + unmanage(c); 1.235 +} 1.236 + 1.237 +static void 1.238 +enternotify(XEvent *e) { 1.239 + Client *c; 1.240 + XCrossingEvent *ev = &e->xcrossing; 1.241 + 1.242 + if(ev->mode != NotifyNormal || ev->detail == NotifyInferior) 1.243 + return; 1.244 + if((c = getclient(ev->window)) && isvisible(c)) 1.245 + focus(c); 1.246 + else if(ev->window == root) { 1.247 + selscreen = True; 1.248 + for(c = stack; c && !isvisible(c); c = c->snext); 1.249 + focus(c); 1.250 + } 1.251 +} 1.252 + 1.253 +static void 1.254 +expose(XEvent *e) { 1.255 + XExposeEvent *ev = &e->xexpose; 1.256 + 1.257 + if(ev->count == 0) { 1.258 + if(barwin == ev->window) 1.259 + drawstatus(); 1.260 + } 1.261 +} 1.262 + 1.263 +static void 1.264 +keypress(XEvent *e) { 1.265 + static unsigned int len = sizeof key / sizeof key[0]; 1.266 + unsigned int i; 1.267 + KeySym keysym; 1.268 + XKeyEvent *ev = &e->xkey; 1.269 + 1.270 + keysym = XKeycodeToKeysym(dpy, (KeyCode)ev->keycode, 0); 1.271 + for(i = 0; i < len; i++) 1.272 + if(keysym == key[i].keysym 1.273 + && CLEANMASK(key[i].mod) == CLEANMASK(ev->state)) 1.274 + { 1.275 + if(key[i].func) 1.276 + key[i].func(&key[i].arg); 1.277 + } 1.278 +} 1.279 + 1.280 +static void 1.281 +leavenotify(XEvent *e) { 1.282 + XCrossingEvent *ev = &e->xcrossing; 1.283 + 1.284 + if((ev->window == root) && !ev->same_screen) { 1.285 + selscreen = False; 1.286 + focus(NULL); 1.287 + } 1.288 +} 1.289 + 1.290 +static void 1.291 +mappingnotify(XEvent *e) { 1.292 + XMappingEvent *ev = &e->xmapping; 1.293 + 1.294 + XRefreshKeyboardMapping(ev); 1.295 + if(ev->request == MappingKeyboard) 1.296 + grabkeys(); 1.297 +} 1.298 + 1.299 +static void 1.300 +maprequest(XEvent *e) { 1.301 + static XWindowAttributes wa; 1.302 + XMapRequestEvent *ev = &e->xmaprequest; 1.303 + 1.304 + if(!XGetWindowAttributes(dpy, ev->window, &wa)) 1.305 + return; 1.306 + if(wa.override_redirect) 1.307 + return; 1.308 + if(!getclient(ev->window)) 1.309 + manage(ev->window, &wa); 1.310 +} 1.311 + 1.312 +static void 1.313 +propertynotify(XEvent *e) { 1.314 + Client *c; 1.315 + Window trans; 1.316 + XPropertyEvent *ev = &e->xproperty; 1.317 + 1.318 + if(ev->state == PropertyDelete) 1.319 + return; /* ignore */ 1.320 + if((c = getclient(ev->window))) { 1.321 + switch (ev->atom) { 1.322 + default: break; 1.323 + case XA_WM_TRANSIENT_FOR: 1.324 + XGetTransientForHint(dpy, c->win, &trans); 1.325 + if(!c->isversatile && (c->isversatile = (getclient(trans) != NULL))) 1.326 + lt->arrange(); 1.327 + break; 1.328 + case XA_WM_NORMAL_HINTS: 1.329 + updatesizehints(c); 1.330 + break; 1.331 + } 1.332 + if(ev->atom == XA_WM_NAME || ev->atom == netatom[NetWMName]) { 1.333 + updatetitle(c); 1.334 + if(c == sel) 1.335 + drawstatus(); 1.336 + } 1.337 + } 1.338 +} 1.339 + 1.340 +static void 1.341 +unmapnotify(XEvent *e) { 1.342 + Client *c; 1.343 + XUnmapEvent *ev = &e->xunmap; 1.344 + 1.345 + if((c = getclient(ev->window))) 1.346 + unmanage(c); 1.347 +} 1.348 + 1.349 +/* extern */ 1.350 + 1.351 +void (*handler[LASTEvent]) (XEvent *) = { 1.352 + [ButtonPress] = buttonpress, 1.353 + [ConfigureRequest] = configurerequest, 1.354 + [DestroyNotify] = destroynotify, 1.355 + [EnterNotify] = enternotify, 1.356 + [LeaveNotify] = leavenotify, 1.357 + [Expose] = expose, 1.358 + [KeyPress] = keypress, 1.359 + [MappingNotify] = mappingnotify, 1.360 + [MapRequest] = maprequest, 1.361 + [PropertyNotify] = propertynotify, 1.362 + [UnmapNotify] = unmapnotify 1.363 +}; 1.364 + 1.365 +void 1.366 +grabkeys(void) { 1.367 + static unsigned int len = sizeof key / sizeof key[0]; 1.368 + unsigned int i; 1.369 + KeyCode code; 1.370 + 1.371 + XUngrabKey(dpy, AnyKey, AnyModifier, root); 1.372 + for(i = 0; i < len; i++) { 1.373 + code = XKeysymToKeycode(dpy, key[i].keysym); 1.374 + XGrabKey(dpy, code, key[i].mod, root, True, 1.375 + GrabModeAsync, GrabModeAsync); 1.376 + XGrabKey(dpy, code, key[i].mod | LockMask, root, True, 1.377 + GrabModeAsync, GrabModeAsync); 1.378 + XGrabKey(dpy, code, key[i].mod | numlockmask, root, True, 1.379 + GrabModeAsync, GrabModeAsync); 1.380 + XGrabKey(dpy, code, key[i].mod | numlockmask | LockMask, root, True, 1.381 + GrabModeAsync, GrabModeAsync); 1.382 + } 1.383 +}