about summary refs log tree commit diff stats
path: root/mouse.c
diff options
context:
space:
mode:
authorAnselm R. Garbe <garbeam@wmii.de>2006-07-11 21:24:10 +0200
committerAnselm R. Garbe <garbeam@wmii.de>2006-07-11 21:24:10 +0200
commitb9da4b082eb658b4142b61c149212f414f7653b6 (patch)
tree146a4e480866d1497c3ec3ccf2b71305b07a5f03 /mouse.c
parent5ed16faecb94b551ea00ea940e8d719211576de8 (diff)
downloaddwm-b9da4b082eb658b4142b61c149212f414f7653b6.tar.gz
added mouse-based resizals
Diffstat (limited to 'mouse.c')
-rw-r--r--mouse.c100
1 files changed, 100 insertions, 0 deletions
diff --git a/mouse.c b/mouse.c
new file mode 100644
index 0000000..2b5d63b
--- /dev/null
+++ b/mouse.c
@@ -0,0 +1,100 @@
+/*
+ * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com>
+ * (C)opyright MMVI Kris Maglione <fbsdaemon@gmail.com>
+ * See LICENSE file for license details.
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+
+#include "wm.h"
+
+#define ButtonMask      (ButtonPressMask | ButtonReleaseMask)
+#define MouseMask       (ButtonMask | PointerMotionMask)
+
+static void
+mmatch(Client *c, int x1, int y1, int x2, int y2)
+{
+	c->r[RFloat].width = abs(x1 - x2);
+	c->r[RFloat].height = abs(y1 - y2);
+	c->r[RFloat].width -=
+		(c->r[RFloat].width - c->size.base_width) % c->size.width_inc;
+	c->r[RFloat].height -=
+		(c->r[RFloat].height - c->size.base_height) % c->size.height_inc;
+	if(c->size.min_width && c->r[RFloat].width < c->size.min_width)
+		c->r[RFloat].width = c->size.min_width;
+	if(c->size.min_height && c->r[RFloat].height < c->size.min_height)
+		c->r[RFloat].height = c->size.min_height;
+	if(c->size.max_width && c->r[RFloat].width > c->size.max_width)
+		c->r[RFloat].width = c->size.max_width;
+	if(c->size.max_height && c->r[RFloat].height > c->size.max_height)
+		c->r[RFloat].height = c->size.max_height;
+	c->r[RFloat].x = (x1 <= x2) ? x1 : x1 - c->r[RFloat].width;
+	c->r[RFloat].y = (y1 <= y2) ? y1 : y1 - c->r[RFloat].height;
+}
+
+void
+mresize(Client *c)
+{
+	XEvent ev;
+	int old_cx, old_cy;
+
+	old_cx = c->r[RFloat].x;
+	old_cy = c->r[RFloat].y;
+	if(XGrabPointer(dpy, c->win, False, MouseMask, GrabModeAsync, GrabModeAsync,
+				None, cursor[CurResize], CurrentTime) != GrabSuccess)
+		return;
+	XGrabServer(dpy);
+	XWarpPointer(dpy, None, c->win, 0, 0, 0, 0,
+			c->r[RFloat].width, c->r[RFloat].height);
+	for(;;) {
+		XMaskEvent(dpy, MouseMask, &ev);
+		switch(ev.type) {
+		default: break;
+		case MotionNotify:
+			XUngrabServer(dpy);
+			mmatch(c, old_cx, old_cy, ev.xmotion.x, ev.xmotion.y);
+			resize(c);
+			XGrabServer(dpy);
+			break;
+		case ButtonRelease:
+			XUngrabPointer(dpy, CurrentTime);
+			return;
+		}
+	}
+}
+
+void
+mmove(Client *c)
+{
+	XEvent ev;
+	int x1, y1, old_cx, old_cy, di;
+	unsigned int dui;
+	Window dummy;
+
+	old_cx = c->r[RFloat].x;
+	old_cy = c->r[RFloat].y;
+	if(XGrabPointer(dpy, c->win, False, MouseMask, GrabModeAsync, GrabModeAsync,
+				None, cursor[CurMove], CurrentTime) != GrabSuccess)
+		return;
+	XQueryPointer(dpy, root, &dummy, &dummy, &x1, &y1, &di, &di, &dui);
+	XGrabServer(dpy);
+	for(;;) {
+		XMaskEvent(dpy, MouseMask, &ev);
+		switch (ev.type) {
+		default: break;
+		case MotionNotify:
+			XUngrabServer(dpy);
+			c->r[RFloat].x = old_cx + (ev.xmotion.x - x1);
+			c->r[RFloat].y = old_cy + (ev.xmotion.y - y1);
+			resize(c);
+			XGrabServer(dpy);
+			break;
+		case ButtonRelease:
+			XUngrabServer(dpy);
+			XUngrabPointer(dpy, CurrentTime);
+			return;
+		}
+	}
+}