mirror of
https://github.com/mintycube/dotfiles.git
synced 2024-10-22 14:05:41 +02:00
315 lines
7.0 KiB
C
315 lines
7.0 KiB
C
void
|
|
persistmonitorstate(Monitor *m)
|
|
{
|
|
Client *c;
|
|
unsigned int i;
|
|
|
|
setmonitortags(m);
|
|
setmonitorfields(m);
|
|
|
|
/* Set client atoms */
|
|
for (i = 1, c = m->clients; c; c = c->next, ++i) {
|
|
c->idx = i;
|
|
persistclientstate(c);
|
|
}
|
|
}
|
|
|
|
int
|
|
restoremonitorstate(Monitor *m)
|
|
{
|
|
return getmonitortags(m) | getmonitorfields(m);
|
|
}
|
|
|
|
void
|
|
persistclientstate(Client *c)
|
|
{
|
|
setclienttags(c);
|
|
setclientfields(c);
|
|
savewindowfloatposition(c, c->mon);
|
|
}
|
|
|
|
int
|
|
restoreclientstate(Client *c)
|
|
{
|
|
int restored = getclientfields(c);
|
|
getclienttags(c);
|
|
restorewindowfloatposition(c, c->mon ? c->mon : selmon);
|
|
return restored;
|
|
}
|
|
|
|
void setmonitorfields(Monitor *m)
|
|
{
|
|
unsigned int i;
|
|
char atom[22] = {0};
|
|
Atom monitor_fields;
|
|
|
|
sprintf(atom, "_DWM_MONITOR_FIELDS_%u", m->num);
|
|
monitor_fields = XInternAtom(dpy, atom, False);
|
|
|
|
/* Perists workspace information in 32 bits laid out like this:
|
|
*
|
|
* |0|0000|0|0000|0000|0000|0000|0000|000|000
|
|
* | | | | | | | | | |-- nmaster
|
|
* | | | | | | | | |-- nstack
|
|
* | | | | | | | |-- layout
|
|
* | | | | | | |-- flextile LAYOUT (split)
|
|
* | | | | | |-- flextile MASTER
|
|
* | | | | |-- flextile STACK1
|
|
* | | | |-- flextile STACK2
|
|
* | | |-- flextile mirror layout (indicated by negative layout)
|
|
* | |
|
|
* | |-- reserved
|
|
* |-- showbar
|
|
*/
|
|
for (i = 0; i <= NUMTAGS; i++) {
|
|
uint32_t data[] = {
|
|
(m->pertag->nmasters[i] & 0x7) |
|
|
(getlayoutindex(m->pertag->ltidxs[i][m->pertag->sellts[i]]) & 0xF) << 6 |
|
|
m->showbar << 31
|
|
};
|
|
|
|
XChangeProperty(dpy, root, monitor_fields, XA_CARDINAL, 32,
|
|
i ? PropModeAppend : PropModeReplace, (unsigned char *)data, 1);
|
|
}
|
|
}
|
|
|
|
int
|
|
getlayoutindex(const Layout *layout)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < LENGTH(layouts) && &layouts[i] != layout; i++);
|
|
if (i == LENGTH(layouts))
|
|
i = 0;
|
|
return i;
|
|
}
|
|
|
|
int
|
|
getmonitorfields(Monitor *m)
|
|
{
|
|
int di, layout_index;
|
|
unsigned int i, restored = 0;
|
|
unsigned int tags = m->tagset[m->seltags] << 1;
|
|
unsigned long dl, nitems;
|
|
unsigned char *p = NULL;
|
|
char atom[22] = {0};
|
|
Atom da, state = None;
|
|
|
|
sprintf(atom, "_DWM_MONITOR_FIELDS_%u", m->num);
|
|
Atom dwm_monitor = XInternAtom(dpy, atom, False);
|
|
if (!dwm_monitor)
|
|
return 0;
|
|
|
|
for (i = 0; i <= NUMTAGS; i++) {
|
|
if (!(XGetWindowProperty(dpy, root, dwm_monitor, i, (NUMTAGS + 1) * sizeof dl,
|
|
False, AnyPropertyType, &da, &di, &nitems, &dl, &p) == Success && p)) {
|
|
break;
|
|
}
|
|
|
|
if (!nitems) {
|
|
XFree(p);
|
|
break;
|
|
}
|
|
|
|
/* See bit layout in the persistmonitorstate function */
|
|
state = *(Atom *)p;
|
|
|
|
m->pertag->nmasters[i] = state & 0x7;
|
|
layout_index = (state >> 6) & 0xF;
|
|
if (layout_index < LENGTH(layouts))
|
|
m->pertag->ltidxs[i][m->pertag->sellts[i]] = &layouts[layout_index];
|
|
|
|
if (!restored && i && (tags & (1 << i))) {
|
|
m->nmaster = m->pertag->nmasters[i];
|
|
m->sellt = m->pertag->sellts[i];
|
|
m->lt[m->sellt] = m->pertag->ltidxs[i][m->sellt];
|
|
m->showbar = (state >> 31) & 0x1;
|
|
restored = 1;
|
|
}
|
|
|
|
XFree(p);
|
|
}
|
|
|
|
return restored;
|
|
}
|
|
|
|
void
|
|
setmonitortags(Monitor *m)
|
|
{
|
|
char atom[22] = {0};
|
|
Atom monitor_tags;
|
|
|
|
sprintf(atom, "_DWM_MONITOR_TAGS_%u", m->num);
|
|
monitor_tags = XInternAtom(dpy, atom, False);
|
|
|
|
uint32_t data[] = { m->tagset[m->seltags] };
|
|
XChangeProperty(dpy, root, monitor_tags, XA_CARDINAL, 32, PropModeReplace, (unsigned char *)data, 1);
|
|
}
|
|
|
|
int
|
|
getmonitortags(Monitor *m)
|
|
{
|
|
int di;
|
|
unsigned long dl, nitems;
|
|
unsigned char *p = NULL;
|
|
char atom[22] = {0};
|
|
Atom da, monitor_tags = None, tags;
|
|
|
|
sprintf(atom, "_DWM_MONITOR_TAGS_%u", m->num);
|
|
monitor_tags = XInternAtom(dpy, atom, False);
|
|
|
|
if (!(XGetWindowProperty(dpy, root, monitor_tags, 0L, sizeof dl,
|
|
False, AnyPropertyType, &da, &di, &nitems, &dl, &p) == Success && p)) {
|
|
return 0;
|
|
}
|
|
|
|
if (nitems) {
|
|
tags = *(Atom *)p;
|
|
m->tagset[m->seltags] = tags & TAGMASK;
|
|
}
|
|
|
|
XFree(p);
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
setclientfields(Client *c)
|
|
{
|
|
/* Perists client information in 32 bits laid out like this:
|
|
*
|
|
* |00000000|00000|0|0|0|0|0|0|0|0|00000000|000
|
|
* | | | | | | | | | | | |-- monitor index
|
|
* | | | | | | | | | | |-- client index
|
|
* | | | | | | | | | |-- isfloating
|
|
* | | | | | | | | |-- ispermanent
|
|
* | | | | | | | |-- isterminal
|
|
* | | | | | | |-- noswallow
|
|
* | | | | | |-- issteam
|
|
* | | | | |-- issticky
|
|
* | | | |-- fakefullscreen
|
|
* | | |-- isfreesize
|
|
* | |
|
|
* | |-- reserved
|
|
* |-- scratchkey (for scratchpads)
|
|
*/
|
|
uint32_t data[] = {
|
|
(c->mon->num & 0x7)
|
|
| (c->idx & 0xFF) << 3
|
|
| (c->isfloating & 0x1) << 11
|
|
| (c->fakefullscreen & 0x1) << 17
|
|
};
|
|
XChangeProperty(dpy, c->win, clientatom[ClientFields], XA_CARDINAL, 32, PropModeReplace, (unsigned char *)data, 1);
|
|
}
|
|
|
|
int
|
|
getclientfields(Client *c)
|
|
{
|
|
Monitor *m;
|
|
Atom fields = getatomprop(c, clientatom[ClientFields], AnyPropertyType);
|
|
if (fields == None)
|
|
return 0;
|
|
|
|
/* See bit layout in the setclientfields function */
|
|
for (m = mons; m; m = m->next)
|
|
if (m->num == (fields & 0x7)) {
|
|
c->mon = m;
|
|
break;
|
|
}
|
|
c->idx = (fields >> 3) & 0xFF;
|
|
c->isfloating = (fields >> 11) & 0x1;
|
|
c->fakefullscreen = (fields >> 17) & 0x1;
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
setclienttags(Client *c)
|
|
{
|
|
uint32_t data[] = { c->tags };
|
|
XChangeProperty(dpy, c->win, clientatom[ClientTags], XA_CARDINAL, 32, PropModeReplace, (unsigned char *)data, 1);
|
|
}
|
|
|
|
int
|
|
getclienttags(Client *c)
|
|
{
|
|
Atom tags = getatomprop(c, clientatom[ClientTags], AnyPropertyType);
|
|
if (tags == None)
|
|
return 0;
|
|
|
|
c->tags = tags & TAGMASK;
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
savewindowfloatposition(Client *c, Monitor *m)
|
|
{
|
|
char atom[22] = {0};
|
|
if (c->sfx == -9999)
|
|
return;
|
|
|
|
sprintf(atom, "_DWM_FLOATPOS_%u", m->num);
|
|
uint32_t pos[] = { (MAX(c->sfx - m->mx, 0) & 0xffff) | ((MAX(c->sfy - m->my, 0) & 0xffff) << 16) };
|
|
XChangeProperty(dpy, c->win, XInternAtom(dpy, atom, False), XA_CARDINAL, 32, PropModeReplace, (unsigned char *)pos, 1);
|
|
|
|
sprintf(atom, "_DWM_FLOATSIZE_%u", m->num);
|
|
uint32_t size[] = { (c->sfw & 0xffff) | ((c->sfh & 0xffff) << 16) };
|
|
XChangeProperty(dpy, c->win, XInternAtom(dpy, atom, False), XA_CARDINAL, 32, PropModeReplace, (unsigned char *)size, 1);
|
|
|
|
XSync(dpy, False);
|
|
}
|
|
|
|
int
|
|
restorewindowfloatposition(Client *c, Monitor *m)
|
|
{
|
|
char atom[22] = {0};
|
|
Atom key, value;
|
|
int x, y, w, h;
|
|
|
|
if (m == NULL)
|
|
return 0;
|
|
|
|
sprintf(atom, "_DWM_FLOATPOS_%u", m->num);
|
|
|
|
key = XInternAtom(dpy, atom, False);
|
|
if (!key)
|
|
return 0;
|
|
|
|
value = getatomprop(c, key, AnyPropertyType);
|
|
if (!value)
|
|
return 0;
|
|
|
|
x = value & 0xffff;
|
|
y = value >> 16;
|
|
|
|
sprintf(atom, "_DWM_FLOATSIZE_%u", m->num);
|
|
|
|
key = XInternAtom(dpy, atom, False);
|
|
if (!key)
|
|
return 0;
|
|
|
|
value = getatomprop(c, key, AnyPropertyType);
|
|
if (!value)
|
|
return 0;
|
|
|
|
w = value & 0xffff;
|
|
h = value >> 16;
|
|
|
|
if (w <= 0 || h <= 0) {
|
|
fprintf(stderr, "restorewindowfloatposition: bad float values x = %d, y = %d, w = %d, h = %d for client = %s\n", x, y, w, h, c->name);
|
|
return 0;
|
|
}
|
|
|
|
c->sfx = m->mx + x;
|
|
c->sfy = m->my + y;
|
|
c->sfw = w;
|
|
c->sfh = h;
|
|
|
|
if (c->isfloating) {
|
|
c->x = c->sfx;
|
|
c->y = c->sfy;
|
|
c->w = c->sfw;
|
|
c->h = c->sfh;
|
|
}
|
|
|
|
return 1;
|
|
}
|