feat: initial metaworkspace implementation
This commit is contained in:
parent
9724383228
commit
4607399dc8
2 changed files with 165 additions and 83 deletions
26
config.h
26
config.h
|
@ -1,5 +1,6 @@
|
|||
/* appearance */
|
||||
#include <X11/X.h>
|
||||
#include <X11/Xutil.h>
|
||||
static const unsigned int borderpx = 2;
|
||||
static const unsigned int snap = 0;
|
||||
static const unsigned int swallowfloating = 0;
|
||||
|
@ -34,6 +35,8 @@ static const unsigned int alphas[][3] = {
|
|||
|
||||
/* tagging */
|
||||
static const char *tags[] = { "1", "2", "3", "4", "5", "6", "7", "8", "9" };
|
||||
static const char *metaworkspaces[] = { "1", "2", "3", "4" };
|
||||
|
||||
static const Rule rules[] = {
|
||||
/* xprop(1):
|
||||
* WM_CLASS(STRING) = instance, class
|
||||
|
@ -86,13 +89,13 @@ static const char *termcmd[] = { "alacritty", NULL };
|
|||
static const Key keys[] = {
|
||||
/* modifier key function argument */
|
||||
// movement
|
||||
// { MODKEY|ShiftMask, XK_l, incnmaster, {.i = +1 } },
|
||||
// { MODKEY|ShiftMask, XK_h, incnmaster, {.i = -1 } },
|
||||
{ MODKEY, XK_d, spawn, {.v = dmenucmd } },
|
||||
{ MODKEY, XK_Return, spawn, {.v = termcmd } },
|
||||
{ MODKEY, XK_b, togglebar, {0} },
|
||||
{ MODKEY, XK_j, focusstack, {.i = +1 } },
|
||||
{ MODKEY, XK_k, focusstack, {.i = -1 } },
|
||||
// { MODKEY|ShiftMask, XK_l, incnmaster, {.i = +1 } },
|
||||
// { MODKEY|ShiftMask, XK_h, incnmaster, {.i = -1 } },
|
||||
{ MODKEY, XK_n, setmfact, {.f = -0.05} },
|
||||
{ MODKEY, XK_m, setmfact, {.f = +0.05} },
|
||||
{ MODKEY, XK_c, zoom, {0} },
|
||||
|
@ -112,6 +115,20 @@ static const Key keys[] = {
|
|||
{ MODKEY|ControlMask, XK_0, tag, {.ui = ~0 } },
|
||||
{ MODKEY, XK_p, invertdir, {0} },
|
||||
|
||||
// metaws
|
||||
{ MODKEY, XK_Page_Up, traversemetaws, {.i = +1} },
|
||||
{ MODKEY, XK_Page_Down, traversemetaws, {.i = -1} },
|
||||
{ MODKEY|Mod1Mask, XK_h, viewmetaws, {.ui = 0} },
|
||||
{ MODKEY|Mod1Mask, XK_j, viewmetaws, {.ui = 1} },
|
||||
{ MODKEY|Mod1Mask, XK_k, viewmetaws, {.ui = 2} },
|
||||
{ MODKEY|Mod1Mask, XK_l, viewmetaws, {.ui = 3} },
|
||||
|
||||
{ MODKEY|Mod1Mask|ControlMask, XK_h, tagmetaws, {.i = 0} },
|
||||
{ MODKEY|Mod1Mask|ControlMask, XK_j, tagmetaws, {.i = 1} },
|
||||
{ MODKEY|Mod1Mask|ControlMask, XK_k, tagmetaws, {.i = 2} },
|
||||
{ MODKEY|Mod1Mask|ControlMask, XK_l, tagmetaws, {.i = 3} },
|
||||
{ MODKEY|Mod1Mask|ControlMask, XK_m, tagmetaws, {.i = -1} },
|
||||
|
||||
// apps
|
||||
{ MODKEY, XK_w, spawn, SHCMD("firefox") },
|
||||
{ MODKEY|ShiftMask, XK_p, spawn, SHCMD("cast") },
|
||||
|
@ -146,7 +163,7 @@ static const Key keys[] = {
|
|||
};
|
||||
|
||||
/* button definitions */
|
||||
/* click can be ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle, ClkClientWin, or ClkRootWin */
|
||||
/* click can be ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle, ClkClientWin, ClkRootWin, or ClkMetaWorkspace */
|
||||
static const Button buttons[] = {
|
||||
/* click event mask button function argument */
|
||||
{ ClkClientWin, MODKEY, Button1, movemouse, {0} },
|
||||
|
@ -161,4 +178,7 @@ static const Button buttons[] = {
|
|||
{ ClkTagBar, ControlMask, Button1, tag, {0} },
|
||||
{ ClkTagBar, Mod1Mask, Button1, toggletag, {0} },
|
||||
{ ClkTagBar, ControlMask|ShiftMask, Button1, toggleview, {0} },
|
||||
|
||||
{ ClkMwSymbol, 0, Button1, traversemetaws, {.i = +1} },
|
||||
{ ClkMwSymbol, 0, Button3, traversemetaws, {.i = -1} },
|
||||
};
|
||||
|
|
102
dwm.c
102
dwm.c
|
@ -55,7 +55,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[metaws] & 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)
|
||||
|
@ -72,7 +72,7 @@ enum { NetSupported, NetWMName, NetWMState, NetWMCheck,
|
|||
NetWMWindowTypeDialog, NetClientList, NetLast }; /* EWMH atoms */
|
||||
enum { WMProtocols, WMDelete, WMState, WMTakeFocus, WMLast }; /* default atoms */
|
||||
enum { ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle,
|
||||
ClkClientWin, ClkRootWin, ClkLast }; /* clicks */
|
||||
ClkClientWin, ClkRootWin, ClkLast, ClkMwSymbol }; /* clicks */
|
||||
|
||||
typedef union {
|
||||
int i;
|
||||
|
@ -99,7 +99,7 @@ struct Client {
|
|||
int oldx, oldy, oldw, oldh;
|
||||
int basew, baseh, incw, inch, maxw, maxh, minw, minh, hintsvalid;
|
||||
int bw, oldbw;
|
||||
unsigned int tags;
|
||||
unsigned int tags[32];
|
||||
int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen, isterminal, noswallow;
|
||||
pid_t pid;
|
||||
Client *next;
|
||||
|
@ -232,6 +232,7 @@ static void sigterm(int unused);
|
|||
static int solitary(Client *c);
|
||||
static void spawn(const Arg *arg);
|
||||
static void tag(const Arg *arg);
|
||||
static void tagmetaws(const Arg *arg);
|
||||
static void tagview(const Arg *arg);
|
||||
static void tagmon(const Arg *arg);
|
||||
static void togglebar(const Arg *arg);
|
||||
|
@ -239,6 +240,7 @@ static void togglefloating(const Arg *arg);
|
|||
static void togglefullscr(const Arg *arg);
|
||||
static void toggletag(const Arg *arg);
|
||||
static void toggleview(const Arg *arg);
|
||||
static void traversemetaws(const Arg *arg);
|
||||
static void unfocus(Client *c, int setfocus);
|
||||
static void unmanage(Client *c, int destroyed);
|
||||
static void unmapnotify(XEvent *e);
|
||||
|
@ -253,6 +255,7 @@ static void updatetitle(Client *c);
|
|||
static void updatewindowtype(Client *c);
|
||||
static void updatewmhints(Client *c);
|
||||
static void view(const Arg *arg);
|
||||
static void viewmetaws(const Arg *arg);
|
||||
static Client *wintoclient(Window w);
|
||||
static Monitor *wintomon(Window w);
|
||||
static int xerror(Display *dpy, XErrorEvent *ee);
|
||||
|
@ -306,6 +309,8 @@ static int useargb = 0;
|
|||
static Visual *visual;
|
||||
static int depth;
|
||||
static Colormap cmap;
|
||||
static int metaws = 0;
|
||||
static char mwsymbol[128] = {0};
|
||||
|
||||
static xcb_connection_t *xcon;
|
||||
|
||||
|
@ -327,7 +332,7 @@ applyrules(Client *c)
|
|||
|
||||
/* rule matching */
|
||||
c->isfloating = 0;
|
||||
c->tags = 0;
|
||||
c->tags[metaws] = 0;
|
||||
XGetClassHint(dpy, c->win, &ch);
|
||||
class = ch.res_class ? ch.res_class : broken;
|
||||
instance = ch.res_name ? ch.res_name : broken;
|
||||
|
@ -341,7 +346,7 @@ applyrules(Client *c)
|
|||
c->isterminal = r->isterminal;
|
||||
c->noswallow = r->noswallow;
|
||||
c->isfloating = r->isfloating;
|
||||
c->tags |= r->tags;
|
||||
c->tags[metaws] |= r->tags;
|
||||
for (m = mons; m && m->num != r->monitor; m = m->next);
|
||||
if (m)
|
||||
c->mon = m;
|
||||
|
@ -351,7 +356,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[metaws] = c->tags[metaws] & TAGMASK ? c->tags[metaws] & TAGMASK : c->mon->tagset[c->mon->seltags];
|
||||
}
|
||||
|
||||
int
|
||||
|
@ -534,11 +539,16 @@ buttonpress(XEvent *e)
|
|||
focus(NULL);
|
||||
}
|
||||
if (ev->window == selmon->barwin) {
|
||||
i = x = 0;
|
||||
i = 0;
|
||||
x = TEXTW(mwsymbol);
|
||||
|
||||
do
|
||||
x += TEXTW(tags[i]);
|
||||
while (ev->x >= x && ++i < LENGTH(tags));
|
||||
if (i < LENGTH(tags)) {
|
||||
|
||||
if (ev->x < TEXTW(mwsymbol))
|
||||
click = ClkMwSymbol;
|
||||
else if (i < LENGTH(tags)) {
|
||||
click = ClkTagBar;
|
||||
arg.ui = 1 << i;
|
||||
} else if (ev->x < x + TEXTW(selmon->ltsymbol))
|
||||
|
@ -808,7 +818,7 @@ dirtomon(int dir)
|
|||
void
|
||||
drawbar(Monitor *m)
|
||||
{
|
||||
int x, w, tw = 0;
|
||||
int w, x = 0, tw = 0;
|
||||
int boxs = drw->fonts->h / 9;
|
||||
int boxw = drw->fonts->h / 6 + 2;
|
||||
unsigned int i, occ = 0, urg = 0;
|
||||
|
@ -818,32 +828,38 @@ drawbar(Monitor *m)
|
|||
return;
|
||||
|
||||
/* 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]);
|
||||
if (m == selmon) { /* status is only drawn on selected monitor */
|
||||
tw = TEXTW(stext) - lrpad + 2; /* 2px right padding */
|
||||
drw_text(drw, m->ww - tw, 0, tw, bh, 0, stext, 0);
|
||||
}
|
||||
|
||||
for (c = m->clients; c; c = c->next) {
|
||||
occ |= c->tags;
|
||||
occ |= c->tags[metaws];
|
||||
if (c->isurgent)
|
||||
urg |= c->tags;
|
||||
urg |= c->tags[metaws];
|
||||
}
|
||||
x = 0;
|
||||
|
||||
w = TEXTW(mwsymbol);
|
||||
x = drw_text(drw, x, 0, w, bh, lrpad / 2, mwsymbol, 0);
|
||||
|
||||
for (i = 0; i < LENGTH(tags); i++) {
|
||||
w = TEXTW(tags[i]);
|
||||
drw_setscheme(drw, scheme[m->tagset[m->seltags] & 1 << i ? SchemeSel : SchemeNorm]);
|
||||
drw_text(drw, x, 0, w, bh, lrpad / 2, tags[i], urg & 1 << i);
|
||||
if (occ & 1 << i)
|
||||
drw_rect(drw, x + boxs, boxs, boxw, boxw,
|
||||
m == selmon && selmon->sel && selmon->sel->tags & 1 << i,
|
||||
m == selmon && selmon->sel && selmon->sel->tags[metaws] & 1 << i,
|
||||
urg & 1 << i);
|
||||
x += w;
|
||||
}
|
||||
|
||||
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 - tw - x) > bh) {
|
||||
if (m->sel) {
|
||||
drw_setscheme(drw, scheme[m == selmon ? SchemeSel : SchemeNorm]);
|
||||
|
@ -1187,7 +1203,7 @@ manage(Window w, XWindowAttributes *wa)
|
|||
updatetitle(c);
|
||||
if (XGetTransientForHint(dpy, w, &trans) && (t = wintoclient(trans))) {
|
||||
c->mon = t->mon;
|
||||
c->tags = t->tags;
|
||||
c->tags[metaws] = t->tags[metaws];
|
||||
} else {
|
||||
c->mon = selmon;
|
||||
applyrules(c);
|
||||
|
@ -1271,7 +1287,7 @@ monocle(Monitor *m)
|
|||
|
||||
for (c = m->clients; c; c = c->next)
|
||||
if (ISVISIBLE(c))
|
||||
n++;
|
||||
++n;
|
||||
if (n > 0) /* override layout symbol */
|
||||
snprintf(m->ltsymbol, sizeof m->ltsymbol, "[%d]", n);
|
||||
for (c = nexttiled(m->clients); c; c = nexttiled(c->next))
|
||||
|
@ -1591,7 +1607,7 @@ sendmon(Client *c, Monitor *m)
|
|||
Client *pc = selmon->stack;
|
||||
|
||||
c->mon = m;
|
||||
c->tags = m->tagset[m->seltags]; /* assign tags of target monitor */
|
||||
c->tags[metaws] = m->tagset[m->seltags]; /* assign tags of target monitor */
|
||||
|
||||
if (m->spawnmaster)
|
||||
attach(c);
|
||||
|
@ -1783,6 +1799,7 @@ setup(void)
|
|||
scheme = ecalloc(LENGTH(colors), sizeof(Clr *));
|
||||
for (i = 0; i < LENGTH(colors); i++)
|
||||
scheme[i] = drw_scm_create(drw, colors[i], alphas[i], 3);
|
||||
snprintf(mwsymbol, LENGTH(mwsymbol), "[%s]", metaworkspaces[metaws]);
|
||||
/* init bars */
|
||||
updatebars();
|
||||
updatestatus();
|
||||
|
@ -1890,12 +1907,34 @@ void
|
|||
tag(const Arg *arg)
|
||||
{
|
||||
if (selmon->sel && arg->ui & TAGMASK) {
|
||||
selmon->sel->tags = arg->ui & TAGMASK;
|
||||
selmon->sel->tags[metaws] = arg->ui & TAGMASK;
|
||||
focus(NULL);
|
||||
arrange(selmon);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
tagmetaws(const Arg *arg)
|
||||
{
|
||||
int i;
|
||||
unsigned int ot;
|
||||
|
||||
if (!selmon->sel || arg->i >= (int)LENGTH(metaworkspaces))
|
||||
return;
|
||||
|
||||
if (arg->i == -1)
|
||||
for (i = 0; i < LENGTH(metaworkspaces); ++i)
|
||||
selmon->sel->tags[i] = selmon->sel->tags[metaws];
|
||||
else {
|
||||
ot = selmon->sel->tags[metaws];
|
||||
for (i = 0; i < LENGTH(metaworkspaces); ++i)
|
||||
selmon->sel->tags[i] = 0;
|
||||
selmon->sel->tags[arg->i] = ot;
|
||||
}
|
||||
focus(NULL);
|
||||
arrange(selmon);
|
||||
}
|
||||
|
||||
void
|
||||
tagmon(const Arg *arg)
|
||||
{
|
||||
|
@ -1963,9 +2002,9 @@ toggletag(const Arg *arg)
|
|||
|
||||
if (!selmon->sel)
|
||||
return;
|
||||
newtags = selmon->sel->tags ^ (arg->ui & TAGMASK);
|
||||
newtags = selmon->sel->tags[metaws] ^ (arg->ui & TAGMASK);
|
||||
if (newtags) {
|
||||
selmon->sel->tags = newtags;
|
||||
selmon->sel->tags[metaws] = newtags;
|
||||
focus(NULL);
|
||||
arrange(selmon);
|
||||
}
|
||||
|
@ -1983,6 +2022,17 @@ toggleview(const Arg *arg)
|
|||
}
|
||||
}
|
||||
|
||||
void
|
||||
traversemetaws(const Arg *arg)
|
||||
{
|
||||
if (!arg->i) return;
|
||||
metaws = (metaws + arg->i + (LENGTH(metaworkspaces))) % LENGTH(metaworkspaces);
|
||||
snprintf(mwsymbol, LENGTH(mwsymbol), "[%s]", metaworkspaces[metaws]);
|
||||
focus(NULL);
|
||||
arrange(NULL);
|
||||
drawbars();
|
||||
}
|
||||
|
||||
void
|
||||
unfocus(Client *c, int setfocus)
|
||||
{
|
||||
|
@ -2307,6 +2357,18 @@ view(const Arg *arg)
|
|||
arrange(selmon);
|
||||
}
|
||||
|
||||
void
|
||||
viewmetaws(const Arg *arg)
|
||||
{
|
||||
if (arg->ui < LENGTH(metaworkspaces)) {
|
||||
metaws = arg->ui;
|
||||
snprintf(mwsymbol, LENGTH(mwsymbol), "[%s]", metaworkspaces[metaws]);
|
||||
focus(NULL);
|
||||
arrange(NULL);
|
||||
drawbars();
|
||||
}
|
||||
}
|
||||
|
||||
pid_t
|
||||
winpid(Window w)
|
||||
{
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue