arg@327: /* arg@327: * (C)opyright MMVI Anselm R. Garbe arg@327: * See LICENSE file for license details. arg@327: */ arg@327: #include "dwm.h" arg@382: #include arg@327: arg@380: /* static */ arg@380: arg@382: static Client * arg@382: minclient() arg@382: { arg@382: Client *c, *min; arg@382: arg@443: if((clients && clients->isfloat) || arrange == dofloat) arg@443: return clients; /* don't touch floating order */ arg@382: for(min = c = clients; c; c = c->next) arg@382: if(c->weight < min->weight) arg@382: min = c; arg@382: return min; arg@382: } arg@382: arg@442: static void arg@381: reorder() arg@380: { arg@382: Client *c, *newclients, *tail; arg@380: arg@382: newclients = tail = NULL; arg@382: while((c = minclient())) { arg@381: detach(c); arg@382: if(tail) { arg@382: c->prev = tail; arg@382: tail->next = c; arg@382: tail = c; arg@381: } arg@381: else arg@382: tail = newclients = c; arg@380: } arg@382: clients = newclients; arg@380: } arg@380: arg@430: static Client * arg@430: nexttiled(Client *c) arg@430: { arg@433: for(c = getnext(c); c && c->isfloat; c = getnext(c->next)); arg@430: return c; arg@430: } arg@430: arg@327: /* extern */ arg@327: arg@327: void (*arrange)(Arg *) = DEFMODE; arg@327: arg@327: void arg@378: detach(Client *c) arg@378: { arg@378: if(c->prev) arg@378: c->prev->next = c->next; arg@378: if(c->next) arg@378: c->next->prev = c->prev; arg@378: if(c == clients) arg@378: clients = c->next; arg@378: c->next = c->prev = NULL; arg@378: } arg@378: arg@378: void arg@327: dofloat(Arg *arg) arg@327: { arg@402: Client *c; arg@400: arg@400: maximized = False; arg@327: arg@327: for(c = clients; c; c = c->next) { arg@327: if(isvisible(c)) { arg@327: resize(c, True, TopLeft); arg@327: } arg@327: else arg@327: ban(c); arg@327: } arg@446: if(!sel || !isvisible(sel)) { arg@450: for(c = stack; c && !isvisible(c); c = c->snext); arg@450: focus(c); arg@446: } arg@327: restack(); arg@327: } arg@327: arg@327: void arg@327: dotile(Arg *arg) arg@327: { arg@327: int h, i, n, w; arg@402: Client *c; arg@400: arg@400: maximized = False; arg@327: arg@327: w = sw - mw; arg@430: for(n = 0, c = clients; c; c = c->next) arg@430: if(isvisible(c) && !c->isfloat) arg@327: n++; arg@327: arg@327: if(n > 1) arg@327: h = (sh - bh) / (n - 1); arg@327: else arg@327: h = sh - bh; arg@327: arg@327: for(i = 0, c = clients; c; c = c->next) { arg@327: if(isvisible(c)) { arg@327: if(c->isfloat) { arg@327: resize(c, True, TopLeft); arg@327: continue; arg@327: } arg@327: if(n == 1) { arg@327: c->x = sx; arg@327: c->y = sy + bh; arg@327: c->w = sw - 2; arg@327: c->h = sh - 2 - bh; arg@327: } arg@327: else if(i == 0) { arg@327: c->x = sx; arg@327: c->y = sy + bh; arg@327: c->w = mw - 2; arg@327: c->h = sh - 2 - bh; arg@327: } arg@327: else if(h > bh) { arg@327: c->x = sx + mw; arg@327: c->y = sy + (i - 1) * h + bh; arg@327: c->w = w - 2; arg@327: if(i + 1 == n) arg@327: c->h = sh - c->y - 2; arg@327: else arg@327: c->h = h - 2; arg@327: } arg@327: else { /* fallback if h < bh */ arg@327: c->x = sx + mw; arg@327: c->y = sy + bh; arg@327: c->w = w - 2; arg@327: c->h = sh - 2 - bh; arg@327: } arg@327: resize(c, False, TopLeft); arg@327: i++; arg@327: } arg@327: else arg@327: ban(c); arg@327: } arg@446: if(!sel || !isvisible(sel)) { arg@450: for(c = stack; c && !isvisible(c); c = c->snext); arg@450: focus(c); arg@446: } arg@327: restack(); arg@327: } arg@327: arg@327: void arg@327: focusnext(Arg *arg) arg@327: { arg@327: Client *c; arg@327: arg@327: if(!sel) arg@327: return; arg@327: arg@327: if(!(c = getnext(sel->next))) arg@327: c = getnext(clients); arg@327: if(c) { arg@327: focus(c); arg@327: restack(); arg@327: } arg@327: } arg@327: arg@327: void arg@327: focusprev(Arg *arg) arg@327: { arg@327: Client *c; arg@327: arg@327: if(!sel) arg@327: return; arg@327: arg@327: if(!(c = getprev(sel->prev))) { arg@327: for(c = clients; c && c->next; c = c->next); arg@327: c = getprev(c); arg@327: } arg@327: if(c) { arg@327: focus(c); arg@327: restack(); arg@327: } arg@327: } arg@327: arg@420: Bool arg@420: isvisible(Client *c) arg@420: { arg@420: unsigned int i; arg@420: arg@420: for(i = 0; i < ntags; i++) arg@420: if(c->tags[i] && seltag[i]) arg@420: return True; arg@420: return False; arg@420: } arg@420: arg@415: void arg@422: resizecol(Arg *arg) arg@415: { arg@423: unsigned int n; arg@423: Client *c; arg@418: arg@430: for(n = 0, c = clients; c; c = c->next) arg@430: if(isvisible(c) && !c->isfloat) arg@423: n++; arg@423: if(!sel || sel->isfloat || n < 2 || (arrange != dotile) || maximized) arg@415: return; arg@423: arg@415: if(sel == getnext(clients)) { arg@425: if(mw + arg->i > sw - 100 || mw + arg->i < 100) arg@415: return; arg@415: mw += arg->i; arg@415: } arg@415: else { arg@425: if(mw - arg->i > sw - 100 || mw - arg->i < 100) arg@415: return; arg@415: mw -= arg->i; arg@415: } arg@415: arrange(NULL); arg@415: } arg@415: arg@327: void arg@327: restack() arg@327: { arg@327: Client *c; arg@327: XEvent ev; arg@436: arg@437: if(!sel) { arg@437: drawstatus(); arg@436: return; arg@437: } arg@436: if(sel->isfloat || arrange == dofloat) { arg@436: XRaiseWindow(dpy, sel->win); arg@436: XRaiseWindow(dpy, sel->twin); arg@436: } arg@446: if(arrange != dofloat) arg@436: for(c = nexttiled(clients); c; c = nexttiled(c->next)) { arg@436: XLowerWindow(dpy, c->twin); arg@436: XLowerWindow(dpy, c->win); arg@327: } arg@327: drawall(); arg@327: XSync(dpy, False); arg@327: while(XCheckMaskEvent(dpy, EnterWindowMask, &ev)); arg@327: } arg@327: arg@327: void arg@327: togglemode(Arg *arg) arg@327: { arg@333: arrange = (arrange == dofloat) ? dotile : dofloat; arg@327: if(sel) arg@327: arrange(NULL); arg@327: else arg@327: drawstatus(); arg@327: } arg@327: arg@327: void arg@327: toggleview(Arg *arg) arg@327: { arg@327: unsigned int i; arg@327: arg@327: seltag[arg->i] = !seltag[arg->i]; arg@327: for(i = 0; i < ntags && !seltag[i]; i++); arg@327: if(i == ntags) arg@327: seltag[arg->i] = True; /* cannot toggle last view */ arg@381: reorder(); arg@327: arrange(NULL); arg@327: } arg@327: arg@327: void arg@327: view(Arg *arg) arg@327: { arg@327: unsigned int i; arg@327: arg@327: for(i = 0; i < ntags; i++) arg@327: seltag[i] = False; arg@327: seltag[arg->i] = True; arg@381: reorder(); arg@327: arrange(NULL); arg@327: } arg@327: arg@327: void arg@395: viewall(Arg *arg) arg@395: { arg@395: unsigned int i; arg@395: arg@395: for(i = 0; i < ntags; i++) arg@395: seltag[i] = True; arg@397: reorder(); arg@395: arrange(NULL); arg@395: } arg@395: arg@395: void arg@327: zoom(Arg *arg) arg@327: { arg@423: unsigned int n; arg@423: Client *c; arg@327: arg@430: for(n = 0, c = clients; c; c = c->next) arg@430: if(isvisible(c) && !c->isfloat) arg@423: n++; arg@423: if(!sel || sel->isfloat || n < 2 || (arrange != dotile) || maximized) arg@327: return; arg@327: arg@430: if((c = sel) == nexttiled(clients)) arg@433: if(!(c = nexttiled(c->next))) arg@429: return; arg@443: detach(c); arg@443: if(clients) arg@443: clients->prev = c; arg@443: c->next = clients; arg@443: clients = c; arg@378: focus(c); arg@327: arrange(NULL); arg@327: }