From 7c4ed94757dcc7c62d6bee03aeeb7b2947c84916 Mon Sep 17 00:00:00 2001 From: okan Date: Mon, 7 Jan 2013 20:32:19 +0000 Subject: [PATCH 1/7] unbreak xinerama support from r1.41 for panning setups --- screen.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/screen.c b/screen.c index 01c2f73..d0518c5 100644 --- a/screen.c +++ b/screen.c @@ -143,8 +143,8 @@ screen_find_xinerama(struct screen_ctx *sc, int x, int y) y >= info->y_org && y < info->y_org + info->height) { geom.x = info->x_org; geom.y = info->y_org; - geom.w = info->width; - geom.h = info->height; + geom.w = info->x_org + info->width; + geom.h = info->y_org + info->height; break; } } From 4ffe56b9a3dfe958c1ad51a0064efe8094a306da Mon Sep 17 00:00:00 2001 From: okan Date: Mon, 7 Jan 2013 21:45:24 +0000 Subject: [PATCH 2/7] revert previous --- screen.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/screen.c b/screen.c index d0518c5..01c2f73 100644 --- a/screen.c +++ b/screen.c @@ -143,8 +143,8 @@ screen_find_xinerama(struct screen_ctx *sc, int x, int y) y >= info->y_org && y < info->y_org + info->height) { geom.x = info->x_org; geom.y = info->y_org; - geom.w = info->x_org + info->width; - geom.h = info->y_org + info->height; + geom.w = info->width; + geom.h = info->height; break; } } From e7b85cfb2f4d04e283d448cc3a09014e858eeb94 Mon Sep 17 00:00:00 2001 From: okan Date: Mon, 7 Jan 2013 21:53:23 +0000 Subject: [PATCH 3/7] fix menu/client placement in panning setups; XineramaQueryScreens gives us the width of the psuedo screen, but here we need the edge instead (xmax/ymax); just re-use w/h here for now. --- client.c | 2 ++ menu.c | 2 ++ 2 files changed, 4 insertions(+) diff --git a/client.c b/client.c index 530a077..6fcfb88 100644 --- a/client.c +++ b/client.c @@ -660,6 +660,8 @@ client_placecalc(struct client_ctx *cc) xu_ptr_getpos(sc->rootwin, &xmouse, &ymouse); xine = screen_find_xinerama(sc, xmouse, ymouse); + xine.w += xine.x; + xine.h += xine.y; xmouse = MAX(xmouse, xine.x) - cc->geom.w / 2; ymouse = MAX(ymouse, xine.y) - cc->geom.h / 2; diff --git a/menu.c b/menu.c index f0d8f54..b392a08 100644 --- a/menu.c +++ b/menu.c @@ -394,6 +394,8 @@ menu_draw(struct screen_ctx *sc, struct menu_ctx *mc, struct menu_q *menuq, } xine = screen_find_xinerama(sc, mc->x, mc->y); + xine.w += xine.x; + xine.h += xine.y; xsave = mc->x; ysave = mc->y; From bf9d9815976c0afae2e87afeebaa57b02b1d911c Mon Sep 17 00:00:00 2001 From: okan Date: Tue, 8 Jan 2013 04:12:51 +0000 Subject: [PATCH 4/7] teach screen_find_xinerama() about gap and adjust (simplify) callers; menu becomes gap-aware for free. --- client.c | 33 +++++++++++---------------------- screen.c | 10 +++++----- 2 files changed, 16 insertions(+), 27 deletions(-) diff --git a/client.c b/client.c index 6fcfb88..459d024 100644 --- a/client.c +++ b/client.c @@ -286,10 +286,7 @@ client_maximize(struct client_ctx *cc) cc->geom.x + cc->geom.w / 2, cc->geom.y + cc->geom.h / 2); - cc->geom.x = xine.x + sc->gap.left; - cc->geom.y = xine.y + sc->gap.top; - cc->geom.h = xine.h - (sc->gap.top + sc->gap.bottom); - cc->geom.w = xine.w - (sc->gap.left + sc->gap.right); + cc->geom = xine; cc->bwidth = 0; cc->flags |= CLIENT_MAXIMIZED; @@ -329,9 +326,8 @@ client_vmaximize(struct client_ctx *cc) cc->geom.x + cc->geom.w / 2, cc->geom.y + cc->geom.h / 2); - cc->geom.y = xine.y + sc->gap.top; - cc->geom.h = xine.h - (cc->bwidth * 2) - (sc->gap.top + - sc->gap.bottom); + cc->geom.y = xine.y; + cc->geom.h = xine.h - (cc->bwidth * 2); cc->flags |= CLIENT_VMAXIMIZED; resize: @@ -370,9 +366,8 @@ client_hmaximize(struct client_ctx *cc) cc->geom.x + cc->geom.w / 2, cc->geom.y + cc->geom.h / 2); - cc->geom.x = xine.x + sc->gap.left; - cc->geom.w = xine.w - (cc->bwidth * 2) - (sc->gap.left + - sc->gap.right); + cc->geom.x = xine.x; + cc->geom.w = xine.w - (cc->bwidth * 2); cc->flags |= CLIENT_HMAXIMIZED; resize: @@ -672,22 +667,16 @@ client_placecalc(struct client_ctx *cc) yslack = xine.h - cc->geom.h - cc->bwidth * 2; if (xslack >= xine.x) { - cc->geom.x = MAX(MIN(xmouse, xslack), - xine.x + sc->gap.left); - if (cc->geom.x > (xslack - sc->gap.right)) - cc->geom.x -= sc->gap.right; + cc->geom.x = MAX(MIN(xmouse, xslack), xine.x); } else { - cc->geom.x = xine.x + sc->gap.left; - cc->geom.w = xine.w - sc->gap.left; + cc->geom.x = xine.x; + cc->geom.w = xine.w; } if (yslack >= xine.y) { - cc->geom.y = MAX(MIN(ymouse, yslack), - xine.y + sc->gap.top); - if (cc->geom.y > (yslack - sc->gap.bottom)) - cc->geom.y -= sc->gap.bottom; + cc->geom.y = MAX(MIN(ymouse, yslack), xine.y); } else { - cc->geom.y = xine.y + sc->gap.top; - cc->geom.h = xine.h - sc->gap.top; + cc->geom.y = xine.y; + cc->geom.h = xine.h; } } } diff --git a/screen.c b/screen.c index 01c2f73..6b7de6b 100644 --- a/screen.c +++ b/screen.c @@ -132,7 +132,7 @@ screen_find_xinerama(struct screen_ctx *sc, int x, int y) struct geom geom; int i; - geom = sc->view; + geom = sc->work; if (sc->xinerama == NULL) return (geom); @@ -141,10 +141,10 @@ screen_find_xinerama(struct screen_ctx *sc, int x, int y) info = &sc->xinerama[i]; if (x >= info->x_org && x < info->x_org + info->width && y >= info->y_org && y < info->y_org + info->height) { - geom.x = info->x_org; - geom.y = info->y_org; - geom.w = info->width; - geom.h = info->height; + geom.x = info->x_org + sc->gap.left; + geom.y = info->y_org + sc->gap.top; + geom.w = info->width - (sc->gap.left + sc->gap.right); + geom.h = info->height - (sc->gap.top + sc->gap.bottom); break; } } From 3a7596968b740328c41f6807800d004c75e3c62f Mon Sep 17 00:00:00 2001 From: okan Date: Tue, 8 Jan 2013 15:16:04 +0000 Subject: [PATCH 5/7] add per-group vert/horiz tiling support; introduces 2 new bind commands, 'vtile' and 'htile'; from Alexander Polakov. --- calmwm.h | 7 ++++ client.c | 118 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ conf.c | 4 ++ cwmrc.5 | 8 +++- kbfunc.c | 13 ++++++ 5 files changed, 149 insertions(+), 1 deletion(-) diff --git a/calmwm.h b/calmwm.h index 47f60e7..c02db59 100644 --- a/calmwm.h +++ b/calmwm.h @@ -76,6 +76,10 @@ #define ARG_CHAR 0x0001 #define ARG_INT 0x0002 + +#define CWM_TILE_HORIZ 0x0001 +#define CWM_TILE_VERT 0x0002 + union arg { char *c; int i; @@ -321,6 +325,7 @@ void client_freeze(struct client_ctx *); void client_getsizehints(struct client_ctx *); void client_hide(struct client_ctx *); void client_hmaximize(struct client_ctx *); +void client_htile(struct client_ctx *); void client_leave(struct client_ctx *); void client_lower(struct client_ctx *); void client_map(struct client_ctx *); @@ -338,6 +343,7 @@ int client_snapcalc(int, int, int, int, int); void client_transient(struct client_ctx *); void client_unhide(struct client_ctx *); void client_vmaximize(struct client_ctx *); +void client_vtile(struct client_ctx *); void client_warp(struct client_ctx *); void group_alltoggle(struct screen_ctx *); @@ -408,6 +414,7 @@ void kbfunc_quit_wm(struct client_ctx *, union arg *); void kbfunc_restart(struct client_ctx *, union arg *); void kbfunc_ssh(struct client_ctx *, union arg *); void kbfunc_term(struct client_ctx *, union arg *); +void kbfunc_tile(struct client_ctx *, union arg *); void mousefunc_menu_cmd(struct client_ctx *, void *); void mousefunc_menu_group(struct client_ctx *, void *); diff --git a/client.c b/client.c index 459d024..4895817 100644 --- a/client.c +++ b/client.c @@ -866,3 +866,121 @@ client_snapcalc(int n0, int n1, int e0, int e1, int snapdist) else return (0); } + +void +client_htile(struct client_ctx *cc) +{ + struct client_ctx *ci; + struct group_ctx *gc = cc->group; + struct screen_ctx *sc = cc->sc; + struct geom xine; + int i, n, mh, x, h, w; + + if (!gc) + return; + i = n = 0; + + TAILQ_FOREACH(ci, &gc->clients, group_entry) { + if (ci->flags & CLIENT_HIDDEN || + ci->flags & CLIENT_IGNORE || (ci == cc)) + continue; + n++; + } + if (n == 0) + return; + + xine = screen_find_xinerama(sc, + cc->geom.x + cc->geom.w / 2, + cc->geom.y + cc->geom.h / 2); + + if (cc->flags & CLIENT_VMAXIMIZED || + cc->geom.h + (cc->bwidth * 2) >= xine.h) + return; + + cc->flags &= ~CLIENT_HMAXIMIZED; + cc->geom.x = xine.x; + cc->geom.y = xine.y; + cc->geom.w = xine.w - (cc->bwidth * 2); + client_resize(cc, 1); + client_ptrwarp(cc); + + mh = cc->geom.h + (cc->bwidth * 2); + x = xine.x; + w = xine.w / n; + h = xine.h - mh; + TAILQ_FOREACH(ci, &gc->clients, group_entry) { + if (ci->flags & CLIENT_HIDDEN || + ci->flags & CLIENT_IGNORE || (ci == cc)) + continue; + ci->bwidth = Conf.bwidth; + ci->geom.y = xine.y + mh; + ci->geom.x = x; + ci->geom.h = h - (ci->bwidth * 2); + ci->geom.w = w - (ci->bwidth * 2); + if (i + 1 == n) + ci->geom.w = xine.x + xine.w - + ci->geom.x - (ci->bwidth * 2); + x += w; + client_resize(ci, 1); + i++; + } +} + +void +client_vtile(struct client_ctx *cc) +{ + struct client_ctx *ci; + struct group_ctx *gc = cc->group; + struct screen_ctx *sc = cc->sc; + struct geom xine; + int i, n, mw, y, h, w; + + if (!gc) + return; + i = n = 0; + + TAILQ_FOREACH(ci, &gc->clients, group_entry) { + if (ci->flags & CLIENT_HIDDEN || + ci->flags & CLIENT_IGNORE || (ci == cc)) + continue; + n++; + } + if (n == 0) + return; + + xine = screen_find_xinerama(sc, + cc->geom.x + cc->geom.w / 2, + cc->geom.y + cc->geom.h / 2); + + if (cc->flags & CLIENT_HMAXIMIZED || + cc->geom.w + (cc->bwidth * 2) >= xine.w) + return; + + cc->flags &= ~CLIENT_VMAXIMIZED; + cc->geom.x = xine.x; + cc->geom.y = xine.y; + cc->geom.h = xine.h - (cc->bwidth * 2); + client_resize(cc, 1); + client_ptrwarp(cc); + + mw = cc->geom.w + (cc->bwidth * 2); + y = xine.y; + h = xine.h / n; + w = xine.w - mw; + TAILQ_FOREACH(ci, &gc->clients, group_entry) { + if (ci->flags & CLIENT_HIDDEN || + ci->flags & CLIENT_IGNORE || (ci == cc)) + continue; + ci->bwidth = Conf.bwidth; + ci->geom.y = y; + ci->geom.x = xine.x + mw; + ci->geom.h = h - (ci->bwidth * 2); + ci->geom.w = w - (ci->bwidth * 2); + if (i + 1 == n) + ci->geom.h = xine.y + xine.h - + ci->geom.y - (ci->bwidth * 2); + y += h; + client_resize(ci, 1); + i++; + } +} diff --git a/conf.c b/conf.c index 6ef9375..67f58a6 100644 --- a/conf.c +++ b/conf.c @@ -375,6 +375,10 @@ static struct { {.i = (CWM_LEFT|CWM_PTRMOVE|CWM_BIGMOVE)} }, { "bigptrmoveright", kbfunc_moveresize, 0, {.i = (CWM_RIGHT|CWM_PTRMOVE|CWM_BIGMOVE)} }, + { "htile", kbfunc_tile, KBFLAG_NEEDCLIENT, + {.i = CWM_TILE_HORIZ } }, + { "vtile", kbfunc_tile, KBFLAG_NEEDCLIENT, + {.i = CWM_TILE_VERT } }, }; /* diff --git a/cwmrc.5 b/cwmrc.5 index c8ea841..b475d5e 100644 --- a/cwmrc.5 +++ b/cwmrc.5 @@ -14,7 +14,7 @@ .\" ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF .\" OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. .\" -.Dd $Mdocdate: November 29 2012 $ +.Dd $Mdocdate: December 17 2012 $ .Dt CWMRC 5 .Os .Sh NAME @@ -438,6 +438,12 @@ pixels right. Move pointer 10 times .Ar moveamount pixels left. +.It htile +Current window is placed at the top of the screen and maximized +horizontally, other windows in its group share remaining screen space. +.It vtile +Current window is placed on the left of the screen and maximized +vertically, other windows in its group share remaining screen space. .El .Sh MOUSEBIND COMMAND LIST .Bl -tag -width 18n -compact diff --git a/kbfunc.c b/kbfunc.c index d61193f..c742f7e 100644 --- a/kbfunc.c +++ b/kbfunc.c @@ -479,3 +479,16 @@ kbfunc_restart(struct client_ctx *cc, union arg *arg) (void)setsid(); (void)execvp(cwm_argv[0], cwm_argv); } + +void +kbfunc_tile(struct client_ctx *cc, union arg *arg) +{ + switch (arg->i) { + case CWM_TILE_HORIZ: + client_htile(cc); + break; + case CWM_TILE_VERT: + client_vtile(cc); + break; + } +} From 62acbee4b3f0bb69db00c070d8b92ed0db2caf5d Mon Sep 17 00:00:00 2001 From: okan Date: Thu, 10 Jan 2013 15:28:11 +0000 Subject: [PATCH 6/7] set the initial group to '1', missed by recent off-by-one group numbering re-work; discovered the hard way by sthen@. ok sthen@ --- group.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/group.c b/group.c index d226578..b8bf59a 100644 --- a/group.c +++ b/group.c @@ -160,7 +160,7 @@ group_init(struct screen_ctx *sc) xu_ewmh_net_showing_desktop(sc); xu_ewmh_net_virtual_roots(sc); - group_setactive(sc, 0); + group_setactive(sc, 1); } void From 47aa485fa259965609d3e13cce7a03ac64b14e6f Mon Sep 17 00:00:00 2001 From: okan Date: Sun, 13 Jan 2013 13:55:12 +0000 Subject: [PATCH 7/7] put back r1.68 which allows an empty group to be sticky; behavior change noticed by Thomas Pfaff and diagnosis why we need to group_setactive in this case by Alexander Polakov. replace XXX with a useful comment. --- group.c | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/group.c b/group.c index b8bf59a..6644dce 100644 --- a/group.c +++ b/group.c @@ -269,8 +269,12 @@ group_hidetoggle(struct screen_ctx *sc, int idx) if (gc->hidden) group_show(sc, gc); - else + else { group_hide(sc, gc); + /* make clients stick to empty group */ + if (TAILQ_EMPTY(&gc->clients)) + group_setactive(sc, idx); + } } void