diff --git a/patches/dwm-alwayscenter-20200625-f04cac6.diff b/patches/dwm-alwayscenter-20200625-f04cac6.diff deleted file mode 100644 index 03ea9ef..0000000 --- a/patches/dwm-alwayscenter-20200625-f04cac6.diff +++ /dev/null @@ -1,12 +0,0 @@ -diff -up dwm/dwm.c dwmmod/dwm.c ---- dwm/dwm.c 2020-06-25 00:21:30.383692180 -0300 -+++ dwmmod/dwm.c 2020-06-25 00:20:35.643692330 -0300 -@@ -1057,6 +1057,8 @@ manage(Window w, XWindowAttributes *wa) - updatewindowtype(c); - updatesizehints(c); - updatewmhints(c); -+ c->x = c->mon->mx + (c->mon->mw - WIDTH(c)) / 2; -+ c->y = c->mon->my + (c->mon->mh - HEIGHT(c)) / 2; - XSelectInput(dpy, w, EnterWindowMask|FocusChangeMask|PropertyChangeMask|StructureNotifyMask); - grabbuttons(c, 0); - if (!c->isfloating) diff --git a/patches/dwm-attachdirection-6.2.diff b/patches/dwm-attachdirection-6.2.diff deleted file mode 100644 index aab43b5..0000000 --- a/patches/dwm-attachdirection-6.2.diff +++ /dev/null @@ -1,232 +0,0 @@ -From faa0d3a06229906b8cd30cdeb4360ee9608f74bd Mon Sep 17 00:00:00 2001 -From: MLquest8 -Date: Thu, 18 Jun 2020 15:14:56 +0400 -Subject: [PATCH] attachdirection updated. Added attachtop option(5) that - attaches the newest client below the last master/on top of the stack. - ---- - config.def.h | 1 + - dwm.c | 146 +++++++++++++++++++++++++++++++++++++++++++++++++-- - 2 files changed, 143 insertions(+), 4 deletions(-) - -diff --git a/config.def.h b/config.def.h -index 1c0b587..d7c089b 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -35,6 +35,7 @@ static const Rule rules[] = { - static const float mfact = 0.55; /* factor of master area size [0.05..0.95] */ - static const int nmaster = 1; /* number of clients in master area */ - static const int resizehints = 1; /* 1 means respect size hints in tiled resizals */ -+static const int attachdirection = 0; /* 0 default, 1 above, 2 aside, 3 below, 4 bottom, 5 top */ - - static const Layout layouts[] = { - /* symbol arrange function */ -diff --git a/dwm.c b/dwm.c -index 9fd0286..2c1b143 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -49,7 +49,8 @@ - #define CLEANMASK(mask) (mask & ~(numlockmask|LockMask) & (ShiftMask|ControlMask|Mod1Mask|Mod2Mask|Mod3Mask|Mod4Mask|Mod5Mask)) - #define INTERSECT(x,y,w,h,m) (MAX(0, MIN((x)+(w),(m)->wx+(m)->ww) - MAX((x),(m)->wx)) \ - * MAX(0, MIN((y)+(h),(m)->wy+(m)->wh) - MAX((y),(m)->wy))) --#define ISVISIBLE(C) ((C->tags & C->mon->tagset[C->mon->seltags])) -+#define ISVISIBLEONTAG(C, T) ((C->tags & T)) -+#define ISVISIBLE(C) ISVISIBLEONTAG(C, C->mon->tagset[C->mon->seltags]) - #define LENGTH(X) (sizeof X / sizeof X[0]) - #define MOUSEMASK (BUTTONMASK|PointerMotionMask) - #define WIDTH(X) ((X)->w + 2 * (X)->bw) -@@ -147,6 +148,11 @@ static int applysizehints(Client *c, int *x, int *y, int *w, int *h, int interac - static void arrange(Monitor *m); - static void arrangemon(Monitor *m); - static void attach(Client *c); -+static void attachabove(Client *c); -+static void attachaside(Client *c); -+static void attachbelow(Client *c); -+static void attachbottom(Client *c); -+static void attachtop(Client *c); - static void attachstack(Client *c); - static void buttonpress(XEvent *e); - static void checkotherwm(void); -@@ -184,6 +190,7 @@ static void maprequest(XEvent *e); - static void monocle(Monitor *m); - static void motionnotify(XEvent *e); - static void movemouse(const Arg *arg); -+static Client *nexttagged(Client *c); - static Client *nexttiled(Client *c); - static void pop(Client *); - static void propertynotify(XEvent *e); -@@ -407,6 +414,73 @@ attach(Client *c) - c->mon->clients = c; - } - -+void -+attachabove(Client *c) -+{ -+ if (c->mon->sel == NULL || c->mon->sel == c->mon->clients || c->mon->sel->isfloating) { -+ attach(c); -+ return; -+ } -+ -+ Client *at; -+ for (at = c->mon->clients; at->next != c->mon->sel; at = at->next); -+ c->next = at->next; -+ at->next = c; -+} -+ -+void -+attachaside(Client *c) { -+ Client *at = nexttagged(c); -+ if(!at) { -+ attach(c); -+ return; -+ } -+ c->next = at->next; -+ at->next = c; -+} -+ -+void -+attachbelow(Client *c) -+{ -+ if(c->mon->sel == NULL || c->mon->sel == c || c->mon->sel->isfloating) { -+ attach(c); -+ return; -+ } -+ c->next = c->mon->sel->next; -+ c->mon->sel->next = c; -+} -+ -+void -+attachbottom(Client *c) -+{ -+ Client *below = c->mon->clients; -+ for (; below && below->next; below = below->next); -+ c->next = NULL; -+ if (below) -+ below->next = c; -+ else -+ c->mon->clients = c; -+} -+ -+void -+attachtop(Client *c) -+{ -+ int n; -+ Monitor *m = selmon; -+ Client *below; -+ -+ for (n = 1, below = c->mon->clients; -+ below && below->next && (below->isfloating || !ISVISIBLEONTAG(below, c->tags) || n != m->nmaster); -+ n = below->isfloating || !ISVISIBLEONTAG(below, c->tags) ? n + 0 : n + 1, below = below->next); -+ c->next = NULL; -+ if (below) { -+ c->next = below->next; -+ below->next = c; -+ } -+ else -+ c->mon->clients = c; -+} -+ - void - attachstack(Client *c) - { -@@ -1063,7 +1137,25 @@ manage(Window w, XWindowAttributes *wa) - c->isfloating = c->oldstate = trans != None || c->isfixed; - if (c->isfloating) - XRaiseWindow(dpy, c->win); -- attach(c); -+ switch(attachdirection){ -+ case 1: -+ attachabove(c); -+ break; -+ case 2: -+ attachaside(c); -+ break; -+ case 3: -+ attachbelow(c); -+ break; -+ case 4: -+ attachbottom(c); -+ break; -+ case 5: -+ attachtop(c); -+ break; -+ default: -+ attach(c); -+ } - attachstack(c); - XChangeProperty(dpy, root, netatom[NetClientList], XA_WINDOW, 32, PropModeAppend, - (unsigned char *) &(c->win), 1); -@@ -1193,6 +1285,16 @@ movemouse(const Arg *arg) - } - } - -+Client * -+nexttagged(Client *c) { -+ Client *walked = c->mon->clients; -+ for(; -+ walked && (walked->isfloating || !ISVISIBLEONTAG(walked, c->tags)); -+ walked = walked->next -+ ); -+ return walked; -+} -+ - Client * - nexttiled(Client *c) - { -@@ -1418,7 +1520,25 @@ sendmon(Client *c, Monitor *m) - detachstack(c); - c->mon = m; - c->tags = m->tagset[m->seltags]; /* assign tags of target monitor */ -- attach(c); -+ switch(attachdirection){ -+ case 1: -+ attachabove(c); -+ break; -+ case 2: -+ attachaside(c); -+ break; -+ case 3: -+ attachbelow(c); -+ break; -+ case 4: -+ attachbottom(c); -+ break; -+ case 5: -+ attachtop(c); -+ break; -+ default: -+ attach(c); -+ } - attachstack(c); - focus(NULL); - arrange(NULL); -@@ -1900,7 +2020,25 @@ updategeom(void) - m->clients = c->next; - detachstack(c); - c->mon = mons; -- attach(c); -+ switch(attachdirection){ -+ case 1: -+ attachabove(c); -+ break; -+ case 2: -+ attachaside(c); -+ break; -+ case 3: -+ attachbelow(c); -+ break; -+ case 4: -+ attachbottom(c); -+ break; -+ case 5: -+ attachtop(c); -+ break; -+ default: -+ attach(c); -+ } - attachstack(c); - } - if (m == selmon) --- -2.26.2 - diff --git a/patches/dwm-cfacts-vanitygaps-6.2_combo.diff b/patches/dwm-cfacts-vanitygaps-6.2_combo.diff deleted file mode 100644 index ce694f9..0000000 --- a/patches/dwm-cfacts-vanitygaps-6.2_combo.diff +++ /dev/null @@ -1,1110 +0,0 @@ -From 1f9992ae1a745a86294a555051ea17ba4ef5ce5f Mon Sep 17 00:00:00 2001 -From: bakkeby -Date: Mon, 6 Apr 2020 12:04:55 +0200 -Subject: [PATCH 1/2] Adding cfacts patch which provides the ability to assign - different weights to clients in their respective stack in tiled layout. - -Refer to https://dwm.suckless.org/patches/cfacts/ ---- - config.def.h | 3 +++ - dwm.c | 35 ++++++++++++++++++++++++++++++++--- - 2 files changed, 35 insertions(+), 3 deletions(-) - -diff --git a/config.def.h b/config.def.h -index 1c0b587..83910c1 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -70,6 +70,9 @@ static Key keys[] = { - { MODKEY, XK_d, incnmaster, {.i = -1 } }, - { MODKEY, XK_h, setmfact, {.f = -0.05} }, - { MODKEY, XK_l, setmfact, {.f = +0.05} }, -+ { MODKEY|ShiftMask, XK_h, setcfact, {.f = +0.25} }, -+ { MODKEY|ShiftMask, XK_l, setcfact, {.f = -0.25} }, -+ { MODKEY|ShiftMask, XK_o, setcfact, {.f = 0.00} }, - { MODKEY, XK_Return, zoom, {0} }, - { MODKEY, XK_Tab, view, {0} }, - { MODKEY|ShiftMask, XK_c, killclient, {0} }, -diff --git a/dwm.c b/dwm.c -index 4465af1..5592c57 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -87,6 +87,7 @@ typedef struct Client Client; - struct Client { - char name[256]; - float mina, maxa; -+ float cfact; - int x, y, w, h; - int oldx, oldy, oldw, oldh; - int basew, baseh, incw, inch, maxw, maxh, minw, minh; -@@ -200,6 +201,7 @@ static void setclientstate(Client *c, long state); - static void setfocus(Client *c); - static void setfullscreen(Client *c, int fullscreen); - static void setlayout(const Arg *arg); -+static void setcfact(const Arg *arg); - static void setmfact(const Arg *arg); - static void setup(void); - static void seturgent(Client *c, int urg); -@@ -1029,6 +1031,7 @@ manage(Window w, XWindowAttributes *wa) - c->w = c->oldw = wa->width; - c->h = c->oldh = wa->height; - c->oldbw = wa->border_width; -+ c->cfact = 1.0; - - updatetitle(c); - if (XGetTransientForHint(dpy, w, &trans) && (t = wintoclient(trans))) { -@@ -1511,6 +1514,24 @@ setlayout(const Arg *arg) - drawbar(selmon); - } - -+void -+setcfact(const Arg *arg) { -+ float f; -+ Client *c; -+ -+ c = selmon->sel; -+ -+ if(!arg || !c || !selmon->lt[selmon->sellt]->arrange) -+ return; -+ f = arg->f + c->cfact; -+ if(arg->f == 0.0) -+ f = 1.0; -+ else if(f < 0.25 || f > 4.0) -+ return; -+ c->cfact = f; -+ arrange(selmon); -+} -+ - /* arg > 1.0 will set mfact absolutely */ - void - setmfact(const Arg *arg) -@@ -1674,9 +1695,15 @@ void - tile(Monitor *m) - { - unsigned int i, n, h, mw, my, ty; -+ float mfacts = 0, sfacts = 0; - Client *c; - -- for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++); -+ for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++) { -+ if (n < m->nmaster) -+ mfacts += c->cfact; -+ else -+ sfacts += c->cfact; -+ } - if (n == 0) - return; - -@@ -1686,13 +1713,15 @@ tile(Monitor *m) - mw = m->ww; - for (i = my = ty = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) - if (i < m->nmaster) { -- h = (m->wh - my) / (MIN(n, m->nmaster) - i); -+ h = (m->wh - my) * (c->cfact / mfacts); - resize(c, m->wx, m->wy + my, mw - (2*c->bw), h - (2*c->bw), 0); - my += HEIGHT(c); -+ mfacts -= c->cfact; - } else { -- h = (m->wh - ty) / (n - i); -+ h = (m->wh - ty) * (c->cfact / sfacts); - resize(c, m->wx + mw, m->wy + ty, m->ww - mw - (2*c->bw), h - (2*c->bw), 0); - ty += HEIGHT(c); -+ sfacts -= c->cfact; - } - } - --- -2.19.1 - - -From 051e4de72079bb0b8e50d2faa61b9a0ef36434b5 Mon Sep 17 00:00:00 2001 -From: bakkeby -Date: Fri, 8 May 2020 16:51:00 +0200 -Subject: [PATCH 2/2] vanitygaps - adds gaps to layouts - -This patch differentiates between inner and outer gaps as well as -horizontal and vertical gaps. - -The logic of these layouts also aims to be pixel perfect by ensuring -an even split of the available space and re-distributing the remainder -among the available clients. ---- - config.def.h | 38 ++- - dwm.c | 43 +-- - vanitygaps.c | 822 +++++++++++++++++++++++++++++++++++++++++++++++++++ - 3 files changed, 867 insertions(+), 36 deletions(-) - create mode 100644 vanitygaps.c - -diff --git a/config.def.h b/config.def.h -index 83910c1..91a9cfc 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -3,6 +3,11 @@ - /* appearance */ - static const unsigned int borderpx = 1; /* border pixel of windows */ - static const unsigned int snap = 32; /* snap pixel */ -+static const unsigned int gappih = 20; /* horiz inner gap between windows */ -+static const unsigned int gappiv = 10; /* vert inner gap between windows */ -+static const unsigned int gappoh = 10; /* horiz outer gap between windows and screen edge */ -+static const unsigned int gappov = 30; /* vert outer gap between windows and screen edge */ -+static int smartgaps = 0; /* 1 means no outer gap when there is only one window */ - static const int showbar = 1; /* 0 means no bar */ - static const int topbar = 1; /* 0 means bottom bar */ - static const char *fonts[] = { "monospace:size=10" }; -@@ -36,11 +41,26 @@ static const float mfact = 0.55; /* factor of master area size [0.05..0.95] - static const int nmaster = 1; /* number of clients in master area */ - static const int resizehints = 1; /* 1 means respect size hints in tiled resizals */ - -+#define FORCE_VSPLIT 1 /* nrowgrid layout: force two clients to always split vertically */ -+#include "vanitygaps.c" -+ - static const Layout layouts[] = { - /* symbol arrange function */ - { "[]=", tile }, /* first entry is default */ -- { "><>", NULL }, /* no layout function means floating behavior */ - { "[M]", monocle }, -+ { "[@]", spiral }, -+ { "[\\]", dwindle }, -+ { "H[]", deck }, -+ { "TTT", bstack }, -+ { "===", bstackhoriz }, -+ { "HHH", grid }, -+ { "###", nrowgrid }, -+ { "---", horizgrid }, -+ { ":::", gaplessgrid }, -+ { "|M|", centeredmaster }, -+ { ">M>", centeredfloatingmaster }, -+ { "><>", NULL }, /* no layout function means floating behavior */ -+ { NULL, NULL }, - }; - - /* key definitions */ -@@ -74,6 +94,22 @@ static Key keys[] = { - { MODKEY|ShiftMask, XK_l, setcfact, {.f = -0.25} }, - { MODKEY|ShiftMask, XK_o, setcfact, {.f = 0.00} }, - { MODKEY, XK_Return, zoom, {0} }, -+ { MODKEY|Mod4Mask, XK_u, incrgaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_u, incrgaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_i, incrigaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_i, incrigaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_o, incrogaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_o, incrogaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_6, incrihgaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_6, incrihgaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_7, incrivgaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_7, incrivgaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_8, incrohgaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_8, incrohgaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_9, incrovgaps, {.i = +1 } }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_9, incrovgaps, {.i = -1 } }, -+ { MODKEY|Mod4Mask, XK_0, togglegaps, {0} }, -+ { MODKEY|Mod4Mask|ShiftMask, XK_0, defaultgaps, {0} }, - { MODKEY, XK_Tab, view, {0} }, - { MODKEY|ShiftMask, XK_c, killclient, {0} }, - { MODKEY, XK_t, setlayout, {.v = &layouts[0]} }, -diff --git a/dwm.c b/dwm.c -index 5592c57..7d503cb 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -120,6 +120,10 @@ struct Monitor { - int by; /* bar geometry */ - int mx, my, mw, mh; /* screen size */ - int wx, wy, ww, wh; /* window area */ -+ int gappih; /* horizontal gap between windows */ -+ int gappiv; /* vertical gap between windows */ -+ int gappoh; /* horizontal outer gaps */ -+ int gappov; /* vertical outer gaps */ - unsigned int seltags; - unsigned int sellt; - unsigned int tagset[2]; -@@ -210,7 +214,6 @@ static void sigchld(int unused); - static void spawn(const Arg *arg); - static void tag(const Arg *arg); - static void tagmon(const Arg *arg); --static void tile(Monitor *); - static void togglebar(const Arg *arg); - static void togglefloating(const Arg *arg); - static void toggletag(const Arg *arg); -@@ -640,6 +643,10 @@ createmon(void) - m->nmaster = nmaster; - m->showbar = showbar; - m->topbar = topbar; -+ m->gappih = gappih; -+ m->gappiv = gappiv; -+ m->gappoh = gappoh; -+ m->gappov = gappov; - m->lt[0] = &layouts[0]; - m->lt[1] = &layouts[1 % LENGTH(layouts)]; - strncpy(m->ltsymbol, layouts[0].symbol, sizeof m->ltsymbol); -@@ -1691,40 +1698,6 @@ tagmon(const Arg *arg) - sendmon(selmon->sel, dirtomon(arg->i)); - } - --void --tile(Monitor *m) --{ -- unsigned int i, n, h, mw, my, ty; -- float mfacts = 0, sfacts = 0; -- Client *c; -- -- for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++) { -- if (n < m->nmaster) -- mfacts += c->cfact; -- else -- sfacts += c->cfact; -- } -- if (n == 0) -- return; -- -- if (n > m->nmaster) -- mw = m->nmaster ? m->ww * m->mfact : 0; -- else -- mw = m->ww; -- for (i = my = ty = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -- if (i < m->nmaster) { -- h = (m->wh - my) * (c->cfact / mfacts); -- resize(c, m->wx, m->wy + my, mw - (2*c->bw), h - (2*c->bw), 0); -- my += HEIGHT(c); -- mfacts -= c->cfact; -- } else { -- h = (m->wh - ty) * (c->cfact / sfacts); -- resize(c, m->wx + mw, m->wy + ty, m->ww - mw - (2*c->bw), h - (2*c->bw), 0); -- ty += HEIGHT(c); -- sfacts -= c->cfact; -- } --} -- - void - togglebar(const Arg *arg) - { -diff --git a/vanitygaps.c b/vanitygaps.c -new file mode 100644 -index 0000000..1a816b6 ---- /dev/null -+++ b/vanitygaps.c -@@ -0,0 +1,822 @@ -+/* Key binding functions */ -+static void defaultgaps(const Arg *arg); -+static void incrgaps(const Arg *arg); -+static void incrigaps(const Arg *arg); -+static void incrogaps(const Arg *arg); -+static void incrohgaps(const Arg *arg); -+static void incrovgaps(const Arg *arg); -+static void incrihgaps(const Arg *arg); -+static void incrivgaps(const Arg *arg); -+static void togglegaps(const Arg *arg); -+/* Layouts (delete the ones you do not need) */ -+static void bstack(Monitor *m); -+static void bstackhoriz(Monitor *m); -+static void centeredmaster(Monitor *m); -+static void centeredfloatingmaster(Monitor *m); -+static void deck(Monitor *m); -+static void dwindle(Monitor *m); -+static void fibonacci(Monitor *m, int s); -+static void grid(Monitor *m); -+static void nrowgrid(Monitor *m); -+static void spiral(Monitor *m); -+static void tile(Monitor *m); -+/* Internals */ -+static void getgaps(Monitor *m, int *oh, int *ov, int *ih, int *iv, unsigned int *nc); -+static void getfacts(Monitor *m, int msize, int ssize, float *mf, float *sf, int *mr, int *sr); -+static void setgaps(int oh, int ov, int ih, int iv); -+ -+/* Settings */ -+#if !PERTAG_PATCH -+static int enablegaps = 1; -+#endif // PERTAG_PATCH -+ -+void -+setgaps(int oh, int ov, int ih, int iv) -+{ -+ if (oh < 0) oh = 0; -+ if (ov < 0) ov = 0; -+ if (ih < 0) ih = 0; -+ if (iv < 0) iv = 0; -+ -+ selmon->gappoh = oh; -+ selmon->gappov = ov; -+ selmon->gappih = ih; -+ selmon->gappiv = iv; -+ arrange(selmon); -+} -+ -+void -+togglegaps(const Arg *arg) -+{ -+ #if PERTAG_PATCH -+ selmon->pertag->enablegaps[selmon->pertag->curtag] = !selmon->pertag->enablegaps[selmon->pertag->curtag]; -+ #else -+ enablegaps = !enablegaps; -+ #endif // PERTAG_PATCH -+ arrange(NULL); -+} -+ -+void -+defaultgaps(const Arg *arg) -+{ -+ setgaps(gappoh, gappov, gappih, gappiv); -+} -+ -+void -+incrgaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh + arg->i, -+ selmon->gappov + arg->i, -+ selmon->gappih + arg->i, -+ selmon->gappiv + arg->i -+ ); -+} -+ -+void -+incrigaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh, -+ selmon->gappov, -+ selmon->gappih + arg->i, -+ selmon->gappiv + arg->i -+ ); -+} -+ -+void -+incrogaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh + arg->i, -+ selmon->gappov + arg->i, -+ selmon->gappih, -+ selmon->gappiv -+ ); -+} -+ -+void -+incrohgaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh + arg->i, -+ selmon->gappov, -+ selmon->gappih, -+ selmon->gappiv -+ ); -+} -+ -+void -+incrovgaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh, -+ selmon->gappov + arg->i, -+ selmon->gappih, -+ selmon->gappiv -+ ); -+} -+ -+void -+incrihgaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh, -+ selmon->gappov, -+ selmon->gappih + arg->i, -+ selmon->gappiv -+ ); -+} -+ -+void -+incrivgaps(const Arg *arg) -+{ -+ setgaps( -+ selmon->gappoh, -+ selmon->gappov, -+ selmon->gappih, -+ selmon->gappiv + arg->i -+ ); -+} -+ -+void -+getgaps(Monitor *m, int *oh, int *ov, int *ih, int *iv, unsigned int *nc) -+{ -+ unsigned int n, oe, ie; -+ #if PERTAG_PATCH -+ oe = ie = selmon->pertag->enablegaps[selmon->pertag->curtag]; -+ #else -+ oe = ie = enablegaps; -+ #endif // PERTAG_PATCH -+ Client *c; -+ -+ for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++); -+ if (smartgaps && n == 1) { -+ oe = 0; // outer gaps disabled when only one client -+ } -+ -+ *oh = m->gappoh*oe; // outer horizontal gap -+ *ov = m->gappov*oe; // outer vertical gap -+ *ih = m->gappih*ie; // inner horizontal gap -+ *iv = m->gappiv*ie; // inner vertical gap -+ *nc = n; // number of clients -+} -+ -+void -+getfacts(Monitor *m, int msize, int ssize, float *mf, float *sf, int *mr, int *sr) -+{ -+ unsigned int n; -+ float mfacts = 0, sfacts = 0; -+ int mtotal = 0, stotal = 0; -+ Client *c; -+ -+ for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++) -+ if (n < m->nmaster) -+ mfacts += c->cfact; -+ else -+ sfacts += c->cfact; -+ -+ for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++) -+ if (n < m->nmaster) -+ mtotal += msize * (c->cfact / mfacts); -+ else -+ stotal += ssize * (c->cfact / sfacts); -+ -+ *mf = mfacts; // total factor of master area -+ *sf = sfacts; // total factor of stack area -+ *mr = msize - mtotal; // the remainder (rest) of pixels after a cfacts master split -+ *sr = ssize - stotal; // the remainder (rest) of pixels after a cfacts stack split -+} -+ -+/*** -+ * Layouts -+ */ -+ -+/* -+ * Bottomstack layout + gaps -+ * https://dwm.suckless.org/patches/bottomstack/ -+ */ -+static void -+bstack(Monitor *m) -+{ -+ unsigned int i, n; -+ int oh, ov, ih, iv; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int sx = 0, sy = 0, sh = 0, sw = 0; -+ float mfacts, sfacts; -+ int mrest, srest; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ sx = mx = m->wx + ov; -+ sy = my = m->wy + oh; -+ sh = mh = m->wh - 2*oh; -+ mw = m->ww - 2*ov - iv * (MIN(n, m->nmaster) - 1); -+ sw = m->ww - 2*ov - iv * (n - m->nmaster - 1); -+ -+ if (m->nmaster && n > m->nmaster) { -+ sh = (mh - ih) * (1 - m->mfact); -+ mh = mh - ih - sh; -+ sx = mx; -+ sy = my + mh + ih; -+ } -+ -+ getfacts(m, mw, sw, &mfacts, &sfacts, &mrest, &srest); -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) { -+ if (i < m->nmaster) { -+ resize(c, mx, my, mw * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), mh - (2*c->bw), 0); -+ mx += WIDTH(c) + iv; -+ } else { -+ resize(c, sx, sy, sw * (c->cfact / sfacts) + ((i - m->nmaster) < srest ? 1 : 0) - (2*c->bw), sh - (2*c->bw), 0); -+ sx += WIDTH(c) + iv; -+ } -+ } -+} -+ -+static void -+bstackhoriz(Monitor *m) -+{ -+ unsigned int i, n; -+ int oh, ov, ih, iv; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int sx = 0, sy = 0, sh = 0, sw = 0; -+ float mfacts, sfacts; -+ int mrest, srest; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ sx = mx = m->wx + ov; -+ sy = my = m->wy + oh; -+ mh = m->wh - 2*oh; -+ sh = m->wh - 2*oh - ih * (n - m->nmaster - 1); -+ mw = m->ww - 2*ov - iv * (MIN(n, m->nmaster) - 1); -+ sw = m->ww - 2*ov; -+ -+ if (m->nmaster && n > m->nmaster) { -+ sh = (mh - ih) * (1 - m->mfact); -+ mh = mh - ih - sh; -+ sy = my + mh + ih; -+ sh = m->wh - mh - 2*oh - ih * (n - m->nmaster); -+ } -+ -+ getfacts(m, mw, sh, &mfacts, &sfacts, &mrest, &srest); -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) { -+ if (i < m->nmaster) { -+ resize(c, mx, my, mw * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), mh - (2*c->bw), 0); -+ mx += WIDTH(c) + iv; -+ } else { -+ resize(c, sx, sy, sw - (2*c->bw), sh * (c->cfact / sfacts) + ((i - m->nmaster) < srest ? 1 : 0) - (2*c->bw), 0); -+ sy += HEIGHT(c) + ih; -+ } -+ } -+} -+ -+/* -+ * Centred master layout + gaps -+ * https://dwm.suckless.org/patches/centeredmaster/ -+ */ -+void -+centeredmaster(Monitor *m) -+{ -+ unsigned int i, n; -+ int oh, ov, ih, iv; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int lx = 0, ly = 0, lw = 0, lh = 0; -+ int rx = 0, ry = 0, rw = 0, rh = 0; -+ float mfacts = 0, lfacts = 0, rfacts = 0; -+ int mtotal = 0, ltotal = 0, rtotal = 0; -+ int mrest = 0, lrest = 0, rrest = 0; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ /* initialize areas */ -+ mx = m->wx + ov; -+ my = m->wy + oh; -+ mh = m->wh - 2*oh - ih * ((!m->nmaster ? n : MIN(n, m->nmaster)) - 1); -+ mw = m->ww - 2*ov; -+ lh = m->wh - 2*oh - ih * (((n - m->nmaster) / 2) - 1); -+ rh = m->wh - 2*oh - ih * (((n - m->nmaster) / 2) - ((n - m->nmaster) % 2 ? 0 : 1)); -+ -+ if (m->nmaster && n > m->nmaster) { -+ /* go mfact box in the center if more than nmaster clients */ -+ if (n - m->nmaster > 1) { -+ /* ||<-S->|<---M--->|<-S->|| */ -+ mw = (m->ww - 2*ov - 2*iv) * m->mfact; -+ lw = (m->ww - mw - 2*ov - 2*iv) / 2; -+ rw = (m->ww - mw - 2*ov - 2*iv) - lw; -+ mx += lw + iv; -+ } else { -+ /* ||<---M--->|<-S->|| */ -+ mw = (mw - iv) * m->mfact; -+ lw = 0; -+ rw = m->ww - mw - iv - 2*ov; -+ } -+ lx = m->wx + ov; -+ ly = m->wy + oh; -+ rx = mx + mw + iv; -+ ry = m->wy + oh; -+ } -+ -+ /* calculate facts */ -+ for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++) { -+ if (!m->nmaster || n < m->nmaster) -+ mfacts += c->cfact; -+ else if ((n - m->nmaster) % 2) -+ lfacts += c->cfact; // total factor of left hand stack area -+ else -+ rfacts += c->cfact; // total factor of right hand stack area -+ } -+ -+ for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++) -+ if (!m->nmaster || n < m->nmaster) -+ mtotal += mh * (c->cfact / mfacts); -+ else if ((n - m->nmaster) % 2) -+ ltotal += lh * (c->cfact / lfacts); -+ else -+ rtotal += rh * (c->cfact / rfacts); -+ -+ mrest = mh - mtotal; -+ lrest = lh - ltotal; -+ rrest = rh - rtotal; -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) { -+ if (!m->nmaster || i < m->nmaster) { -+ /* nmaster clients are stacked vertically, in the center of the screen */ -+ resize(c, mx, my, mw - (2*c->bw), mh * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), 0); -+ my += HEIGHT(c) + ih; -+ } else { -+ /* stack clients are stacked vertically */ -+ if ((i - m->nmaster) % 2 ) { -+ resize(c, lx, ly, lw - (2*c->bw), lh * (c->cfact / lfacts) + ((i - 2*m->nmaster) < 2*lrest ? 1 : 0) - (2*c->bw), 0); -+ ly += HEIGHT(c) + ih; -+ } else { -+ resize(c, rx, ry, rw - (2*c->bw), rh * (c->cfact / rfacts) + ((i - 2*m->nmaster) < 2*rrest ? 1 : 0) - (2*c->bw), 0); -+ ry += HEIGHT(c) + ih; -+ } -+ } -+ } -+} -+ -+void -+centeredfloatingmaster(Monitor *m) -+{ -+ unsigned int i, n; -+ float mfacts, sfacts; -+ float mivf = 1.0; // master inner vertical gap factor -+ int oh, ov, ih, iv, mrest, srest; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int sx = 0, sy = 0, sh = 0, sw = 0; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ sx = mx = m->wx + ov; -+ sy = my = m->wy + oh; -+ sh = mh = m->wh - 2*oh; -+ mw = m->ww - 2*ov - iv*(n - 1); -+ sw = m->ww - 2*ov - iv*(n - m->nmaster - 1); -+ -+ if (m->nmaster && n > m->nmaster) { -+ mivf = 0.8; -+ /* go mfact box in the center if more than nmaster clients */ -+ if (m->ww > m->wh) { -+ mw = m->ww * m->mfact - iv*mivf*(MIN(n, m->nmaster) - 1); -+ mh = m->wh * 0.9; -+ } else { -+ mw = m->ww * 0.9 - iv*mivf*(MIN(n, m->nmaster) - 1); -+ mh = m->wh * m->mfact; -+ } -+ mx = m->wx + (m->ww - mw) / 2; -+ my = m->wy + (m->wh - mh - 2*oh) / 2; -+ -+ sx = m->wx + ov; -+ sy = m->wy + oh; -+ sh = m->wh - 2*oh; -+ } -+ -+ getfacts(m, mw, sw, &mfacts, &sfacts, &mrest, &srest); -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -+ if (i < m->nmaster) { -+ /* nmaster clients are stacked horizontally, in the center of the screen */ -+ resize(c, mx, my, mw * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), mh - (2*c->bw), 0); -+ mx += WIDTH(c) + iv*mivf; -+ } else { -+ /* stack clients are stacked horizontally */ -+ resize(c, sx, sy, sw * (c->cfact / sfacts) + ((i - m->nmaster) < srest ? 1 : 0) - (2*c->bw), sh - (2*c->bw), 0); -+ sx += WIDTH(c) + iv; -+ } -+} -+ -+/* -+ * Deck layout + gaps -+ * https://dwm.suckless.org/patches/deck/ -+ */ -+void -+deck(Monitor *m) -+{ -+ unsigned int i, n; -+ int oh, ov, ih, iv; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int sx = 0, sy = 0, sh = 0, sw = 0; -+ float mfacts, sfacts; -+ int mrest, srest; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ sx = mx = m->wx + ov; -+ sy = my = m->wy + oh; -+ sh = mh = m->wh - 2*oh - ih * (MIN(n, m->nmaster) - 1); -+ sw = mw = m->ww - 2*ov; -+ -+ if (m->nmaster && n > m->nmaster) { -+ sw = (mw - iv) * (1 - m->mfact); -+ mw = mw - iv - sw; -+ sx = mx + mw + iv; -+ sh = m->wh - 2*oh; -+ } -+ -+ getfacts(m, mh, sh, &mfacts, &sfacts, &mrest, &srest); -+ -+ if (n - m->nmaster > 0) /* override layout symbol */ -+ snprintf(m->ltsymbol, sizeof m->ltsymbol, "D %d", n - m->nmaster); -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -+ if (i < m->nmaster) { -+ resize(c, mx, my, mw - (2*c->bw), mh * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), 0); -+ my += HEIGHT(c) + ih; -+ } else { -+ resize(c, sx, sy, sw - (2*c->bw), sh - (2*c->bw), 0); -+ } -+} -+ -+/* -+ * Fibonacci layout + gaps -+ * https://dwm.suckless.org/patches/fibonacci/ -+ */ -+void -+fibonacci(Monitor *m, int s) -+{ -+ unsigned int i, n; -+ int nx, ny, nw, nh; -+ int oh, ov, ih, iv; -+ int nv, hrest = 0, wrest = 0, r = 1; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ nx = m->wx + ov; -+ ny = m->wy + oh; -+ nw = m->ww - 2*ov; -+ nh = m->wh - 2*oh; -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next)) { -+ if (r) { -+ if ((i % 2 && (nh - ih) / 2 <= (bh + 2*c->bw)) -+ || (!(i % 2) && (nw - iv) / 2 <= (bh + 2*c->bw))) { -+ r = 0; -+ } -+ if (r && i < n - 1) { -+ if (i % 2) { -+ nv = (nh - ih) / 2; -+ hrest = nh - 2*nv - ih; -+ nh = nv; -+ } else { -+ nv = (nw - iv) / 2; -+ wrest = nw - 2*nv - iv; -+ nw = nv; -+ } -+ -+ if ((i % 4) == 2 && !s) -+ nx += nw + iv; -+ else if ((i % 4) == 3 && !s) -+ ny += nh + ih; -+ } -+ -+ if ((i % 4) == 0) { -+ if (s) { -+ ny += nh + ih; -+ nh += hrest; -+ } -+ else { -+ nh -= hrest; -+ ny -= nh + ih; -+ } -+ } -+ else if ((i % 4) == 1) { -+ nx += nw + iv; -+ nw += wrest; -+ } -+ else if ((i % 4) == 2) { -+ ny += nh + ih; -+ nh += hrest; -+ if (i < n - 1) -+ nw += wrest; -+ } -+ else if ((i % 4) == 3) { -+ if (s) { -+ nx += nw + iv; -+ nw -= wrest; -+ } else { -+ nw -= wrest; -+ nx -= nw + iv; -+ nh += hrest; -+ } -+ } -+ if (i == 0) { -+ if (n != 1) { -+ nw = (m->ww - iv - 2*ov) - (m->ww - iv - 2*ov) * (1 - m->mfact); -+ wrest = 0; -+ } -+ ny = m->wy + oh; -+ } -+ else if (i == 1) -+ nw = m->ww - nw - iv - 2*ov; -+ i++; -+ } -+ -+ resize(c, nx, ny, nw - (2*c->bw), nh - (2*c->bw), False); -+ } -+} -+ -+void -+dwindle(Monitor *m) -+{ -+ fibonacci(m, 1); -+} -+ -+void -+spiral(Monitor *m) -+{ -+ fibonacci(m, 0); -+} -+ -+/* -+ * Gappless grid layout + gaps (ironically) -+ * https://dwm.suckless.org/patches/gaplessgrid/ -+ */ -+void -+gaplessgrid(Monitor *m) -+{ -+ unsigned int i, n; -+ int x, y, cols, rows, ch, cw, cn, rn, rrest, crest; // counters -+ int oh, ov, ih, iv; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ /* grid dimensions */ -+ for (cols = 0; cols <= n/2; cols++) -+ if (cols*cols >= n) -+ break; -+ if (n == 5) /* set layout against the general calculation: not 1:2:2, but 2:3 */ -+ cols = 2; -+ rows = n/cols; -+ cn = rn = 0; // reset column no, row no, client count -+ -+ ch = (m->wh - 2*oh - ih * (rows - 1)) / rows; -+ cw = (m->ww - 2*ov - iv * (cols - 1)) / cols; -+ rrest = (m->wh - 2*oh - ih * (rows - 1)) - ch * rows; -+ crest = (m->ww - 2*ov - iv * (cols - 1)) - cw * cols; -+ x = m->wx + ov; -+ y = m->wy + oh; -+ -+ for (i = 0, c = nexttiled(m->clients); c; i++, c = nexttiled(c->next)) { -+ if (i/rows + 1 > cols - n%cols) { -+ rows = n/cols + 1; -+ ch = (m->wh - 2*oh - ih * (rows - 1)) / rows; -+ rrest = (m->wh - 2*oh - ih * (rows - 1)) - ch * rows; -+ } -+ resize(c, -+ x, -+ y + rn*(ch + ih) + MIN(rn, rrest), -+ cw + (cn < crest ? 1 : 0) - 2*c->bw, -+ ch + (rn < rrest ? 1 : 0) - 2*c->bw, -+ 0); -+ rn++; -+ if (rn >= rows) { -+ rn = 0; -+ x += cw + ih + (cn < crest ? 1 : 0); -+ cn++; -+ } -+ } -+} -+ -+/* -+ * Gridmode layout + gaps -+ * https://dwm.suckless.org/patches/gridmode/ -+ */ -+void -+grid(Monitor *m) -+{ -+ unsigned int i, n; -+ int cx, cy, cw, ch, cc, cr, chrest, cwrest, cols, rows; -+ int oh, ov, ih, iv; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ -+ /* grid dimensions */ -+ for (rows = 0; rows <= n/2; rows++) -+ if (rows*rows >= n) -+ break; -+ cols = (rows && (rows - 1) * rows >= n) ? rows - 1 : rows; -+ -+ /* window geoms (cell height/width) */ -+ ch = (m->wh - 2*oh - ih * (rows - 1)) / (rows ? rows : 1); -+ cw = (m->ww - 2*ov - iv * (cols - 1)) / (cols ? cols : 1); -+ chrest = (m->wh - 2*oh - ih * (rows - 1)) - ch * rows; -+ cwrest = (m->ww - 2*ov - iv * (cols - 1)) - cw * cols; -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) { -+ cc = i / rows; -+ cr = i % rows; -+ cx = m->wx + ov + cc * (cw + iv) + MIN(cc, cwrest); -+ cy = m->wy + oh + cr * (ch + ih) + MIN(cr, chrest); -+ resize(c, cx, cy, cw + (cc < cwrest ? 1 : 0) - 2*c->bw, ch + (cr < chrest ? 1 : 0) - 2*c->bw, False); -+ } -+} -+ -+/* -+ * Horizontal grid layout + gaps -+ * https://dwm.suckless.org/patches/horizgrid/ -+ */ -+void -+horizgrid(Monitor *m) { -+ Client *c; -+ unsigned int n, i; -+ int oh, ov, ih, iv; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int sx = 0, sy = 0, sh = 0, sw = 0; -+ int ntop, nbottom = 1; -+ float mfacts = 0, sfacts = 0; -+ int mrest, srest, mtotal = 0, stotal = 0; -+ -+ /* Count windows */ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ if (n <= 2) -+ ntop = n; -+ else { -+ ntop = n / 2; -+ nbottom = n - ntop; -+ } -+ sx = mx = m->wx + ov; -+ sy = my = m->wy + oh; -+ sh = mh = m->wh - 2*oh; -+ sw = mw = m->ww - 2*ov; -+ -+ if (n > ntop) { -+ sh = (mh - ih) / 2; -+ mh = mh - ih - sh; -+ sy = my + mh + ih; -+ mw = m->ww - 2*ov - iv * (ntop - 1); -+ sw = m->ww - 2*ov - iv * (nbottom - 1); -+ } -+ -+ /* calculate facts */ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -+ if (i < ntop) -+ mfacts += c->cfact; -+ else -+ sfacts += c->cfact; -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -+ if (i < ntop) -+ mtotal += mh * (c->cfact / mfacts); -+ else -+ stotal += sw * (c->cfact / sfacts); -+ -+ mrest = mh - mtotal; -+ srest = sw - stotal; -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -+ if (i < ntop) { -+ resize(c, mx, my, mw * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), mh - (2*c->bw), 0); -+ mx += WIDTH(c) + iv; -+ } else { -+ resize(c, sx, sy, sw * (c->cfact / sfacts) + ((i - ntop) < srest ? 1 : 0) - (2*c->bw), sh - (2*c->bw), 0); -+ sx += WIDTH(c) + iv; -+ } -+} -+ -+/* -+ * nrowgrid layout + gaps -+ * https://dwm.suckless.org/patches/nrowgrid/ -+ */ -+void -+nrowgrid(Monitor *m) -+{ -+ unsigned int n; -+ int ri = 0, ci = 0; /* counters */ -+ int oh, ov, ih, iv; /* vanitygap settings */ -+ unsigned int cx, cy, cw, ch; /* client geometry */ -+ unsigned int uw = 0, uh = 0, uc = 0; /* utilization trackers */ -+ unsigned int cols, rows = m->nmaster + 1; -+ Client *c; -+ -+ /* count clients */ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ -+ /* nothing to do here */ -+ if (n == 0) -+ return; -+ -+ /* force 2 clients to always split vertically */ -+ if (FORCE_VSPLIT && n == 2) -+ rows = 1; -+ -+ /* never allow empty rows */ -+ if (n < rows) -+ rows = n; -+ -+ /* define first row */ -+ cols = n / rows; -+ uc = cols; -+ cy = m->wy + oh; -+ ch = (m->wh - 2*oh - ih*(rows - 1)) / rows; -+ uh = ch; -+ -+ for (c = nexttiled(m->clients); c; c = nexttiled(c->next), ci++) { -+ if (ci == cols) { -+ uw = 0; -+ ci = 0; -+ ri++; -+ -+ /* next row */ -+ cols = (n - uc) / (rows - ri); -+ uc += cols; -+ cy = m->wy + oh + uh + ih; -+ uh += ch + ih; -+ } -+ -+ cx = m->wx + ov + uw; -+ cw = (m->ww - 2*ov - uw) / (cols - ci); -+ uw += cw + iv; -+ -+ resize(c, cx, cy, cw - (2*c->bw), ch - (2*c->bw), 0); -+ } -+} -+ -+/* -+ * Default tile layout + gaps -+ */ -+static void -+tile(Monitor *m) -+{ -+ unsigned int i, n; -+ int oh, ov, ih, iv; -+ int mx = 0, my = 0, mh = 0, mw = 0; -+ int sx = 0, sy = 0, sh = 0, sw = 0; -+ float mfacts, sfacts; -+ int mrest, srest; -+ Client *c; -+ -+ getgaps(m, &oh, &ov, &ih, &iv, &n); -+ if (n == 0) -+ return; -+ -+ sx = mx = m->wx + ov; -+ sy = my = m->wy + oh; -+ mh = m->wh - 2*oh - ih * (MIN(n, m->nmaster) - 1); -+ sh = m->wh - 2*oh - ih * (n - m->nmaster - 1); -+ sw = mw = m->ww - 2*ov; -+ -+ if (m->nmaster && n > m->nmaster) { -+ sw = (mw - iv) * (1 - m->mfact); -+ mw = mw - iv - sw; -+ sx = mx + mw + iv; -+ } -+ -+ getfacts(m, mh, sh, &mfacts, &sfacts, &mrest, &srest); -+ -+ for (i = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) -+ if (i < m->nmaster) { -+ resize(c, mx, my, mw - (2*c->bw), mh * (c->cfact / mfacts) + (i < mrest ? 1 : 0) - (2*c->bw), 0); -+ my += HEIGHT(c) + ih; -+ } else { -+ resize(c, sx, sy, sw - (2*c->bw), sh * (c->cfact / sfacts) + ((i - m->nmaster) < srest ? 1 : 0) - (2*c->bw), 0); -+ sy += HEIGHT(c) + ih; -+ } -+} -\ No newline at end of file --- -2.19.1 - diff --git a/patches/dwm-dwmblocks-6.2.diff b/patches/dwm-dwmblocks-6.2.diff deleted file mode 100644 index 7ccf3be..0000000 --- a/patches/dwm-dwmblocks-6.2.diff +++ /dev/null @@ -1,401 +0,0 @@ -diff -ruN dwm-6.2-ori/config.def.h dwm-6.2/config.def.h ---- dwm-6.2-ori/config.def.h 2019-02-02 18:25:28.000000000 +0530 -+++ dwm-6.2/config.def.h 2020-12-27 19:45:35.127385861 +0530 -@@ -12,10 +12,26 @@ - static const char col_gray3[] = "#bbbbbb"; - static const char col_gray4[] = "#eeeeee"; - static const char col_cyan[] = "#005577"; -+static const char col1[] = "#ffffff"; -+static const char col2[] = "#ffffff"; -+static const char col3[] = "#ffffff"; -+static const char col4[] = "#ffffff"; -+static const char col5[] = "#ffffff"; -+static const char col6[] = "#ffffff"; -+ -+enum { SchemeNorm, SchemeCol1, SchemeCol2, SchemeCol3, SchemeCol4, -+ SchemeCol5, SchemeCol6, SchemeSel }; /* color schemes */ -+ - static const char *colors[][3] = { - /* fg bg border */ -- [SchemeNorm] = { col_gray3, col_gray1, col_gray2 }, -- [SchemeSel] = { col_gray4, col_cyan, col_cyan }, -+ [SchemeNorm] = { col_gray3, col_gray1, col_gray2 }, -+ [SchemeCol1] = { col1, col_gray1, col_gray2 }, -+ [SchemeCol2] = { col2, col_gray1, col_gray2 }, -+ [SchemeCol3] = { col3, col_gray1, col_gray2 }, -+ [SchemeCol4] = { col4, col_gray1, col_gray2 }, -+ [SchemeCol5] = { col5, col_gray1, col_gray2 }, -+ [SchemeCol6] = { col6, col_gray1, col_gray2 }, -+ [SchemeSel] = { col_gray4, col_cyan, col_cyan }, - }; - - /* tagging */ -@@ -103,7 +119,9 @@ - { ClkLtSymbol, 0, Button1, setlayout, {0} }, - { ClkLtSymbol, 0, Button3, setlayout, {.v = &layouts[2]} }, - { ClkWinTitle, 0, Button2, zoom, {0} }, -- { ClkStatusText, 0, Button2, spawn, {.v = termcmd } }, -+ { ClkStatusText, 0, Button1, sigdwmblocks, {.i = 1} }, -+ { ClkStatusText, 0, Button2, sigdwmblocks, {.i = 2} }, -+ { ClkStatusText, 0, Button3, sigdwmblocks, {.i = 3} }, - { ClkClientWin, MODKEY, Button1, movemouse, {0} }, - { ClkClientWin, MODKEY, Button2, togglefloating, {0} }, - { ClkClientWin, MODKEY, Button3, resizemouse, {0} }, -diff -ruN dwm-6.2-ori/dwm.c dwm-6.2/dwm.c ---- dwm-6.2-ori/dwm.c 2019-02-02 18:25:28.000000000 +0530 -+++ dwm-6.2/dwm.c 2021-08-03 19:15:58.734660076 +0530 -@@ -40,6 +40,7 @@ - #include - #endif /* XINERAMA */ - #include -+#include - - #include "drw.h" - #include "util.h" -@@ -56,10 +57,16 @@ - #define HEIGHT(X) ((X)->h + 2 * (X)->bw) - #define TAGMASK ((1 << LENGTH(tags)) - 1) - #define TEXTW(X) (drw_fontset_getwidth(drw, (X)) + lrpad) -+#define TTEXTW(X) (drw_fontset_getwidth(drw, (X))) -+ -+#define STATUSLENGTH 256 -+#define DWMBLOCKSLOCKFILE "/var/local/dwmblocks/dwmblocks.pid" -+#define DELIMITERENDCHAR 10 -+#define LSPAD (lrpad / 2) /* padding on left side of status text */ -+#define RSPAD (lrpad / 2) /* padding on right side of status text */ - - /* enums */ --enum { CurNormal, CurResize, CurMove, CurLast }; /* cursor */ --enum { SchemeNorm, SchemeSel }; /* color schemes */ -+enum { CurNormal, CurHand, CurResize, CurMove, CurLast }; /* cursor */ - enum { NetSupported, NetWMName, NetWMState, NetWMCheck, - NetWMFullscreen, NetActiveWindow, NetWMWindowType, - NetWMWindowTypeDialog, NetClientList, NetLast }; /* EWMH atoms */ -@@ -124,6 +131,7 @@ - unsigned int tagset[2]; - int showbar; - int topbar; -+ int statushandcursor; - Client *clients; - Client *sel; - Client *stack; -@@ -205,6 +213,7 @@ - static void seturgent(Client *c, int urg); - static void showhide(Client *c); - static void sigchld(int unused); -+static void sigdwmblocks(const Arg *arg); - static void spawn(const Arg *arg); - static void tag(const Arg *arg); - static void tagmon(const Arg *arg); -@@ -219,6 +228,7 @@ - static void updatebarpos(Monitor *m); - static void updatebars(void); - static void updateclientlist(void); -+static void updatedwmblockssig(int x); - static int updategeom(void); - static void updatenumlockmask(void); - static void updatesizehints(Client *c); -@@ -236,12 +246,15 @@ - - /* variables */ - static const char broken[] = "broken"; --static char stext[256]; -+static char stextc[STATUSLENGTH]; -+static char stexts[STATUSLENGTH]; - static int screen; - static int sw, sh; /* X display screen geometry width, height */ --static int bh, blw = 0; /* bar geometry */ -+static int bh, blw, ble; /* bar geometry */ -+static int wstext; /* width of status text */ - static int lrpad; /* sum of left and right padding for text */ - static int (*xerrorxlib)(Display *, XErrorEvent *); -+static unsigned int dwmblockssig; - static unsigned int numlockmask = 0; - static void (*handler[LASTEvent]) (XEvent *) = { - [ButtonPress] = buttonpress, -@@ -416,13 +429,13 @@ - void - buttonpress(XEvent *e) - { -- unsigned int i, x, click; -+ int i, x; -+ unsigned int click; - Arg arg = {0}; - Client *c; - Monitor *m; - XButtonPressedEvent *ev = &e->xbutton; - -- click = ClkRootWin; - /* focus monitor if necessary */ - if ((m = wintomon(ev->window)) && m != selmon) { - unfocus(selmon->sel, 1); -@@ -430,25 +443,29 @@ - focus(NULL); - } - if (ev->window == selmon->barwin) { -- i = x = 0; -- do -- x += TEXTW(tags[i]); -- while (ev->x >= x && ++i < LENGTH(tags)); -- if (i < LENGTH(tags)) { -- click = ClkTagBar; -- arg.ui = 1 << i; -- } else if (ev->x < x + blw) -- click = ClkLtSymbol; -- else if (ev->x > selmon->ww - TEXTW(stext)) -- click = ClkStatusText; -- else -- click = ClkWinTitle; -+ if (ev->x < ble - blw) { -+ i = -1, x = -ev->x; -+ do -+ x += TEXTW(tags[++i]); -+ while (x <= 0); -+ click = ClkTagBar; -+ arg.ui = 1 << i; -+ } else if (ev->x < ble) -+ click = ClkLtSymbol; -+ else if (ev->x < selmon->ww - wstext) -+ click = ClkWinTitle; -+ else if ((x = selmon->ww - RSPAD - ev->x) > 0 && (x -= wstext - LSPAD - RSPAD) <= 0) { -+ updatedwmblockssig(x); -+ click = ClkStatusText; -+ } else -+ return; - } else if ((c = wintoclient(ev->window))) { - focus(c); - restack(selmon); - XAllowEvents(dpy, ReplayPointer, CurrentTime); - click = ClkClientWin; -- } -+ } else -+ click = ClkRootWin; - for (i = 0; i < LENGTH(buttons); i++) - if (click == buttons[i].click && buttons[i].func && buttons[i].button == ev->button - && CLEANMASK(buttons[i].mask) == CLEANMASK(ev->state)) -@@ -695,7 +712,7 @@ - void - drawbar(Monitor *m) - { -- int x, w, sw = 0; -+ int x, w; - int boxs = drw->fonts->h / 9; - int boxw = drw->fonts->h / 6 + 2; - unsigned int i, occ = 0, urg = 0; -@@ -703,9 +720,32 @@ - - /* draw status first so it can be overdrawn by tags later */ - if (m == selmon) { /* status is only drawn on selected monitor */ -- drw_setscheme(drw, scheme[SchemeNorm]); -- sw = TEXTW(stext) - lrpad + 2; /* 2px right padding */ -- drw_text(drw, m->ww - sw, 0, sw, bh, 0, stext, 0); -+ char *stc = stextc; -+ char *stp = stextc; -+ char tmp; -+ -+ drw_setscheme(drw, scheme[SchemeNorm]); -+ x = m->ww - wstext; -+ drw_rect(drw, x, 0, LSPAD, bh, 1, 1); x += LSPAD; /* to keep left padding clean */ -+ for (;;) { -+ if ((unsigned char)*stc >= ' ') { -+ stc++; -+ continue; -+ } -+ tmp = *stc; -+ if (stp != stc) { -+ *stc = '\0'; -+ x = drw_text(drw, x, 0, TTEXTW(stp), bh, 0, stp, 0); -+ } -+ if (tmp == '\0') -+ break; -+ if (tmp - DELIMITERENDCHAR - 1 < LENGTH(colors)) -+ drw_setscheme(drw, scheme[tmp - DELIMITERENDCHAR - 1]); -+ *stc = tmp; -+ stp = ++stc; -+ } -+ drw_setscheme(drw, scheme[SchemeNorm]); -+ drw_rect(drw, x, 0, m->ww - x, bh, 1, 1); /* to keep right padding clean */ - } - - for (c = m->clients; c; c = c->next) { -@@ -724,11 +764,17 @@ - urg & 1 << i); - x += w; - } -- w = blw = TEXTW(m->ltsymbol); -+ w = TEXTW(m->ltsymbol); - drw_setscheme(drw, scheme[SchemeNorm]); - x = drw_text(drw, x, 0, w, bh, lrpad / 2, m->ltsymbol, 0); - -- if ((w = m->ww - sw - x) > bh) { -+ if (m == selmon) { -+ blw = w, ble = x; -+ w = m->ww - wstext - x; -+ } else -+ w = m->ww - x; -+ -+ if (w > bh) { - if (m->sel) { - drw_setscheme(drw, scheme[m == selmon ? SchemeSel : SchemeNorm]); - drw_text(drw, x, 0, w, bh, lrpad / 2, m->sel->name, 0); -@@ -1119,17 +1165,24 @@ - motionnotify(XEvent *e) - { - static Monitor *mon = NULL; -+ int x; - Monitor *m; - XMotionEvent *ev = &e->xmotion; - -- if (ev->window != root) -- return; -- if ((m = recttomon(ev->x_root, ev->y_root, 1, 1)) != mon && mon) { -- unfocus(selmon->sel, 1); -- selmon = m; -- focus(NULL); -- } -- mon = m; -+ if (ev->window == root) { -+ if ((m = recttomon(ev->x_root, ev->y_root, 1, 1)) != mon && mon) { -+ unfocus(selmon->sel, 1); -+ selmon = m; -+ focus(NULL); -+ } -+ mon = m; -+ } else if (ev->window == selmon->barwin && (x = selmon->ww - RSPAD - ev->x) > 0 -+ && (x -= wstext - LSPAD - RSPAD) <= 0) -+ updatedwmblockssig(x); -+ else if (selmon->statushandcursor) { -+ selmon->statushandcursor = 0; -+ XDefineCursor(dpy, selmon->barwin, cursor[CurNormal]->cursor); -+ } - } - - void -@@ -1564,6 +1617,7 @@ - netatom[NetClientList] = XInternAtom(dpy, "_NET_CLIENT_LIST", False); - /* init cursors */ - cursor[CurNormal] = drw_cur_create(drw, XC_left_ptr); -+ cursor[CurHand] = drw_cur_create(drw, XC_hand2); - cursor[CurResize] = drw_cur_create(drw, XC_sizing); - cursor[CurMove] = drw_cur_create(drw, XC_fleur); - /* init appearance */ -@@ -1637,6 +1691,37 @@ - } - - void -+sigdwmblocks(const Arg *arg) -+{ -+ static int fd = -1; -+ struct flock fl; -+ union sigval sv; -+ -+ if (!dwmblockssig) -+ return; -+ fl.l_type = F_WRLCK; -+ fl.l_whence = SEEK_SET; -+ fl.l_start = 0; -+ fl.l_len = 0; -+ if (fd != -1) { -+ if (fcntl(fd, F_GETLK, &fl) != -1 && fl.l_type == F_WRLCK) -+ goto signal; -+ close(fd); -+ fl.l_type = F_WRLCK; -+ } -+ if ((fd = open(DWMBLOCKSLOCKFILE, O_RDONLY | O_CLOEXEC)) == -1) -+ return; -+ if (fcntl(fd, F_GETLK, &fl) == -1 || fl.l_type != F_WRLCK) { -+ close(fd); -+ fd = -1; -+ return; -+ } -+signal: -+ sv.sival_int = (dwmblockssig << 8) | arg->i; -+ sigqueue(fl.l_pid, SIGRTMIN, sv); -+} -+ -+void - spawn(const Arg *arg) - { - if (arg->v == dmenucmd) -@@ -1805,7 +1890,7 @@ - XSetWindowAttributes wa = { - .override_redirect = True, - .background_pixmap = ParentRelative, -- .event_mask = ButtonPressMask|ExposureMask -+ .event_mask = ButtonPressMask|ExposureMask|PointerMotionMask - }; - XClassHint ch = {"dwm", "dwm"}; - for (m = mons; m; m = m->next) { -@@ -1847,6 +1932,41 @@ - (unsigned char *) &(c->win), 1); - } - -+void -+updatedwmblockssig(int x) -+{ -+ char *sts = stexts; -+ char *stp = stexts; -+ char tmp; -+ -+ while (*sts != '\0') { -+ if ((unsigned char)*sts >= ' ') { -+ sts++; -+ continue; -+ } -+ tmp = *sts; -+ *sts = '\0'; -+ x += TTEXTW(stp); -+ *sts = tmp; -+ if (x > 0) { -+ if (tmp == DELIMITERENDCHAR) -+ break; -+ if (!selmon->statushandcursor) { -+ selmon->statushandcursor = 1; -+ XDefineCursor(dpy, selmon->barwin, cursor[CurHand]->cursor); -+ } -+ dwmblockssig = tmp; -+ return; -+ } -+ stp = ++sts; -+ } -+ if (selmon->statushandcursor) { -+ selmon->statushandcursor = 0; -+ XDefineCursor(dpy, selmon->barwin, cursor[CurNormal]->cursor); -+ } -+ dwmblockssig = 0; -+} -+ - int - updategeom(void) - { -@@ -1987,9 +2107,27 @@ - void - updatestatus(void) - { -- if (!gettextprop(root, XA_WM_NAME, stext, sizeof(stext))) -- strcpy(stext, "dwm-"VERSION); -- drawbar(selmon); -+ char rawstext[STATUSLENGTH]; -+ -+ if (gettextprop(root, XA_WM_NAME, rawstext, sizeof rawstext)) { -+ char stextp[STATUSLENGTH]; -+ char *stp = stextp, *stc = stextc, *sts = stexts; -+ -+ for (char *rst = rawstext; *rst != '\0'; rst++) -+ if ((unsigned char)*rst >= ' ') -+ *(stp++) = *(stc++) = *(sts++) = *rst; -+ else if ((unsigned char)*rst > DELIMITERENDCHAR) -+ *(stc++) = *rst; -+ else -+ *(sts++) = *rst; -+ *stp = *stc = *sts = '\0'; -+ wstext = TTEXTW(stextp) + LSPAD + RSPAD; -+ } else { -+ strcpy(stextc, "dwm-"VERSION); -+ strcpy(stexts, stextc); -+ wstext = TTEXTW(stextc) + LSPAD + RSPAD; -+ } -+ drawbar(selmon); - } - - void diff --git a/patches/dwm-fullscreen-6.2.diff b/patches/dwm-fullscreen-6.2.diff deleted file mode 100644 index 36e3140..0000000 --- a/patches/dwm-fullscreen-6.2.diff +++ /dev/null @@ -1,56 +0,0 @@ -From 54719285bd1a984e2efce6e8a8eab184fec11abf Mon Sep 17 00:00:00 2001 -From: Sermak -Date: Mon, 8 Jul 2019 01:06:44 +0200 -Subject: [PATCH] Simulate toggleable fullscreen mode - ---- - config.def.h | 1 + - dwm.c | 14 ++++++++++++++ - 2 files changed, 15 insertions(+) - -diff --git a/config.def.h b/config.def.h -index 1c0b587..f774cc5 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -76,6 +76,7 @@ static Key keys[] = { - { MODKEY, XK_t, setlayout, {.v = &layouts[0]} }, - { MODKEY, XK_f, setlayout, {.v = &layouts[1]} }, - { MODKEY, XK_m, setlayout, {.v = &layouts[2]} }, -+ { MODKEY|ShiftMask, XK_f, fullscreen, {0} }, - { MODKEY, XK_space, setlayout, {0} }, - { MODKEY|ShiftMask, XK_space, togglefloating, {0} }, - { MODKEY, XK_0, view, {.ui = ~0 } }, -diff --git a/dwm.c b/dwm.c -index 4465af1..04b1e06 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -199,6 +199,7 @@ static void sendmon(Client *c, Monitor *m); - static void setclientstate(Client *c, long state); - static void setfocus(Client *c); - static void setfullscreen(Client *c, int fullscreen); -+static void fullscreen(const Arg *arg); - static void setlayout(const Arg *arg); - static void setmfact(const Arg *arg); - static void setup(void); -@@ -1497,6 +1498,19 @@ setfullscreen(Client *c, int fullscreen) - } - } - -+Layout *last_layout; -+void -+fullscreen(const Arg *arg) -+{ -+ if (selmon->showbar) { -+ for(last_layout = (Layout *)layouts; last_layout != selmon->lt[selmon->sellt]; last_layout++); -+ setlayout(&((Arg) { .v = &layouts[2] })); -+ } else { -+ setlayout(&((Arg) { .v = last_layout })); -+ } -+ togglebar(arg); -+} -+ - void - setlayout(const Arg *arg) - { --- -2.22.0 diff --git a/patches/dwm-scratchpads-20200414-728d397b.diff b/patches/dwm-scratchpads-20200414-728d397b.diff deleted file mode 100644 index d3e90c0..0000000 --- a/patches/dwm-scratchpads-20200414-728d397b.diff +++ /dev/null @@ -1,199 +0,0 @@ -From 728d397b21982af88737277fd9d6939a7b558786 Mon Sep 17 00:00:00 2001 -From: Christian Tenllado -Date: Tue, 14 Apr 2020 23:31:15 +0200 -Subject: [PATCH] Multiple scratchpads - -This patch enables multiple scratchpads, each with one asigned window. -This enables the same scratchpad workflow that you have in i3. - -Scratchpads are implemented as special tags, whose mask does not -apply to new spawned windows. To assign a window to a scratchpad you -have to set up a rule, as you do with regular tags. - -Windows tagged with scratchpad tags can be set floating or not in the -rules array. Most users would probably want them floating (i3 style), -but having them tiled does also perfectly work and might fit better the -DWM approach. In case they are set floating, the patch moves them to the -center of the screen whenever they are shown. The patch can easily be -modified to make this last feature configurable in the rules array (see -the center patch). - -The togglescratch function, borrowed from the previous scratchpad patch -and slightly modified, can be used to spawn a registered scratchpad -process or toggle its view. This function looks for a window tagged with -the selected scratchpad tag. If it is found its view is toggled. If it is -not found the corresponding registered command is spawned. The -config.def.h shows three examples of its use to spawn a terminal in the -first scratchpad tag, a second terminal running ranger on the second -scratchpad tag and the keepassxc application to manage passwords on a -third scratchpad tag. - -If you prefer to spawn your scratchpad applications from the startup -script, you might opt for binding keys to toggleview instead, as -scratchpads are just special tags (you may even extend the TAGKEYS macro -to generalize the key bindings). ---- - config.def.h | 28 ++++++++++++++++++++++++---- - dwm.c | 43 +++++++++++++++++++++++++++++++++++++++++-- - 2 files changed, 65 insertions(+), 6 deletions(-) - -diff --git a/config.def.h b/config.def.h -index 1c0b587..06265e1 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -18,17 +18,33 @@ static const char *colors[][3] = { - [SchemeSel] = { col_gray4, col_cyan, col_cyan }, - }; - -+typedef struct { -+ const char *name; -+ const void *cmd; -+} Sp; -+const char *spcmd1[] = {"st", "-n", "spterm", "-g", "120x34", NULL }; -+const char *spcmd2[] = {"st", "-n", "spfm", "-g", "144x41", "-e", "ranger", NULL }; -+const char *spcmd3[] = {"keepassxc", NULL }; -+static Sp scratchpads[] = { -+ /* name cmd */ -+ {"spterm", spcmd1}, -+ {"spranger", spcmd2}, -+ {"keepassxc", spcmd3}, -+}; -+ - /* tagging */ - static const char *tags[] = { "1", "2", "3", "4", "5", "6", "7", "8", "9" }; -- - static const Rule rules[] = { - /* xprop(1): - * WM_CLASS(STRING) = instance, class - * WM_NAME(STRING) = title - */ - /* class instance title tags mask isfloating monitor */ -- { "Gimp", NULL, NULL, 0, 1, -1 }, -- { "Firefox", NULL, NULL, 1 << 8, 0, -1 }, -+ { "Gimp", NULL, NULL, 0, 1, -1 }, -+ { "Firefox", NULL, NULL, 1 << 8, 0, -1 }, -+ { NULL, "spterm", NULL, SPTAG(0), 1, -1 }, -+ { NULL, "spfm", NULL, SPTAG(1), 1, -1 }, -+ { NULL, "keepassxc", NULL, SPTAG(2), 0, -1 }, - }; - - /* layout(s) */ -@@ -59,6 +75,7 @@ static char dmenumon[2] = "0"; /* component of dmenucmd, manipulated in spawn() - static const char *dmenucmd[] = { "dmenu_run", "-m", dmenumon, "-fn", dmenufont, "-nb", col_gray1, "-nf", col_gray3, "-sb", col_cyan, "-sf", col_gray4, NULL }; - static const char *termcmd[] = { "st", NULL }; - -+ - static Key keys[] = { - /* modifier key function argument */ - { MODKEY, XK_p, spawn, {.v = dmenucmd } }, -@@ -84,6 +101,9 @@ static Key keys[] = { - { MODKEY, XK_period, focusmon, {.i = +1 } }, - { MODKEY|ShiftMask, XK_comma, tagmon, {.i = -1 } }, - { MODKEY|ShiftMask, XK_period, tagmon, {.i = +1 } }, -+ { MODKEY, XK_y, togglescratch, {.ui = 0 } }, -+ { MODKEY, XK_u, togglescratch, {.ui = 1 } }, -+ { MODKEY, XK_x, togglescratch, {.ui = 2 } }, - TAGKEYS( XK_1, 0) - TAGKEYS( XK_2, 1) - TAGKEYS( XK_3, 2) -@@ -106,7 +126,7 @@ static Button buttons[] = { - { ClkStatusText, 0, Button2, spawn, {.v = termcmd } }, - { ClkClientWin, MODKEY, Button1, movemouse, {0} }, - { ClkClientWin, MODKEY, Button2, togglefloating, {0} }, -- { ClkClientWin, MODKEY, Button3, resizemouse, {0} }, -+ { ClkClientWin, MODKEY, Button1, resizemouse, {0} }, - { ClkTagBar, 0, Button1, view, {0} }, - { ClkTagBar, 0, Button3, toggleview, {0} }, - { ClkTagBar, MODKEY, Button1, tag, {0} }, -diff --git a/dwm.c b/dwm.c -index 4465af1..646aa1a 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -54,7 +54,10 @@ - #define MOUSEMASK (BUTTONMASK|PointerMotionMask) - #define WIDTH(X) ((X)->w + 2 * (X)->bw) - #define HEIGHT(X) ((X)->h + 2 * (X)->bw) --#define TAGMASK ((1 << LENGTH(tags)) - 1) -+#define NUMTAGS (LENGTH(tags) + LENGTH(scratchpads)) -+#define TAGMASK ((1 << NUMTAGS) - 1) -+#define SPTAG(i) ((1 << LENGTH(tags)) << (i)) -+#define SPTAGMASK (((1 << LENGTH(scratchpads))-1) << LENGTH(tags)) - #define TEXTW(X) (drw_fontset_getwidth(drw, (X)) + lrpad) - - /* enums */ -@@ -211,6 +214,7 @@ static void tagmon(const Arg *arg); - static void tile(Monitor *); - static void togglebar(const Arg *arg); - static void togglefloating(const Arg *arg); -+static void togglescratch(const Arg *arg); - static void toggletag(const Arg *arg); - static void toggleview(const Arg *arg); - static void unfocus(Client *c, int setfocus); -@@ -299,6 +303,11 @@ applyrules(Client *c) - { - c->isfloating = r->isfloating; - c->tags |= r->tags; -+ if ((r->tags & SPTAGMASK) && r->isfloating) { -+ c->x = c->mon->wx + (c->mon->ww / 2 - WIDTH(c) / 2); -+ c->y = c->mon->wy + (c->mon->wh / 2 - HEIGHT(c) / 2); -+ } -+ - for (m = mons; m && m->num != r->monitor; m = m->next); - if (m) - c->mon = m; -@@ -308,7 +317,7 @@ applyrules(Client *c) - XFree(ch.res_class); - if (ch.res_name) - XFree(ch.res_name); -- c->tags = c->tags & TAGMASK ? c->tags & TAGMASK : c->mon->tagset[c->mon->seltags]; -+ c->tags = c->tags & TAGMASK ? c->tags & TAGMASK : (c->mon->tagset[c->mon->seltags] & ~SPTAGMASK); - } - - int -@@ -1616,6 +1625,10 @@ showhide(Client *c) - if (!c) - return; - if (ISVISIBLE(c)) { -+ if ((c->tags & SPTAGMASK) && c->isfloating) { -+ c->x = c->mon->wx + (c->mon->ww / 2 - WIDTH(c) / 2); -+ c->y = c->mon->wy + (c->mon->wh / 2 - HEIGHT(c) / 2); -+ } - /* show clients top down */ - XMoveWindow(dpy, c->win, c->x, c->y); - if ((!c->mon->lt[c->mon->sellt]->arrange || c->isfloating) && !c->isfullscreen) -@@ -1719,6 +1732,32 @@ togglefloating(const Arg *arg) - arrange(selmon); - } - -+void -+togglescratch(const Arg *arg) -+{ -+ Client *c; -+ unsigned int found = 0; -+ unsigned int scratchtag = SPTAG(arg->ui); -+ Arg sparg = {.v = scratchpads[arg->ui].cmd}; -+ -+ for (c = selmon->clients; c && !(found = c->tags & scratchtag); c = c->next); -+ if (found) { -+ unsigned int newtagset = selmon->tagset[selmon->seltags] ^ scratchtag; -+ if (newtagset) { -+ selmon->tagset[selmon->seltags] = newtagset; -+ focus(NULL); -+ arrange(selmon); -+ } -+ if (ISVISIBLE(c)) { -+ focus(c); -+ restack(selmon); -+ } -+ } else { -+ selmon->tagset[selmon->seltags] |= scratchtag; -+ spawn(&sparg); -+ } -+} -+ - void - toggletag(const Arg *arg) - { --- -2.20.1 - diff --git a/patches/dwm-stacker-6.2.diff b/patches/dwm-stacker-6.2.diff deleted file mode 100644 index 8fe3b80..0000000 --- a/patches/dwm-stacker-6.2.diff +++ /dev/null @@ -1,197 +0,0 @@ -From d04f2d00688c8b0969d4f10f460c980dd91dac37 Mon Sep 17 00:00:00 2001 -From: MLquest8 -Date: Fri, 12 Jun 2020 16:04:18 +0400 -Subject: [PATCH] stacker updated for version 6.2 - ---- - config.def.h | 14 +++++++-- - dwm.c | 88 ++++++++++++++++++++++++++++++++++++++++------------ - 2 files changed, 80 insertions(+), 22 deletions(-) - -diff --git a/config.def.h b/config.def.h -index 1c0b587..d28f8fc 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -50,6 +50,14 @@ static const Layout layouts[] = { - { MODKEY|ControlMask, KEY, toggleview, {.ui = 1 << TAG} }, \ - { MODKEY|ShiftMask, KEY, tag, {.ui = 1 << TAG} }, \ - { MODKEY|ControlMask|ShiftMask, KEY, toggletag, {.ui = 1 << TAG} }, -+#define STACKKEYS(MOD,ACTION) \ -+ { MOD, XK_j, ACTION##stack, {.i = INC(+1) } }, \ -+ { MOD, XK_k, ACTION##stack, {.i = INC(-1) } }, \ -+ { MOD, XK_grave, ACTION##stack, {.i = PREVSEL } }, \ -+ { MOD, XK_q, ACTION##stack, {.i = 0 } }, \ -+ { MOD, XK_a, ACTION##stack, {.i = 1 } }, \ -+ { MOD, XK_z, ACTION##stack, {.i = 2 } }, \ -+ { MOD, XK_x, ACTION##stack, {.i = -1 } }, - - /* helper for spawning shell commands in the pre dwm-5.0 fashion */ - #define SHCMD(cmd) { .v = (const char*[]){ "/bin/sh", "-c", cmd, NULL } } -@@ -64,8 +72,8 @@ static Key keys[] = { - { MODKEY, XK_p, spawn, {.v = dmenucmd } }, - { MODKEY|ShiftMask, XK_Return, spawn, {.v = termcmd } }, - { MODKEY, XK_b, togglebar, {0} }, -- { MODKEY, XK_j, focusstack, {.i = +1 } }, -- { MODKEY, XK_k, focusstack, {.i = -1 } }, -+ STACKKEYS(MODKEY, focus) -+ STACKKEYS(MODKEY|ShiftMask, push) - { MODKEY, XK_i, incnmaster, {.i = +1 } }, - { MODKEY, XK_d, incnmaster, {.i = -1 } }, - { MODKEY, XK_h, setmfact, {.f = -0.05} }, -@@ -93,7 +101,7 @@ static Key keys[] = { - TAGKEYS( XK_7, 6) - TAGKEYS( XK_8, 7) - TAGKEYS( XK_9, 8) -- { MODKEY|ShiftMask, XK_q, quit, {0} }, -+ { MODKEY|ShiftMask, XK_BackSpace, quit, {0} }, - }; - - /* button definitions */ -diff --git a/dwm.c b/dwm.c -index 9fd0286..6c302c3 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -47,15 +47,21 @@ - /* macros */ - #define BUTTONMASK (ButtonPressMask|ButtonReleaseMask) - #define CLEANMASK(mask) (mask & ~(numlockmask|LockMask) & (ShiftMask|ControlMask|Mod1Mask|Mod2Mask|Mod3Mask|Mod4Mask|Mod5Mask)) -+#define GETINC(X) ((X) - 2000) -+#define INC(X) ((X) + 2000) - #define INTERSECT(x,y,w,h,m) (MAX(0, MIN((x)+(w),(m)->wx+(m)->ww) - MAX((x),(m)->wx)) \ - * MAX(0, MIN((y)+(h),(m)->wy+(m)->wh) - MAX((y),(m)->wy))) -+#define ISINC(X) ((X) > 1000 && (X) < 3000) - #define ISVISIBLE(C) ((C->tags & C->mon->tagset[C->mon->seltags])) -+#define PREVSEL 3000 - #define LENGTH(X) (sizeof X / sizeof X[0]) -+#define MOD(N,M) ((N)%(M) < 0 ? (N)%(M) + (M) : (N)%(M)) - #define MOUSEMASK (BUTTONMASK|PointerMotionMask) - #define WIDTH(X) ((X)->w + 2 * (X)->bw) - #define HEIGHT(X) ((X)->h + 2 * (X)->bw) - #define TAGMASK ((1 << LENGTH(tags)) - 1) - #define TEXTW(X) (drw_fontset_getwidth(drw, (X)) + lrpad) -+#define TRUNC(X,A,B) (MAX((A), MIN((X), (B)))) - - /* enums */ - enum { CurNormal, CurResize, CurMove, CurLast }; /* cursor */ -@@ -187,6 +193,7 @@ static void movemouse(const Arg *arg); - static Client *nexttiled(Client *c); - static void pop(Client *); - static void propertynotify(XEvent *e); -+static void pushstack(const Arg *arg); - static void quit(const Arg *arg); - static Monitor *recttomon(int x, int y, int w, int h); - static void resize(Client *c, int x, int y, int w, int h, int interact); -@@ -207,6 +214,7 @@ static void seturgent(Client *c, int urg); - static void showhide(Client *c); - static void sigchld(int unused); - static void spawn(const Arg *arg); -+static int stackpos(const Arg *arg); - static void tag(const Arg *arg); - static void tagmon(const Arg *arg); - static void tile(Monitor *); -@@ -833,27 +841,16 @@ focusmon(const Arg *arg) - void - focusstack(const Arg *arg) - { -- Client *c = NULL, *i; -+ int i = stackpos(arg); -+ Client *c, *p; - -- if (!selmon->sel) -+ if(i < 0) - return; -- if (arg->i > 0) { -- for (c = selmon->sel->next; c && !ISVISIBLE(c); c = c->next); -- if (!c) -- for (c = selmon->clients; c && !ISVISIBLE(c); c = c->next); -- } else { -- for (i = selmon->clients; i != selmon->sel; i = i->next) -- if (ISVISIBLE(i)) -- c = i; -- if (!c) -- for (; i; i = i->next) -- if (ISVISIBLE(i)) -- c = i; -- } -- if (c) { -- focus(c); -- restack(selmon); -- } -+ -+ for(p = NULL, c = selmon->clients; c && (i || !ISVISIBLE(c)); -+ i -= ISVISIBLE(c) ? 1 : 0, p = c, c = c->next); -+ focus(c ? c : p); -+ restack(selmon); - } - - Atom -@@ -1246,6 +1243,29 @@ propertynotify(XEvent *e) - } - } - -+void -+pushstack(const Arg *arg) { -+ int i = stackpos(arg); -+ Client *sel = selmon->sel, *c, *p; -+ -+ if(i < 0) -+ return; -+ else if(i == 0) { -+ detach(sel); -+ attach(sel); -+ } -+ else { -+ for(p = NULL, c = selmon->clients; c; p = c, c = c->next) -+ if(!(i -= (ISVISIBLE(c) && c != sel))) -+ break; -+ c = c ? c : p; -+ detach(sel); -+ sel->next = c->next; -+ c->next = sel; -+ } -+ arrange(selmon); -+} -+ - void - quit(const Arg *arg) - { -@@ -1653,6 +1673,36 @@ spawn(const Arg *arg) - } - } - -+int -+stackpos(const Arg *arg) { -+ int n, i; -+ Client *c, *l; -+ -+ if(!selmon->clients) -+ return -1; -+ -+ if(arg->i == PREVSEL) { -+ for(l = selmon->stack; l && (!ISVISIBLE(l) || l == selmon->sel); l = l->snext); -+ if(!l) -+ return -1; -+ for(i = 0, c = selmon->clients; c != l; i += ISVISIBLE(c) ? 1 : 0, c = c->next); -+ return i; -+ } -+ else if(ISINC(arg->i)) { -+ if(!selmon->sel) -+ return -1; -+ for(i = 0, c = selmon->clients; c != selmon->sel; i += ISVISIBLE(c) ? 1 : 0, c = c->next); -+ for(n = i; c; n += ISVISIBLE(c) ? 1 : 0, c = c->next); -+ return MOD(i + GETINC(arg->i), n); -+ } -+ else if(arg->i < 0) { -+ for(i = 0, c = selmon->clients; c; i += ISVISIBLE(c) ? 1 : 0, c = c->next); -+ return MAX(i + arg->i, 0); -+ } -+ else -+ return arg->i; -+} -+ - void - tag(const Arg *arg) - { --- -2.26.2 - diff --git a/patches/dwm-sticky-6.1.diff b/patches/dwm-sticky-6.1.diff deleted file mode 100644 index 717793f..0000000 --- a/patches/dwm-sticky-6.1.diff +++ /dev/null @@ -1,58 +0,0 @@ -diff --git a/config.def.h b/config.def.h -index 7054c06..9b5d5b8 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -76,6 +76,7 @@ static Key keys[] = { - { MODKEY, XK_m, setlayout, {.v = &layouts[2]} }, - { MODKEY, XK_space, setlayout, {0} }, - { MODKEY|ShiftMask, XK_space, togglefloating, {0} }, -+ { MODKEY, XK_s, togglesticky, {0} }, - { MODKEY, XK_0, view, {.ui = ~0 } }, - { MODKEY|ShiftMask, XK_0, tag, {.ui = ~0 } }, - { MODKEY, XK_comma, focusmon, {.i = -1 } }, -diff --git a/dwm.c b/dwm.c -index 0362114..0ef5c7f 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -49,7 +49,7 @@ - #define CLEANMASK(mask) (mask & ~(numlockmask|LockMask) & (ShiftMask|ControlMask|Mod1Mask|Mod2Mask|Mod3Mask|Mod4Mask|Mod5Mask)) - #define INTERSECT(x,y,w,h,m) (MAX(0, MIN((x)+(w),(m)->wx+(m)->ww) - MAX((x),(m)->wx)) \ - * MAX(0, MIN((y)+(h),(m)->wy+(m)->wh) - MAX((y),(m)->wy))) --#define ISVISIBLE(C) ((C->tags & C->mon->tagset[C->mon->seltags])) -+#define ISVISIBLE(C) ((C->tags & C->mon->tagset[C->mon->seltags]) || C->issticky) - #define LENGTH(X) (sizeof X / sizeof X[0]) - #define MOUSEMASK (BUTTONMASK|PointerMotionMask) - #define WIDTH(X) ((X)->w + 2 * (X)->bw) -@@ -92,7 +92,7 @@ struct Client { - int basew, baseh, incw, inch, maxw, maxh, minw, minh; - int bw, oldbw; - unsigned int tags; -- int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen; -+ int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen, issticky; - Client *next; - Client *snext; - Monitor *mon; -@@ -211,6 +211,7 @@ static void tagmon(const Arg *arg); - static void tile(Monitor *); - static void togglebar(const Arg *arg); - static void togglefloating(const Arg *arg); -+static void togglesticky(const Arg *arg); - static void toggletag(const Arg *arg); - static void toggleview(const Arg *arg); - static void unfocus(Client *c, int setfocus); -@@ -1713,6 +1714,15 @@ togglefloating(const Arg *arg) - } - - void -+togglesticky(const Arg *arg) -+{ -+ if (!selmon->sel) -+ return; -+ selmon->sel->issticky = !selmon->sel->issticky; -+ arrange(selmon); -+} -+ -+void - toggletag(const Arg *arg) - { - unsigned int newtags; diff --git a/patches/dwm-swallow-20201211-61bb8b2.diff b/patches/dwm-swallow-20201211-61bb8b2.diff deleted file mode 100644 index 6bc0a1f..0000000 --- a/patches/dwm-swallow-20201211-61bb8b2.diff +++ /dev/null @@ -1,412 +0,0 @@ -From f0cdf40e0a7126838d051eb84d84b91421b771d6 Mon Sep 17 00:00:00 2001 -From: 0x1bi -Date: Fri, 11 Dec 2020 10:16:25 -0500 -Subject: [PATCH] fix swallow for openbsd - ---- - config.def.h | 9 +- - config.mk | 3 +- - dwm.c | 235 +++++++++++++++++++++++++++++++++++++++++++++++++-- - 3 files changed, 237 insertions(+), 10 deletions(-) - -diff --git a/config.def.h b/config.def.h -index 1c0b587..fe51476 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -3,6 +3,7 @@ - /* appearance */ - static const unsigned int borderpx = 1; /* border pixel of windows */ - static const unsigned int snap = 32; /* snap pixel */ -+static const int swallowfloating = 0; /* 1 means swallow floating windows by default */ - static const int showbar = 1; /* 0 means no bar */ - static const int topbar = 1; /* 0 means bottom bar */ - static const char *fonts[] = { "monospace:size=10" }; -@@ -26,9 +27,11 @@ static const Rule rules[] = { - * WM_CLASS(STRING) = instance, class - * WM_NAME(STRING) = title - */ -- /* class instance title tags mask isfloating monitor */ -- { "Gimp", NULL, NULL, 0, 1, -1 }, -- { "Firefox", NULL, NULL, 1 << 8, 0, -1 }, -+ /* class instance title tags mask isfloating isterminal noswallow monitor */ -+ { "Gimp", NULL, NULL, 0, 1, 0, 0, -1 }, -+ { "Firefox", NULL, NULL, 1 << 8, 0, 0, -1, -1 }, -+ { "St", NULL, NULL, 0, 0, 1, 0, -1 }, -+ { NULL, NULL, "Event Tester", 0, 0, 0, 1, -1 }, /* xev */ - }; - - /* layout(s) */ -diff --git a/config.mk b/config.mk -index 7084c33..ff9e508 100644 ---- a/config.mk -+++ b/config.mk -@@ -19,10 +19,11 @@ FREETYPELIBS = -lfontconfig -lXft - FREETYPEINC = /usr/include/freetype2 - # OpenBSD (uncomment) - #FREETYPEINC = ${X11INC}/freetype2 -+#KVMLIB = -lkvm - - # includes and libs - INCS = -I${X11INC} -I${FREETYPEINC} --LIBS = -L${X11LIB} -lX11 ${XINERAMALIBS} ${FREETYPELIBS} -+LIBS = -L${X11LIB} -lX11 ${XINERAMALIBS} ${FREETYPELIBS} -lX11-xcb -lxcb -lxcb-res ${KVMLIB} - - # flags - CPPFLAGS = -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_POSIX_C_SOURCE=200809L -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS} -diff --git a/dwm.c b/dwm.c -index 664c527..0b20086 100644 ---- a/dwm.c -+++ b/dwm.c -@@ -40,6 +40,12 @@ - #include - #endif /* XINERAMA */ - #include -+#include -+#include -+#ifdef __OpenBSD__ -+#include -+#include -+#endif /* __OpenBSD */ - - #include "drw.h" - #include "util.h" -@@ -92,9 +98,11 @@ struct Client { - int basew, baseh, incw, inch, maxw, maxh, minw, minh; - int bw, oldbw; - unsigned int tags; -- int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen; -+ int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen, isterminal, noswallow; -+ pid_t pid; - Client *next; - Client *snext; -+ Client *swallowing; - Monitor *mon; - Window win; - }; -@@ -138,6 +146,8 @@ typedef struct { - const char *title; - unsigned int tags; - int isfloating; -+ int isterminal; -+ int noswallow; - int monitor; - } Rule; - -@@ -235,6 +245,12 @@ static int xerrordummy(Display *dpy, XErrorEvent *ee); - static int xerrorstart(Display *dpy, XErrorEvent *ee); - static void zoom(const Arg *arg); - -+static pid_t getparentprocess(pid_t p); -+static int isdescprocess(pid_t p, pid_t c); -+static Client *swallowingclient(Window w); -+static Client *termforwin(const Client *c); -+static pid_t winpid(Window w); -+ - /* variables */ - static const char broken[] = "broken"; - static char stext[256]; -@@ -269,6 +285,8 @@ static Drw *drw; - static Monitor *mons, *selmon; - static Window root, wmcheckwin; - -+static xcb_connection_t *xcon; -+ - /* configuration, allows nested code to access above variables */ - #include "config.h" - -@@ -298,6 +316,8 @@ applyrules(Client *c) - && (!r->class || strstr(class, r->class)) - && (!r->instance || strstr(instance, r->instance))) - { -+ c->isterminal = r->isterminal; -+ c->noswallow = r->noswallow; - c->isfloating = r->isfloating; - c->tags |= r->tags; - for (m = mons; m && m->num != r->monitor; m = m->next); -@@ -414,6 +434,53 @@ attachstack(Client *c) - c->mon->stack = c; - } - -+void -+swallow(Client *p, Client *c) -+{ -+ -+ if (c->noswallow || c->isterminal) -+ return; -+ if (c->noswallow && !swallowfloating && c->isfloating) -+ return; -+ -+ detach(c); -+ detachstack(c); -+ -+ setclientstate(c, WithdrawnState); -+ XUnmapWindow(dpy, p->win); -+ -+ p->swallowing = c; -+ c->mon = p->mon; -+ -+ Window w = p->win; -+ p->win = c->win; -+ c->win = w; -+ updatetitle(p); -+ XMoveResizeWindow(dpy, p->win, p->x, p->y, p->w, p->h); -+ arrange(p->mon); -+ configure(p); -+ updateclientlist(); -+} -+ -+void -+unswallow(Client *c) -+{ -+ c->win = c->swallowing->win; -+ -+ free(c->swallowing); -+ c->swallowing = NULL; -+ -+ /* unfullscreen the client */ -+ setfullscreen(c, 0); -+ updatetitle(c); -+ arrange(c->mon); -+ XMapWindow(dpy, c->win); -+ XMoveResizeWindow(dpy, c->win, c->x, c->y, c->w, c->h); -+ setclientstate(c, NormalState); -+ focus(NULL); -+ arrange(c->mon); -+} -+ - void - buttonpress(XEvent *e) - { -@@ -653,6 +720,9 @@ destroynotify(XEvent *e) - - if ((c = wintoclient(ev->window))) - unmanage(c, 1); -+ -+ else if ((c = swallowingclient(ev->window))) -+ unmanage(c->swallowing, 1); - } - - void -@@ -1018,12 +1088,13 @@ killclient(const Arg *arg) - void - manage(Window w, XWindowAttributes *wa) - { -- Client *c, *t = NULL; -+ Client *c, *t = NULL, *term = NULL; - Window trans = None; - XWindowChanges wc; - - c = ecalloc(1, sizeof(Client)); - c->win = w; -+ c->pid = winpid(w); - /* geometry */ - c->x = c->oldx = wa->x; - c->y = c->oldy = wa->y; -@@ -1038,6 +1109,7 @@ manage(Window w, XWindowAttributes *wa) - } else { - c->mon = selmon; - applyrules(c); -+ term = termforwin(c); - } - - if (c->x + WIDTH(c) > c->mon->mx + c->mon->mw) -@@ -1074,6 +1146,8 @@ manage(Window w, XWindowAttributes *wa) - c->mon->sel = c; - arrange(c->mon); - XMapWindow(dpy, c->win); -+ if (term) -+ swallow(term, c); - focus(NULL); - } - -@@ -1768,6 +1842,20 @@ unmanage(Client *c, int destroyed) - Monitor *m = c->mon; - XWindowChanges wc; - -+ if (c->swallowing) { -+ unswallow(c); -+ return; -+ } -+ -+ Client *s = swallowingclient(c->win); -+ if (s) { -+ free(s->swallowing); -+ s->swallowing = NULL; -+ arrange(m); -+ focus(NULL); -+ return; -+ } -+ - detach(c); - detachstack(c); - if (!destroyed) { -@@ -1782,9 +1870,12 @@ unmanage(Client *c, int destroyed) - XUngrabServer(dpy); - } - free(c); -- focus(NULL); -- updateclientlist(); -- arrange(m); -+ -+ if (!s) { -+ arrange(m); -+ focus(NULL); -+ updateclientlist(); -+ } - } - - void -@@ -2047,6 +2138,136 @@ view(const Arg *arg) - arrange(selmon); - } - -+pid_t -+winpid(Window w) -+{ -+ -+ pid_t result = 0; -+ -+#ifdef __linux__ -+ xcb_res_client_id_spec_t spec = {0}; -+ spec.client = w; -+ spec.mask = XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID; -+ -+ xcb_generic_error_t *e = NULL; -+ xcb_res_query_client_ids_cookie_t c = xcb_res_query_client_ids(xcon, 1, &spec); -+ xcb_res_query_client_ids_reply_t *r = xcb_res_query_client_ids_reply(xcon, c, &e); -+ -+ if (!r) -+ return (pid_t)0; -+ -+ xcb_res_client_id_value_iterator_t i = xcb_res_query_client_ids_ids_iterator(r); -+ for (; i.rem; xcb_res_client_id_value_next(&i)) { -+ spec = i.data->spec; -+ if (spec.mask & XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID) { -+ uint32_t *t = xcb_res_client_id_value_value(i.data); -+ result = *t; -+ break; -+ } -+ } -+ -+ free(r); -+ -+ if (result == (pid_t)-1) -+ result = 0; -+ -+#endif /* __linux__ */ -+ -+#ifdef __OpenBSD__ -+ Atom type; -+ int format; -+ unsigned long len, bytes; -+ unsigned char *prop; -+ pid_t ret; -+ -+ if (XGetWindowProperty(dpy, w, XInternAtom(dpy, "_NET_WM_PID", 0), 0, 1, False, AnyPropertyType, &type, &format, &len, &bytes, &prop) != Success || !prop) -+ return 0; -+ -+ ret = *(pid_t*)prop; -+ XFree(prop); -+ result = ret; -+ -+#endif /* __OpenBSD__ */ -+ return result; -+} -+ -+pid_t -+getparentprocess(pid_t p) -+{ -+ unsigned int v = 0; -+ -+#ifdef __linux__ -+ FILE *f; -+ char buf[256]; -+ snprintf(buf, sizeof(buf) - 1, "/proc/%u/stat", (unsigned)p); -+ -+ if (!(f = fopen(buf, "r"))) -+ return 0; -+ -+ fscanf(f, "%*u %*s %*c %u", &v); -+ fclose(f); -+#endif /* __linux__*/ -+ -+#ifdef __OpenBSD__ -+ int n; -+ kvm_t *kd; -+ struct kinfo_proc *kp; -+ -+ kd = kvm_openfiles(NULL, NULL, NULL, KVM_NO_FILES, NULL); -+ if (!kd) -+ return 0; -+ -+ kp = kvm_getprocs(kd, KERN_PROC_PID, p, sizeof(*kp), &n); -+ v = kp->p_ppid; -+#endif /* __OpenBSD__ */ -+ -+ return (pid_t)v; -+} -+ -+int -+isdescprocess(pid_t p, pid_t c) -+{ -+ while (p != c && c != 0) -+ c = getparentprocess(c); -+ -+ return (int)c; -+} -+ -+Client * -+termforwin(const Client *w) -+{ -+ Client *c; -+ Monitor *m; -+ -+ if (!w->pid || w->isterminal) -+ return NULL; -+ -+ for (m = mons; m; m = m->next) { -+ for (c = m->clients; c; c = c->next) { -+ if (c->isterminal && !c->swallowing && c->pid && isdescprocess(c->pid, w->pid)) -+ return c; -+ } -+ } -+ -+ return NULL; -+} -+ -+Client * -+swallowingclient(Window w) -+{ -+ Client *c; -+ Monitor *m; -+ -+ for (m = mons; m; m = m->next) { -+ for (c = m->clients; c; c = c->next) { -+ if (c->swallowing && c->swallowing->win == w) -+ return c; -+ } -+ } -+ -+ return NULL; -+} -+ - Client * - wintoclient(Window w) - { -@@ -2138,10 +2359,12 @@ main(int argc, char *argv[]) - fputs("warning: no locale support\n", stderr); - if (!(dpy = XOpenDisplay(NULL))) - die("dwm: cannot open display"); -+ if (!(xcon = XGetXCBConnection(dpy))) -+ die("dwm: cannot get xcb connection\n"); - checkotherwm(); - setup(); - #ifdef __OpenBSD__ -- if (pledge("stdio rpath proc exec", NULL) == -1) -+ if (pledge("stdio rpath proc exec ps", NULL) == -1) - die("pledge"); - #endif /* __OpenBSD__ */ - scan(); --- -2.28.0 -