aewl
changeset 18:1efa34c6e1b6
added mouse-based resizals
author | Anselm R. Garbe <garbeam@wmii.de> |
---|---|
date | Tue, 11 Jul 2006 21:24:10 +0200 (2006-07-11) |
parents | aaf520f53110 |
children | b5510d0c6d43 |
files | Makefile README client.c event.c kb.c mouse.c wm.h |
diffstat | 8 files changed, 236 insertions(+), 74 deletions(-) [+] |
line diff
1.1 --- a/Makefile Tue Jul 11 18:59:09 2006 +0200 1.2 +++ b/Makefile Tue Jul 11 21:24:10 2006 +0200 1.3 @@ -3,7 +3,7 @@ 1.4 1.5 include config.mk 1.6 1.7 -WMSRC = bar.c client.c cmd.c draw.c event.c key.c util.c wm.c 1.8 +WMSRC = bar.c client.c cmd.c draw.c event.c kb.c mouse.c util.c wm.c 1.9 WMOBJ = ${WMSRC:.c=.o} 1.10 MENSRC = menu.c draw.c util.c 1.11 MENOBJ = ${MENSRC:.c=.o}
2.1 --- a/README Tue Jul 11 18:59:09 2006 +0200 2.2 +++ b/README Tue Jul 11 21:24:10 2006 +0200 2.3 @@ -5,14 +5,6 @@ 2.4 arranges all windows in a grid. 2.5 2.6 2.7 -Configuration 2.8 -------------- 2.9 -You have to edit the source code for configuration, this WM is intended to 2.10 -provide sane defaults, if something doesn't fits your needs, edit config.h and 2.11 -maybe key.c. To change the status output edit that status variable definition 2.12 -in wm.c. 2.13 - 2.14 - 2.15 Requirements 2.16 ------------ 2.17 In order to build gridwm you need the Xlib header files. 2.18 @@ -46,6 +38,6 @@ 2.19 Configuration 2.20 ------------- 2.21 The configuration of gridwm is done by customizing the config.h source file. To 2.22 -customize the key bindings edit key.c. To change the status output, edit the 2.23 +customize the key bindings edit kb.c. To change the status output, edit the 2.24 status command definition in wm.c. 2.25
3.1 --- a/client.c Tue Jul 11 18:59:09 2006 +0200 3.2 +++ b/client.c Tue Jul 11 21:24:10 2006 +0200 3.3 @@ -10,6 +10,8 @@ 3.4 #include "util.h" 3.5 #include "wm.h" 3.6 3.7 +#define CLIENT_MASK (StructureNotifyMask | PropertyChangeMask | EnterWindowMask) 3.8 + 3.9 void 3.10 update_name(Client *c) 3.11 { 3.12 @@ -70,7 +72,7 @@ 3.13 c->r[RFloat].height = wa->height; 3.14 c->border = wa->border_width; 3.15 XSetWindowBorderWidth(dpy, c->win, 0); 3.16 - XSelectInput(dpy, c->win, StructureNotifyMask | PropertyChangeMask | EnterWindowMask); 3.17 + XSelectInput(dpy, c->win, CLIENT_MASK); 3.18 XGetTransientForHint(dpy, c->win, &c->trans); 3.19 if(!XGetWMNormalHints(dpy, c->win, &c->size, &msize) || !c->size.flags) 3.20 c->size.flags = PSize; 3.21 @@ -95,9 +97,34 @@ 3.22 c->snext = stack; 3.23 stack = c; 3.24 XMapWindow(dpy, c->win); 3.25 + XGrabButton(dpy, AnyButton, Mod1Mask, c->win, False, ButtonPressMask, 3.26 + GrabModeAsync, GrabModeSync, None, None); 3.27 focus(c); 3.28 } 3.29 3.30 +void 3.31 +resize(Client *c) 3.32 +{ 3.33 + XConfigureEvent e; 3.34 + 3.35 + XMoveResizeWindow(dpy, c->win, c->r[RFloat].x, c->r[RFloat].y, 3.36 + c->r[RFloat].width, c->r[RFloat].height); 3.37 + e.type = ConfigureNotify; 3.38 + e.event = c->win; 3.39 + e.window = c->win; 3.40 + e.x = c->r[RFloat].x; 3.41 + e.y = c->r[RFloat].y; 3.42 + e.width = c->r[RFloat].width; 3.43 + e.height = c->r[RFloat].height; 3.44 + e.border_width = c->border; 3.45 + e.above = None; 3.46 + e.override_redirect = False; 3.47 + XSelectInput(dpy, c->win, CLIENT_MASK & ~StructureNotifyMask); 3.48 + XSendEvent(dpy, c->win, False, StructureNotifyMask, (XEvent *)&e); 3.49 + XSelectInput(dpy, c->win, CLIENT_MASK); 3.50 + XFlush(dpy); 3.51 +} 3.52 + 3.53 static int 3.54 dummy_error_handler(Display *dpy, XErrorEvent *error) 3.55 { 3.56 @@ -112,6 +139,7 @@ 3.57 XGrabServer(dpy); 3.58 XSetErrorHandler(dummy_error_handler); 3.59 3.60 + XUngrabButton(dpy, AnyButton, AnyModifier, c->win); 3.61 XUnmapWindow(dpy, c->win); 3.62 XDestroyWindow(dpy, c->title); 3.63 3.64 @@ -126,7 +154,7 @@ 3.65 XFlush(dpy); 3.66 XSetErrorHandler(error_handler); 3.67 XUngrabServer(dpy); 3.68 - flush_events(EnterWindowMask); 3.69 + discard_events(EnterWindowMask); 3.70 if(stack) 3.71 focus(stack); 3.72 }
4.1 --- a/event.c Tue Jul 11 18:59:09 2006 +0200 4.2 +++ b/event.c Tue Jul 11 21:24:10 2006 +0200 4.3 @@ -12,6 +12,7 @@ 4.4 #include "wm.h" 4.5 4.6 /* local functions */ 4.7 +static void buttonpress(XEvent *e); 4.8 static void configurerequest(XEvent *e); 4.9 static void destroynotify(XEvent *e); 4.10 static void enternotify(XEvent *e); 4.11 @@ -23,6 +24,7 @@ 4.12 static void unmapnotify(XEvent *e); 4.13 4.14 void (*handler[LASTEvent]) (XEvent *) = { 4.15 + [ButtonPress] = buttonpress, 4.16 [ConfigureRequest] = configurerequest, 4.17 [DestroyNotify] = destroynotify, 4.18 [EnterNotify] = enternotify, 4.19 @@ -36,7 +38,7 @@ 4.20 }; 4.21 4.22 unsigned int 4.23 -flush_events(long even_mask) 4.24 +discard_events(long even_mask) 4.25 { 4.26 XEvent ev; 4.27 unsigned int n = 0; 4.28 @@ -45,15 +47,37 @@ 4.29 } 4.30 4.31 static void 4.32 +buttonpress(XEvent *e) 4.33 +{ 4.34 + XButtonPressedEvent *ev = &e->xbutton; 4.35 + Client *c; 4.36 + 4.37 + if((c = getclient(ev->window))) { 4.38 + switch(ev->button) { 4.39 + default: 4.40 + break; 4.41 + case Button1: 4.42 + mmove(c); 4.43 + break; 4.44 + case Button2: 4.45 + XLowerWindow(dpy, c->win); 4.46 + break; 4.47 + case Button3: 4.48 + mresize(c); 4.49 + break; 4.50 + } 4.51 + } 4.52 +} 4.53 + 4.54 +static void 4.55 configurerequest(XEvent *e) 4.56 { 4.57 XConfigureRequestEvent *ev = &e->xconfigurerequest; 4.58 XWindowChanges wc; 4.59 Client *c; 4.60 4.61 - c = getclient(ev->window); 4.62 ev->value_mask &= ~CWSibling; 4.63 - if(c) { 4.64 + if((c = getclient(ev->window))) { 4.65 if(ev->value_mask & CWX) 4.66 c->r[RFloat].x = ev->x; 4.67 if(ev->value_mask & CWY)
5.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 5.2 +++ b/kb.c Tue Jul 11 21:24:10 2006 +0200 5.3 @@ -0,0 +1,55 @@ 5.4 +/* 5.5 + * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> 5.6 + * See LICENSE file for license details. 5.7 + */ 5.8 + 5.9 +#include "wm.h" 5.10 + 5.11 +#include <X11/keysym.h> 5.12 + 5.13 +static const char *term[] = { 5.14 + "xterm", "-u8", "-bg", "black", "-fg", "white", "-fn", 5.15 + "-*-terminus-medium-*-*-*-13-*-*-*-*-*-iso10646-*", 0 5.16 +}; 5.17 + 5.18 +static const char *proglist[] = { 5.19 + "sh", "-c", "exec `ls -lL /bin /sbin /usr/bin /usr/local/bin 2>/dev/null | awk 'NF>2 && $1 ~ /^[^d].*x/ {print $NF}' | sort | uniq | gridmenu`", 0 5.20 +}; 5.21 + 5.22 +static Key key[] = { 5.23 + { Mod1Mask, XK_Return, run, term }, 5.24 + { Mod1Mask, XK_p, run, proglist }, 5.25 + { Mod1Mask | ShiftMask, XK_c, kill, NULL}, 5.26 + { Mod1Mask | ShiftMask, XK_q, quit, NULL}, 5.27 +}; 5.28 + 5.29 +void 5.30 +update_keys() 5.31 +{ 5.32 + unsigned int i, len; 5.33 + KeyCode code; 5.34 + 5.35 + len = sizeof(key) / sizeof(key[0]); 5.36 + for(i = 0; i < len; i++) { 5.37 + code = XKeysymToKeycode(dpy, key[i].keysym); 5.38 + XUngrabKey(dpy, code, key[i].mod, root); 5.39 + XGrabKey(dpy, code, key[i].mod, root, True, GrabModeAsync, GrabModeAsync); 5.40 + } 5.41 +} 5.42 + 5.43 +void 5.44 +keypress(XEvent *e) 5.45 +{ 5.46 + XKeyEvent *ev = &e->xkey; 5.47 + unsigned int i, len; 5.48 + KeySym keysym; 5.49 + 5.50 + keysym = XKeycodeToKeysym(dpy, (KeyCode)ev->keycode, 0); 5.51 + len = sizeof(key) / sizeof(key[0]); 5.52 + for(i = 0; i < len; i++) 5.53 + if((keysym == key[i].keysym) && (key[i].mod == ev->state)) { 5.54 + if(key[i].func) 5.55 + key[i].func(key[i].aux); 5.56 + return; 5.57 + } 5.58 +}
6.1 --- a/key.c Tue Jul 11 18:59:09 2006 +0200 6.2 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 6.3 @@ -1,55 +0,0 @@ 6.4 -/* 6.5 - * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> 6.6 - * See LICENSE file for license details. 6.7 - */ 6.8 - 6.9 -#include "wm.h" 6.10 - 6.11 -#include <X11/keysym.h> 6.12 - 6.13 -static const char *term[] = { 6.14 - "xterm", "-u8", "-bg", "black", "-fg", "white", "-fn", 6.15 - "-*-terminus-medium-*-*-*-13-*-*-*-*-*-iso10646-*", 0 6.16 -}; 6.17 - 6.18 -static const char *proglist[] = { 6.19 - "sh", "-c", "exec `ls -lL /bin /sbin /usr/bin /usr/local/bin 2>/dev/null | awk 'NF>2 && $1 ~ /^[^d].*x/ {print $NF}' | sort | uniq | gridmenu`", 0 6.20 -}; 6.21 - 6.22 -static Key key[] = { 6.23 - { Mod1Mask, XK_Return, run, term }, 6.24 - { Mod1Mask, XK_p, run, proglist }, 6.25 - { Mod1Mask | ShiftMask, XK_c, kill, NULL}, 6.26 - { Mod1Mask | ShiftMask, XK_q, quit, NULL}, 6.27 -}; 6.28 - 6.29 -void 6.30 -update_keys() 6.31 -{ 6.32 - unsigned int i, len; 6.33 - KeyCode code; 6.34 - 6.35 - len = sizeof(key) / sizeof(key[0]); 6.36 - for(i = 0; i < len; i++) { 6.37 - code = XKeysymToKeycode(dpy, key[i].keysym); 6.38 - XUngrabKey(dpy, code, key[i].mod, root); 6.39 - XGrabKey(dpy, code, key[i].mod, root, True, GrabModeAsync, GrabModeAsync); 6.40 - } 6.41 -} 6.42 - 6.43 -void 6.44 -keypress(XEvent *e) 6.45 -{ 6.46 - XKeyEvent *ev = &e->xkey; 6.47 - unsigned int i, len; 6.48 - KeySym keysym; 6.49 - 6.50 - keysym = XKeycodeToKeysym(dpy, (KeyCode)ev->keycode, 0); 6.51 - len = sizeof(key) / sizeof(key[0]); 6.52 - for(i = 0; i < len; i++) 6.53 - if((keysym == key[i].keysym) && (key[i].mod == ev->state)) { 6.54 - if(key[i].func) 6.55 - key[i].func(key[i].aux); 6.56 - return; 6.57 - } 6.58 -}
7.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 7.2 +++ b/mouse.c Tue Jul 11 21:24:10 2006 +0200 7.3 @@ -0,0 +1,100 @@ 7.4 +/* 7.5 + * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> 7.6 + * (C)opyright MMVI Kris Maglione <fbsdaemon@gmail.com> 7.7 + * See LICENSE file for license details. 7.8 + */ 7.9 + 7.10 +#include <stdlib.h> 7.11 +#include <string.h> 7.12 +#include <unistd.h> 7.13 + 7.14 +#include "wm.h" 7.15 + 7.16 +#define ButtonMask (ButtonPressMask | ButtonReleaseMask) 7.17 +#define MouseMask (ButtonMask | PointerMotionMask) 7.18 + 7.19 +static void 7.20 +mmatch(Client *c, int x1, int y1, int x2, int y2) 7.21 +{ 7.22 + c->r[RFloat].width = abs(x1 - x2); 7.23 + c->r[RFloat].height = abs(y1 - y2); 7.24 + c->r[RFloat].width -= 7.25 + (c->r[RFloat].width - c->size.base_width) % c->size.width_inc; 7.26 + c->r[RFloat].height -= 7.27 + (c->r[RFloat].height - c->size.base_height) % c->size.height_inc; 7.28 + if(c->size.min_width && c->r[RFloat].width < c->size.min_width) 7.29 + c->r[RFloat].width = c->size.min_width; 7.30 + if(c->size.min_height && c->r[RFloat].height < c->size.min_height) 7.31 + c->r[RFloat].height = c->size.min_height; 7.32 + if(c->size.max_width && c->r[RFloat].width > c->size.max_width) 7.33 + c->r[RFloat].width = c->size.max_width; 7.34 + if(c->size.max_height && c->r[RFloat].height > c->size.max_height) 7.35 + c->r[RFloat].height = c->size.max_height; 7.36 + c->r[RFloat].x = (x1 <= x2) ? x1 : x1 - c->r[RFloat].width; 7.37 + c->r[RFloat].y = (y1 <= y2) ? y1 : y1 - c->r[RFloat].height; 7.38 +} 7.39 + 7.40 +void 7.41 +mresize(Client *c) 7.42 +{ 7.43 + XEvent ev; 7.44 + int old_cx, old_cy; 7.45 + 7.46 + old_cx = c->r[RFloat].x; 7.47 + old_cy = c->r[RFloat].y; 7.48 + if(XGrabPointer(dpy, c->win, False, MouseMask, GrabModeAsync, GrabModeAsync, 7.49 + None, cursor[CurResize], CurrentTime) != GrabSuccess) 7.50 + return; 7.51 + XGrabServer(dpy); 7.52 + XWarpPointer(dpy, None, c->win, 0, 0, 0, 0, 7.53 + c->r[RFloat].width, c->r[RFloat].height); 7.54 + for(;;) { 7.55 + XMaskEvent(dpy, MouseMask, &ev); 7.56 + switch(ev.type) { 7.57 + default: break; 7.58 + case MotionNotify: 7.59 + XUngrabServer(dpy); 7.60 + mmatch(c, old_cx, old_cy, ev.xmotion.x, ev.xmotion.y); 7.61 + resize(c); 7.62 + XGrabServer(dpy); 7.63 + break; 7.64 + case ButtonRelease: 7.65 + XUngrabPointer(dpy, CurrentTime); 7.66 + return; 7.67 + } 7.68 + } 7.69 +} 7.70 + 7.71 +void 7.72 +mmove(Client *c) 7.73 +{ 7.74 + XEvent ev; 7.75 + int x1, y1, old_cx, old_cy, di; 7.76 + unsigned int dui; 7.77 + Window dummy; 7.78 + 7.79 + old_cx = c->r[RFloat].x; 7.80 + old_cy = c->r[RFloat].y; 7.81 + if(XGrabPointer(dpy, c->win, False, MouseMask, GrabModeAsync, GrabModeAsync, 7.82 + None, cursor[CurMove], CurrentTime) != GrabSuccess) 7.83 + return; 7.84 + XQueryPointer(dpy, root, &dummy, &dummy, &x1, &y1, &di, &di, &dui); 7.85 + XGrabServer(dpy); 7.86 + for(;;) { 7.87 + XMaskEvent(dpy, MouseMask, &ev); 7.88 + switch (ev.type) { 7.89 + default: break; 7.90 + case MotionNotify: 7.91 + XUngrabServer(dpy); 7.92 + c->r[RFloat].x = old_cx + (ev.xmotion.x - x1); 7.93 + c->r[RFloat].y = old_cy + (ev.xmotion.y - y1); 7.94 + resize(c); 7.95 + XGrabServer(dpy); 7.96 + break; 7.97 + case ButtonRelease: 7.98 + XUngrabServer(dpy); 7.99 + XUngrabPointer(dpy, CurrentTime); 7.100 + return; 7.101 + } 7.102 + } 7.103 +}
8.1 --- a/wm.h Tue Jul 11 18:59:09 2006 +0200 8.2 +++ b/wm.h Tue Jul 11 21:24:10 2006 +0200 8.3 @@ -11,6 +11,22 @@ 8.4 8.5 #define WM_PROTOCOL_DELWIN 1 8.6 8.7 +typedef struct Client Client; 8.8 +typedef struct Key Key; 8.9 +typedef enum Align Align; 8.10 + 8.11 +enum Align { 8.12 + NORTH = 0x01, 8.13 + EAST = 0x02, 8.14 + SOUTH = 0x04, 8.15 + WEST = 0x08, 8.16 + NEAST = NORTH | EAST, 8.17 + NWEST = NORTH | WEST, 8.18 + SEAST = SOUTH | EAST, 8.19 + SWEST = SOUTH | WEST, 8.20 + CENTER = NEAST | SWEST 8.21 +}; 8.22 + 8.23 /* atoms */ 8.24 enum { WMProtocols, WMDelete, WMLast }; 8.25 enum { NetSupported, NetWMName, NetLast }; 8.26 @@ -21,9 +37,6 @@ 8.27 /* rects */ 8.28 enum { RFloat, RGrid, RLast }; 8.29 8.30 -typedef struct Client Client; 8.31 -typedef struct Key Key; 8.32 - 8.33 struct Client { 8.34 char name[256]; 8.35 char tag[256]; 8.36 @@ -75,14 +88,19 @@ 8.37 extern void focus(Client *c); 8.38 extern void update_name(Client *c); 8.39 extern void draw_client(Client *c); 8.40 +extern void resize(Client *c); 8.41 8.42 /* event.c */ 8.43 -extern unsigned int flush_events(long even_mask); 8.44 +extern unsigned int discard_events(long even_mask); 8.45 8.46 /* key.c */ 8.47 extern void update_keys(); 8.48 extern void keypress(XEvent *e); 8.49 8.50 +/* mouse.c */ 8.51 +extern void mresize(Client *c); 8.52 +extern void mmove(Client *c); 8.53 + 8.54 /* wm.c */ 8.55 extern int error_handler(Display *dpy, XErrorEvent *error); 8.56 extern void send_message(Window w, Atom a, long value);