Compare commits
8 Commits
patch-ipc
...
0aada8a6b5
Author | SHA1 | Date | |
---|---|---|---|
0aada8a6b5 | |||
393b78fad0 | |||
5abeb2ff71 | |||
89f4f0fd7e | |||
84fa4f1113 | |||
062a7f88f8 | |||
7b100d9bf0 | |||
60674bdb3f |
66
IPCClient.c
66
IPCClient.c
@@ -1,66 +0,0 @@
|
|||||||
#include "IPCClient.h"
|
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
#include <sys/epoll.h>
|
|
||||||
|
|
||||||
#include "util.h"
|
|
||||||
|
|
||||||
IPCClient *
|
|
||||||
ipc_client_new(int fd)
|
|
||||||
{
|
|
||||||
IPCClient *c = (IPCClient *)malloc(sizeof(IPCClient));
|
|
||||||
|
|
||||||
if (c == NULL) return NULL;
|
|
||||||
|
|
||||||
// Initialize struct
|
|
||||||
memset(&c->event, 0, sizeof(struct epoll_event));
|
|
||||||
|
|
||||||
c->buffer_size = 0;
|
|
||||||
c->buffer = NULL;
|
|
||||||
c->fd = fd;
|
|
||||||
c->event.data.fd = fd;
|
|
||||||
c->next = NULL;
|
|
||||||
c->prev = NULL;
|
|
||||||
c->subscriptions = 0;
|
|
||||||
|
|
||||||
return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
ipc_list_add_client(IPCClientList *list, IPCClient *nc)
|
|
||||||
{
|
|
||||||
DEBUG("Adding client with fd %d to list\n", nc->fd);
|
|
||||||
|
|
||||||
if (*list == NULL) {
|
|
||||||
// List is empty, point list at first client
|
|
||||||
*list = nc;
|
|
||||||
} else {
|
|
||||||
IPCClient *c;
|
|
||||||
// Go to last client in list
|
|
||||||
for (c = *list; c && c->next; c = c->next)
|
|
||||||
;
|
|
||||||
c->next = nc;
|
|
||||||
nc->prev = c;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
ipc_list_remove_client(IPCClientList *list, IPCClient *c)
|
|
||||||
{
|
|
||||||
IPCClient *cprev = c->prev;
|
|
||||||
IPCClient *cnext = c->next;
|
|
||||||
|
|
||||||
if (cprev != NULL) cprev->next = c->next;
|
|
||||||
if (cnext != NULL) cnext->prev = c->prev;
|
|
||||||
if (c == *list) *list = c->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
IPCClient *
|
|
||||||
ipc_list_get_client(IPCClientList list, int fd)
|
|
||||||
{
|
|
||||||
for (IPCClient *c = list; c; c = c->next) {
|
|
||||||
if (c->fd == fd) return c;
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
61
IPCClient.h
61
IPCClient.h
@@ -1,61 +0,0 @@
|
|||||||
#ifndef IPC_CLIENT_H_
|
|
||||||
#define IPC_CLIENT_H_
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <sys/epoll.h>
|
|
||||||
|
|
||||||
typedef struct IPCClient IPCClient;
|
|
||||||
/**
|
|
||||||
* This structure contains the details of an IPC Client and pointers for a
|
|
||||||
* linked list
|
|
||||||
*/
|
|
||||||
struct IPCClient {
|
|
||||||
int fd;
|
|
||||||
int subscriptions;
|
|
||||||
|
|
||||||
char *buffer;
|
|
||||||
uint32_t buffer_size;
|
|
||||||
|
|
||||||
struct epoll_event event;
|
|
||||||
IPCClient *next;
|
|
||||||
IPCClient *prev;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef IPCClient *IPCClientList;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Allocate memory for new IPCClient with the specified file descriptor and
|
|
||||||
* initialize struct.
|
|
||||||
*
|
|
||||||
* @param fd File descriptor of IPC client
|
|
||||||
*
|
|
||||||
* @return Address to allocated IPCClient struct
|
|
||||||
*/
|
|
||||||
IPCClient *ipc_client_new(int fd);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add an IPC Client to the specified list
|
|
||||||
*
|
|
||||||
* @param list Address of the list to add the client to
|
|
||||||
* @param nc Address of the IPCClient
|
|
||||||
*/
|
|
||||||
void ipc_list_add_client(IPCClientList *list, IPCClient *nc);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Remove an IPCClient from the specified list
|
|
||||||
*
|
|
||||||
* @param list Address of the list to remove the client from
|
|
||||||
* @param c Address of the IPCClient
|
|
||||||
*/
|
|
||||||
void ipc_list_remove_client(IPCClientList *list, IPCClient *c);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get an IPCClient from the specified IPCClient list
|
|
||||||
*
|
|
||||||
* @param list List to remove the client from
|
|
||||||
* @param fd File descriptor of the IPCClient
|
|
||||||
*/
|
|
||||||
IPCClient *ipc_list_get_client(IPCClientList list, int fd);
|
|
||||||
|
|
||||||
#endif // IPC_CLIENT_H_
|
|
10
Makefile
10
Makefile
@@ -6,7 +6,7 @@ include config.mk
|
|||||||
SRC = drw.c dwm.c util.c
|
SRC = drw.c dwm.c util.c
|
||||||
OBJ = ${SRC:.c=.o}
|
OBJ = ${SRC:.c=.o}
|
||||||
|
|
||||||
all: options dwm dwm-msg
|
all: options dwm
|
||||||
|
|
||||||
options:
|
options:
|
||||||
@echo dwm build options:
|
@echo dwm build options:
|
||||||
@@ -25,11 +25,8 @@ config.h:
|
|||||||
dwm: ${OBJ}
|
dwm: ${OBJ}
|
||||||
${CC} -o $@ ${OBJ} ${LDFLAGS}
|
${CC} -o $@ ${OBJ} ${LDFLAGS}
|
||||||
|
|
||||||
dwm-msg: dwm-msg.o
|
|
||||||
${CC} -o $@ $< ${LDFLAGS}
|
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
rm -f dwm dwm-msg ${OBJ} dwm-${VERSION}.tar.gz
|
rm -f dwm ${OBJ} dwm-${VERSION}.tar.gz
|
||||||
|
|
||||||
dist: clean
|
dist: clean
|
||||||
mkdir -p dwm-${VERSION}
|
mkdir -p dwm-${VERSION}
|
||||||
@@ -41,9 +38,8 @@ dist: clean
|
|||||||
|
|
||||||
install: all
|
install: all
|
||||||
mkdir -p ${DESTDIR}${PREFIX}/bin
|
mkdir -p ${DESTDIR}${PREFIX}/bin
|
||||||
cp -f dwm dwm-msg ${DESTDIR}${PREFIX}/bin
|
cp -f dwm ${DESTDIR}${PREFIX}/bin
|
||||||
chmod 755 ${DESTDIR}${PREFIX}/bin/dwm
|
chmod 755 ${DESTDIR}${PREFIX}/bin/dwm
|
||||||
chmod 755 ${DESTDIR}${PREFIX}/bin/dwm-msg
|
|
||||||
mkdir -p ${DESTDIR}${MANPREFIX}/man1
|
mkdir -p ${DESTDIR}${MANPREFIX}/man1
|
||||||
sed "s/VERSION/${VERSION}/g" < dwm.1 > ${DESTDIR}${MANPREFIX}/man1/dwm.1
|
sed "s/VERSION/${VERSION}/g" < dwm.1 > ${DESTDIR}${MANPREFIX}/man1/dwm.1
|
||||||
chmod 644 ${DESTDIR}${MANPREFIX}/man1/dwm.1
|
chmod 644 ${DESTDIR}${MANPREFIX}/man1/dwm.1
|
||||||
|
45
config.def.h
45
config.def.h
@@ -3,15 +3,29 @@
|
|||||||
/* appearance */
|
/* appearance */
|
||||||
static const unsigned int borderpx = 1; /* border pixel of windows */
|
static const unsigned int borderpx = 1; /* border pixel of windows */
|
||||||
static const unsigned int snap = 32; /* snap pixel */
|
static const unsigned int snap = 32; /* snap pixel */
|
||||||
|
<<<<<<< HEAD
|
||||||
static const int showbar = 1; /* 0 means no bar */
|
static const int showbar = 1; /* 0 means no bar */
|
||||||
static const int topbar = 1; /* 0 means bottom bar */
|
static const int topbar = 1; /* 0 means bottom bar */
|
||||||
|
static const int usealtbar = 1; /* 1 means use non-dwm status bar */
|
||||||
|
static const char *altbarclass = "Polybar"; /* Alternate bar class name */
|
||||||
|
static const char *alttrayname = "tray"; /* Polybar tray instance name */
|
||||||
|
static const char *altbarcmd = "$HOME/bar.sh"; /* Alternate bar launch command */
|
||||||
|
=======
|
||||||
|
static const unsigned int systraypinning = 0; /* 0: sloppy systray follows selected monitor, >0: pin systray to monitor X */
|
||||||
|
static const unsigned int systrayonleft = 0; /* 0: systray in the right corner, >0: systray on left of status text */
|
||||||
|
static const unsigned int systrayspacing = 2; /* systray spacing */
|
||||||
|
static const int systraypinningfailfirst = 1; /* 1: if pinning fails, display systray on the first monitor, False: display systray on the last monitor*/
|
||||||
|
static const int showsystray = 1; /* 0 means no systray */
|
||||||
|
static const int showbar = 1; /* 0 means no bar */
|
||||||
|
static const int topbar = 1; /* 0 means bottom bar */
|
||||||
|
>>>>>>> patch-systray
|
||||||
static const char *fonts[] = { "monospace:size=10" };
|
static const char *fonts[] = { "monospace:size=10" };
|
||||||
static const char dmenufont[] = "monospace:size=10";
|
static const char dmenufont[] = "monospace:size=10";
|
||||||
static const char col_gray1[] = "#222222";
|
static const char col_gray1[] = "#222222";
|
||||||
static const char col_gray2[] = "#444444";
|
static const char col_gray2[] = "#444444";
|
||||||
static const char col_gray3[] = "#bbbbbb";
|
static const char col_gray3[] = "#bbbbbb";
|
||||||
static const char col_gray4[] = "#eeeeee";
|
static const char col_gray4[] = "#eeeeee";
|
||||||
static const char col_cyan[] = "#606060";
|
static const char col_cyan[] = "#005577";
|
||||||
static const char *colors[][3] = {
|
static const char *colors[][3] = {
|
||||||
/* fg bg border */
|
/* fg bg border */
|
||||||
[SchemeNorm] = { col_gray3, col_gray1, col_gray2 },
|
[SchemeNorm] = { col_gray3, col_gray1, col_gray2 },
|
||||||
@@ -45,7 +59,7 @@ static const Layout layouts[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/* key definitions */
|
/* key definitions */
|
||||||
#define MODKEY Mod4Mask
|
#define MODKEY Mod1Mask
|
||||||
#define TAGKEYS(KEY,TAG) \
|
#define TAGKEYS(KEY,TAG) \
|
||||||
{ MODKEY, KEY, view, {.ui = 1 << TAG} }, \
|
{ MODKEY, KEY, view, {.ui = 1 << TAG} }, \
|
||||||
{ MODKEY|ControlMask, KEY, toggleview, {.ui = 1 << TAG} }, \
|
{ MODKEY|ControlMask, KEY, toggleview, {.ui = 1 << TAG} }, \
|
||||||
@@ -58,8 +72,9 @@ static const Layout layouts[] = {
|
|||||||
/* commands */
|
/* commands */
|
||||||
static char dmenumon[2] = "0"; /* component of dmenucmd, manipulated in spawn() */
|
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 *dmenucmd[] = { "dmenu_run", "-m", dmenumon, "-fn", dmenufont, "-nb", col_gray1, "-nf", col_gray3, "-sb", col_cyan, "-sf", col_gray4, NULL };
|
||||||
static const char *termcmd[] = { "alacritty", NULL };
|
static const char *termcmd[] = { "st", NULL };
|
||||||
|
|
||||||
|
#include "movestack.c"
|
||||||
static Key keys[] = {
|
static Key keys[] = {
|
||||||
/* modifier key function argument */
|
/* modifier key function argument */
|
||||||
{ MODKEY, XK_p, spawn, {.v = dmenucmd } },
|
{ MODKEY, XK_p, spawn, {.v = dmenucmd } },
|
||||||
@@ -71,6 +86,8 @@ static Key keys[] = {
|
|||||||
{ MODKEY, XK_d, incnmaster, {.i = -1 } },
|
{ MODKEY, XK_d, incnmaster, {.i = -1 } },
|
||||||
{ MODKEY, XK_h, setmfact, {.f = -0.05} },
|
{ MODKEY, XK_h, setmfact, {.f = -0.05} },
|
||||||
{ MODKEY, XK_l, setmfact, {.f = +0.05} },
|
{ MODKEY, XK_l, setmfact, {.f = +0.05} },
|
||||||
|
{ MODKEY|ShiftMask, XK_j, movestack, {.i = +1 } },
|
||||||
|
{ MODKEY|ShiftMask, XK_k, movestack, {.i = -1 } },
|
||||||
{ MODKEY, XK_Return, zoom, {0} },
|
{ MODKEY, XK_Return, zoom, {0} },
|
||||||
{ MODKEY, XK_Tab, view, {0} },
|
{ MODKEY, XK_Tab, view, {0} },
|
||||||
{ MODKEY|ShiftMask, XK_c, killclient, {0} },
|
{ MODKEY|ShiftMask, XK_c, killclient, {0} },
|
||||||
@@ -101,8 +118,8 @@ static Key keys[] = {
|
|||||||
/* click can be ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle, ClkClientWin, or ClkRootWin */
|
/* click can be ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle, ClkClientWin, or ClkRootWin */
|
||||||
static Button buttons[] = {
|
static Button buttons[] = {
|
||||||
/* click event mask button function argument */
|
/* click event mask button function argument */
|
||||||
{ ClkLtSymbol, 0, Button1, setlayout, {0} },
|
{ ClkTagBar, MODKEY, Button1, tag, {0} },
|
||||||
{ ClkLtSymbol, 0, Button3, setlayout, {.v = &layouts[2]} },
|
{ ClkTagBar, MODKEY, Button3, toggletag, {0} },
|
||||||
{ ClkWinTitle, 0, Button2, zoom, {0} },
|
{ ClkWinTitle, 0, Button2, zoom, {0} },
|
||||||
{ ClkStatusText, 0, Button2, spawn, {.v = termcmd } },
|
{ ClkStatusText, 0, Button2, spawn, {.v = termcmd } },
|
||||||
{ ClkClientWin, MODKEY, Button1, movemouse, {0} },
|
{ ClkClientWin, MODKEY, Button1, movemouse, {0} },
|
||||||
@@ -114,21 +131,3 @@ static Button buttons[] = {
|
|||||||
{ ClkTagBar, MODKEY, Button3, toggletag, {0} },
|
{ ClkTagBar, MODKEY, Button3, toggletag, {0} },
|
||||||
};
|
};
|
||||||
|
|
||||||
static const char *ipcsockpath = "/tmp/dwm.sock";
|
|
||||||
static IPCCommand ipccommands[] = {
|
|
||||||
IPCCOMMAND( view, 1, {ARG_TYPE_UINT} ),
|
|
||||||
IPCCOMMAND( toggleview, 1, {ARG_TYPE_UINT} ),
|
|
||||||
IPCCOMMAND( tag, 1, {ARG_TYPE_UINT} ),
|
|
||||||
IPCCOMMAND( toggletag, 1, {ARG_TYPE_UINT} ),
|
|
||||||
IPCCOMMAND( tagmon, 1, {ARG_TYPE_UINT} ),
|
|
||||||
IPCCOMMAND( focusmon, 1, {ARG_TYPE_SINT} ),
|
|
||||||
IPCCOMMAND( focusstack, 1, {ARG_TYPE_SINT} ),
|
|
||||||
IPCCOMMAND( zoom, 1, {ARG_TYPE_NONE} ),
|
|
||||||
IPCCOMMAND( incnmaster, 1, {ARG_TYPE_SINT} ),
|
|
||||||
IPCCOMMAND( killclient, 1, {ARG_TYPE_SINT} ),
|
|
||||||
IPCCOMMAND( togglefloating, 1, {ARG_TYPE_NONE} ),
|
|
||||||
IPCCOMMAND( setmfact, 1, {ARG_TYPE_FLOAT} ),
|
|
||||||
IPCCOMMAND( setlayoutsafe, 1, {ARG_TYPE_PTR} ),
|
|
||||||
IPCCOMMAND( quit, 1, {ARG_TYPE_NONE} )
|
|
||||||
};
|
|
||||||
|
|
||||||
|
@@ -20,13 +20,9 @@ FREETYPEINC = /usr/include/freetype2
|
|||||||
# OpenBSD (uncomment)
|
# OpenBSD (uncomment)
|
||||||
#FREETYPEINC = ${X11INC}/freetype2
|
#FREETYPEINC = ${X11INC}/freetype2
|
||||||
|
|
||||||
# yajl
|
|
||||||
YAJLLIBS = -lyajl
|
|
||||||
YAJLINC = /usr/include/yajl
|
|
||||||
|
|
||||||
# includes and libs
|
# includes and libs
|
||||||
INCS = -I${X11INC} -I${FREETYPEINC} -I${YAJLINC}
|
INCS = -I${X11INC} -I${FREETYPEINC}
|
||||||
LIBS = -L${X11LIB} -lX11 ${XINERAMALIBS} ${FREETYPELIBS} ${YAJLLIBS}
|
LIBS = -L${X11LIB} -lX11 ${XINERAMALIBS} ${FREETYPELIBS}
|
||||||
|
|
||||||
# flags
|
# flags
|
||||||
CPPFLAGS = -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_POSIX_C_SOURCE=200809L -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS}
|
CPPFLAGS = -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_POSIX_C_SOURCE=200809L -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS}
|
||||||
|
445
dwm-anybar-polybar-tray-fix-20200810-bb2e722.diff
Normal file
445
dwm-anybar-polybar-tray-fix-20200810-bb2e722.diff
Normal file
@@ -0,0 +1,445 @@
|
|||||||
|
From 9b5719969ce85c3ecc0238d49c0255c5c2cc79f0 Mon Sep 17 00:00:00 2001
|
||||||
|
From: mihirlad55 <mihirlad55@gmail.com>
|
||||||
|
Date: Mon, 10 Aug 2020 01:39:28 +0000
|
||||||
|
Subject: [PATCH] Add support for managing external status bars
|
||||||
|
|
||||||
|
This patch allows dwm to manage other status bars such as
|
||||||
|
polybar/lemonbar without them needing to set override-redirect. For
|
||||||
|
all intents and purposes, DWM treats this bar as if it were its own
|
||||||
|
and as a result helps the status bar and DWM live in harmony.
|
||||||
|
|
||||||
|
This has a few advantages
|
||||||
|
* The bar does not block fullscreen windows
|
||||||
|
* DWM makes room for the status bar, so windows do not overlap the bar
|
||||||
|
* The bar can be hidden/killed and DWM will not keep an unsightly gap
|
||||||
|
where the bar was
|
||||||
|
* DWM receives EnterNotify events when your cursor enters the bar
|
||||||
|
|
||||||
|
To use another status bar, set usealtbar to 1 in your config.h and set
|
||||||
|
altbarclass to the class name (can be found using xprop) to the class
|
||||||
|
name of your status bar. Also make sure that if your status bar will
|
||||||
|
be displayed on top, topbar is set to 1 in your config, and if it will
|
||||||
|
be displayed on bottom, topbar is set to 0. This patch does not
|
||||||
|
support bars that are not docked at the top or at the bottom of your
|
||||||
|
monitor.
|
||||||
|
|
||||||
|
This verison of the patch fixes handling of polybar's tray.
|
||||||
|
|
||||||
|
The patch is developed at https://github.com/mihirlad55/dwm-anybar
|
||||||
|
---
|
||||||
|
config.def.h | 4 ++
|
||||||
|
dwm.c | 192 +++++++++++++++++++++++++++++++++++++++++++++++----
|
||||||
|
2 files changed, 181 insertions(+), 15 deletions(-)
|
||||||
|
|
||||||
|
diff --git a/config.def.h b/config.def.h
|
||||||
|
index 1c0b587..f45211b 100644
|
||||||
|
--- a/config.def.h
|
||||||
|
+++ b/config.def.h
|
||||||
|
@@ -5,6 +5,10 @@ static const unsigned int borderpx = 1; /* border pixel of windows */
|
||||||
|
static const unsigned int snap = 32; /* snap pixel */
|
||||||
|
static const int showbar = 1; /* 0 means no bar */
|
||||||
|
static const int topbar = 1; /* 0 means bottom bar */
|
||||||
|
+static const int usealtbar = 1; /* 1 means use non-dwm status bar */
|
||||||
|
+static const char *altbarclass = "Polybar"; /* Alternate bar class name */
|
||||||
|
+static const char *alttrayname = "tray"; /* Polybar tray instance name */
|
||||||
|
+static const char *altbarcmd = "$HOME/bar.sh"; /* Alternate bar launch command */
|
||||||
|
static const char *fonts[] = { "monospace:size=10" };
|
||||||
|
static const char dmenufont[] = "monospace:size=10";
|
||||||
|
static const char col_gray1[] = "#222222";
|
||||||
|
diff --git a/dwm.c b/dwm.c
|
||||||
|
index 9fd0286..c1d8ce0 100644
|
||||||
|
--- a/dwm.c
|
||||||
|
+++ b/dwm.c
|
||||||
|
@@ -47,8 +47,8 @@
|
||||||
|
/* macros */
|
||||||
|
#define BUTTONMASK (ButtonPressMask|ButtonReleaseMask)
|
||||||
|
#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 INTERSECT(x,y,w,h,m) (MAX(0, MIN((x)+(w),(m)->mx+(m)->mw) - MAX((x),(m)->mx)) \
|
||||||
|
+ * MAX(0, MIN((y)+(h),(m)->my+(m)->mh) - MAX((y),(m)->my)))
|
||||||
|
#define ISVISIBLE(C) ((C->tags & C->mon->tagset[C->mon->seltags]))
|
||||||
|
#define LENGTH(X) (sizeof X / sizeof X[0])
|
||||||
|
#define MOUSEMASK (BUTTONMASK|PointerMotionMask)
|
||||||
|
@@ -116,7 +116,8 @@ struct Monitor {
|
||||||
|
float mfact;
|
||||||
|
int nmaster;
|
||||||
|
int num;
|
||||||
|
- int by; /* bar geometry */
|
||||||
|
+ int by, bh; /* bar geometry */
|
||||||
|
+ int tx, tw; /* bar tray geometry */
|
||||||
|
int mx, my, mw, mh; /* screen size */
|
||||||
|
int wx, wy, ww, wh; /* window area */
|
||||||
|
unsigned int seltags;
|
||||||
|
@@ -129,6 +130,7 @@ struct Monitor {
|
||||||
|
Client *stack;
|
||||||
|
Monitor *next;
|
||||||
|
Window barwin;
|
||||||
|
+ Window traywin;
|
||||||
|
const Layout *lt[2];
|
||||||
|
};
|
||||||
|
|
||||||
|
@@ -179,6 +181,8 @@ static void incnmaster(const Arg *arg);
|
||||||
|
static void keypress(XEvent *e);
|
||||||
|
static void killclient(const Arg *arg);
|
||||||
|
static void manage(Window w, XWindowAttributes *wa);
|
||||||
|
+static void managealtbar(Window win, XWindowAttributes *wa);
|
||||||
|
+static void managetray(Window win, XWindowAttributes *wa);
|
||||||
|
static void mappingnotify(XEvent *e);
|
||||||
|
static void maprequest(XEvent *e);
|
||||||
|
static void monocle(Monitor *m);
|
||||||
|
@@ -195,6 +199,7 @@ static void resizemouse(const Arg *arg);
|
||||||
|
static void restack(Monitor *m);
|
||||||
|
static void run(void);
|
||||||
|
static void scan(void);
|
||||||
|
+static void scantray(void);
|
||||||
|
static int sendevent(Client *c, Atom proto);
|
||||||
|
static void sendmon(Client *c, Monitor *m);
|
||||||
|
static void setclientstate(Client *c, long state);
|
||||||
|
@@ -207,6 +212,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 void spawnbar();
|
||||||
|
static void tag(const Arg *arg);
|
||||||
|
static void tagmon(const Arg *arg);
|
||||||
|
static void tile(Monitor *);
|
||||||
|
@@ -216,6 +222,8 @@ static void toggletag(const Arg *arg);
|
||||||
|
static void toggleview(const Arg *arg);
|
||||||
|
static void unfocus(Client *c, int setfocus);
|
||||||
|
static void unmanage(Client *c, int destroyed);
|
||||||
|
+static void unmanagealtbar(Window w);
|
||||||
|
+static void unmanagetray(Window w);
|
||||||
|
static void unmapnotify(XEvent *e);
|
||||||
|
static void updatebarpos(Monitor *m);
|
||||||
|
static void updatebars(void);
|
||||||
|
@@ -230,6 +238,7 @@ static void updatewmhints(Client *c);
|
||||||
|
static void view(const Arg *arg);
|
||||||
|
static Client *wintoclient(Window w);
|
||||||
|
static Monitor *wintomon(Window w);
|
||||||
|
+static int wmclasscontains(Window win, const char *class, const char *name);
|
||||||
|
static int xerror(Display *dpy, XErrorEvent *ee);
|
||||||
|
static int xerrordummy(Display *dpy, XErrorEvent *ee);
|
||||||
|
static int xerrorstart(Display *dpy, XErrorEvent *ee);
|
||||||
|
@@ -505,8 +514,10 @@ cleanupmon(Monitor *mon)
|
||||||
|
for (m = mons; m && m->next != mon; m = m->next);
|
||||||
|
m->next = mon->next;
|
||||||
|
}
|
||||||
|
- XUnmapWindow(dpy, mon->barwin);
|
||||||
|
- XDestroyWindow(dpy, mon->barwin);
|
||||||
|
+ if (!usealtbar) {
|
||||||
|
+ XUnmapWindow(dpy, mon->barwin);
|
||||||
|
+ XDestroyWindow(dpy, mon->barwin);
|
||||||
|
+ }
|
||||||
|
free(mon);
|
||||||
|
}
|
||||||
|
|
||||||
|
@@ -568,7 +579,7 @@ configurenotify(XEvent *e)
|
||||||
|
for (c = m->clients; c; c = c->next)
|
||||||
|
if (c->isfullscreen)
|
||||||
|
resizeclient(c, m->mx, m->my, m->mw, m->mh);
|
||||||
|
- XMoveResizeWindow(dpy, m->barwin, m->wx, m->by, m->ww, bh);
|
||||||
|
+ XMoveResizeWindow(dpy, m->barwin, m->wx, m->by, m->ww, m->bh);
|
||||||
|
}
|
||||||
|
focus(NULL);
|
||||||
|
arrange(NULL);
|
||||||
|
@@ -639,6 +650,7 @@ createmon(void)
|
||||||
|
m->nmaster = nmaster;
|
||||||
|
m->showbar = showbar;
|
||||||
|
m->topbar = topbar;
|
||||||
|
+ m->bh = bh;
|
||||||
|
m->lt[0] = &layouts[0];
|
||||||
|
m->lt[1] = &layouts[1 % LENGTH(layouts)];
|
||||||
|
strncpy(m->ltsymbol, layouts[0].symbol, sizeof m->ltsymbol);
|
||||||
|
@@ -649,10 +661,15 @@ void
|
||||||
|
destroynotify(XEvent *e)
|
||||||
|
{
|
||||||
|
Client *c;
|
||||||
|
+ Monitor *m;
|
||||||
|
XDestroyWindowEvent *ev = &e->xdestroywindow;
|
||||||
|
|
||||||
|
if ((c = wintoclient(ev->window)))
|
||||||
|
unmanage(c, 1);
|
||||||
|
+ else if ((m = wintomon(ev->window)) && m->barwin == ev->window)
|
||||||
|
+ unmanagealtbar(ev->window);
|
||||||
|
+ else if (m->traywin == ev->window)
|
||||||
|
+ unmanagetray(ev->window);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -696,6 +713,9 @@ dirtomon(int dir)
|
||||||
|
void
|
||||||
|
drawbar(Monitor *m)
|
||||||
|
{
|
||||||
|
+ if (usealtbar)
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
int x, w, tw = 0;
|
||||||
|
int boxs = drw->fonts->h / 9;
|
||||||
|
int boxw = drw->fonts->h / 6 + 2;
|
||||||
|
@@ -1077,6 +1097,45 @@ manage(Window w, XWindowAttributes *wa)
|
||||||
|
focus(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+managealtbar(Window win, XWindowAttributes *wa)
|
||||||
|
+{
|
||||||
|
+ Monitor *m;
|
||||||
|
+ if (!(m = recttomon(wa->x, wa->y, wa->width, wa->height)))
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
+ m->barwin = win;
|
||||||
|
+ m->by = wa->y;
|
||||||
|
+ bh = m->bh = wa->height;
|
||||||
|
+ updatebarpos(m);
|
||||||
|
+ arrange(m);
|
||||||
|
+ XSelectInput(dpy, win, EnterWindowMask|FocusChangeMask|PropertyChangeMask|StructureNotifyMask);
|
||||||
|
+ XMoveResizeWindow(dpy, win, wa->x, wa->y, wa->width, wa->height);
|
||||||
|
+ XMapWindow(dpy, win);
|
||||||
|
+ XChangeProperty(dpy, root, netatom[NetClientList], XA_WINDOW, 32, PropModeAppend,
|
||||||
|
+ (unsigned char *) &win, 1);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+void
|
||||||
|
+managetray(Window win, XWindowAttributes *wa)
|
||||||
|
+{
|
||||||
|
+ Monitor *m;
|
||||||
|
+ if (!(m = recttomon(wa->x, wa->y, wa->width, wa->height)))
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
+ m->traywin = win;
|
||||||
|
+ m->tx = wa->x;
|
||||||
|
+ m->tw = wa->width;
|
||||||
|
+ updatebarpos(m);
|
||||||
|
+ arrange(m);
|
||||||
|
+ XSelectInput(dpy, win, EnterWindowMask|FocusChangeMask|PropertyChangeMask|StructureNotifyMask);
|
||||||
|
+ XMoveResizeWindow(dpy, win, wa->x, wa->y, wa->width, wa->height);
|
||||||
|
+ XMapWindow(dpy, win);
|
||||||
|
+ XChangeProperty(dpy, root, netatom[NetClientList], XA_WINDOW, 32, PropModeAppend,
|
||||||
|
+ (unsigned char *) &win, 1);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+
|
||||||
|
void
|
||||||
|
mappingnotify(XEvent *e)
|
||||||
|
{
|
||||||
|
@@ -1097,7 +1156,9 @@ maprequest(XEvent *e)
|
||||||
|
return;
|
||||||
|
if (wa.override_redirect)
|
||||||
|
return;
|
||||||
|
- if (!wintoclient(ev->window))
|
||||||
|
+ if (wmclasscontains(ev->window, altbarclass, ""))
|
||||||
|
+ managealtbar(ev->window, &wa);
|
||||||
|
+ else if (!wintoclient(ev->window))
|
||||||
|
manage(ev->window, &wa);
|
||||||
|
}
|
||||||
|
|
||||||
|
@@ -1393,7 +1454,9 @@ scan(void)
|
||||||
|
if (!XGetWindowAttributes(dpy, wins[i], &wa)
|
||||||
|
|| wa.override_redirect || XGetTransientForHint(dpy, wins[i], &d1))
|
||||||
|
continue;
|
||||||
|
- if (wa.map_state == IsViewable || getstate(wins[i]) == IconicState)
|
||||||
|
+ if (wmclasscontains(wins[i], altbarclass, ""))
|
||||||
|
+ managealtbar(wins[i], &wa);
|
||||||
|
+ else if (wa.map_state == IsViewable || getstate(wins[i]) == IconicState)
|
||||||
|
manage(wins[i], &wa);
|
||||||
|
}
|
||||||
|
for (i = 0; i < num; i++) { /* now the transients */
|
||||||
|
@@ -1408,6 +1471,29 @@ scan(void)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+scantray(void)
|
||||||
|
+{
|
||||||
|
+ unsigned int num;
|
||||||
|
+ Window d1, d2, *wins = NULL;
|
||||||
|
+ XWindowAttributes wa;
|
||||||
|
+
|
||||||
|
+ if (XQueryTree(dpy, root, &d1, &d2, &wins, &num)) {
|
||||||
|
+ for (unsigned int i = 0; i < num; i++) {
|
||||||
|
+ if (wmclasscontains(wins[i], altbarclass, alttrayname)) {
|
||||||
|
+ if (!XGetWindowAttributes(dpy, wins[i], &wa))
|
||||||
|
+ break;
|
||||||
|
+ managetray(wins[i], &wa);
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ if (wins)
|
||||||
|
+ XFree(wins);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+
|
||||||
|
+
|
||||||
|
void
|
||||||
|
sendmon(Client *c, Monitor *m)
|
||||||
|
{
|
||||||
|
@@ -1546,7 +1632,7 @@ setup(void)
|
||||||
|
if (!drw_fontset_create(drw, fonts, LENGTH(fonts)))
|
||||||
|
die("no fonts could be loaded.");
|
||||||
|
lrpad = drw->fonts->h;
|
||||||
|
- bh = drw->fonts->h + 2;
|
||||||
|
+ bh = usealtbar ? 0 : drw->fonts->h + 2;
|
||||||
|
updategeom();
|
||||||
|
/* init atoms */
|
||||||
|
utf8string = XInternAtom(dpy, "UTF8_STRING", False);
|
||||||
|
@@ -1595,6 +1681,7 @@ setup(void)
|
||||||
|
XSelectInput(dpy, root, wa.event_mask);
|
||||||
|
grabkeys();
|
||||||
|
focus(NULL);
|
||||||
|
+ spawnbar();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@@ -1653,6 +1740,13 @@ spawn(const Arg *arg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+spawnbar()
|
||||||
|
+{
|
||||||
|
+ if (*altbarcmd)
|
||||||
|
+ system(altbarcmd);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
tag(const Arg *arg)
|
||||||
|
{
|
||||||
|
@@ -1702,9 +1796,18 @@ tile(Monitor *m)
|
||||||
|
void
|
||||||
|
togglebar(const Arg *arg)
|
||||||
|
{
|
||||||
|
+ /**
|
||||||
|
+ * Polybar tray does not raise maprequest event. It must be manually scanned
|
||||||
|
+ * for. Scanning it too early while the tray is being populated would give
|
||||||
|
+ * wrong dimensions.
|
||||||
|
+ */
|
||||||
|
+ if (!selmon->traywin)
|
||||||
|
+ scantray();
|
||||||
|
+
|
||||||
|
selmon->showbar = !selmon->showbar;
|
||||||
|
updatebarpos(selmon);
|
||||||
|
- XMoveResizeWindow(dpy, selmon->barwin, selmon->wx, selmon->by, selmon->ww, bh);
|
||||||
|
+ XMoveResizeWindow(dpy, selmon->barwin, selmon->wx, selmon->by, selmon->ww, selmon->bh);
|
||||||
|
+ XMoveResizeWindow(dpy, selmon->traywin, selmon->tx, selmon->by, selmon->tw, selmon->bh);
|
||||||
|
arrange(selmon);
|
||||||
|
}
|
||||||
|
|
||||||
|
@@ -1787,10 +1890,41 @@ unmanage(Client *c, int destroyed)
|
||||||
|
arrange(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+unmanagealtbar(Window w)
|
||||||
|
+{
|
||||||
|
+ Monitor *m = wintomon(w);
|
||||||
|
+
|
||||||
|
+ if (!m)
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
+ m->barwin = 0;
|
||||||
|
+ m->by = 0;
|
||||||
|
+ m->bh = 0;
|
||||||
|
+ updatebarpos(m);
|
||||||
|
+ arrange(m);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+void
|
||||||
|
+unmanagetray(Window w)
|
||||||
|
+{
|
||||||
|
+ Monitor *m = wintomon(w);
|
||||||
|
+
|
||||||
|
+ if (!m)
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
+ m->traywin = 0;
|
||||||
|
+ m->tx = 0;
|
||||||
|
+ m->tw = 0;
|
||||||
|
+ updatebarpos(m);
|
||||||
|
+ arrange(m);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
unmapnotify(XEvent *e)
|
||||||
|
{
|
||||||
|
Client *c;
|
||||||
|
+ Monitor *m;
|
||||||
|
XUnmapEvent *ev = &e->xunmap;
|
||||||
|
|
||||||
|
if ((c = wintoclient(ev->window))) {
|
||||||
|
@@ -1798,12 +1932,18 @@ unmapnotify(XEvent *e)
|
||||||
|
setclientstate(c, WithdrawnState);
|
||||||
|
else
|
||||||
|
unmanage(c, 0);
|
||||||
|
- }
|
||||||
|
+ } else if ((m = wintomon(ev->window)) && m->barwin == ev->window)
|
||||||
|
+ unmanagealtbar(ev->window);
|
||||||
|
+ else if (m->traywin == ev->window)
|
||||||
|
+ unmanagetray(ev->window);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
updatebars(void)
|
||||||
|
{
|
||||||
|
+ if (usealtbar)
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
Monitor *m;
|
||||||
|
XSetWindowAttributes wa = {
|
||||||
|
.override_redirect = True,
|
||||||
|
@@ -1829,11 +1969,11 @@ updatebarpos(Monitor *m)
|
||||||
|
m->wy = m->my;
|
||||||
|
m->wh = m->mh;
|
||||||
|
if (m->showbar) {
|
||||||
|
- m->wh -= bh;
|
||||||
|
+ m->wh -= m->bh;
|
||||||
|
m->by = m->topbar ? m->wy : m->wy + m->wh;
|
||||||
|
- m->wy = m->topbar ? m->wy + bh : m->wy;
|
||||||
|
+ m->wy = m->topbar ? m->wy + m->bh : m->wy;
|
||||||
|
} else
|
||||||
|
- m->by = -bh;
|
||||||
|
+ m->by = -m->bh;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -2070,13 +2210,35 @@ wintomon(Window w)
|
||||||
|
if (w == root && getrootptr(&x, &y))
|
||||||
|
return recttomon(x, y, 1, 1);
|
||||||
|
for (m = mons; m; m = m->next)
|
||||||
|
- if (w == m->barwin)
|
||||||
|
+ if (w == m->barwin || w == m->traywin)
|
||||||
|
return m;
|
||||||
|
if ((c = wintoclient(w)))
|
||||||
|
return c->mon;
|
||||||
|
return selmon;
|
||||||
|
}
|
||||||
|
|
||||||
|
+int
|
||||||
|
+wmclasscontains(Window win, const char *class, const char *name)
|
||||||
|
+{
|
||||||
|
+ XClassHint ch = { NULL, NULL };
|
||||||
|
+ int res = 1;
|
||||||
|
+
|
||||||
|
+ if (XGetClassHint(dpy, win, &ch)) {
|
||||||
|
+ if (ch.res_name && strstr(ch.res_name, name) == NULL)
|
||||||
|
+ res = 0;
|
||||||
|
+ if (ch.res_class && strstr(ch.res_class, class) == NULL)
|
||||||
|
+ res = 0;
|
||||||
|
+ } else
|
||||||
|
+ res = 0;
|
||||||
|
+
|
||||||
|
+ if (ch.res_class)
|
||||||
|
+ XFree(ch.res_class);
|
||||||
|
+ if (ch.res_name)
|
||||||
|
+ XFree(ch.res_name);
|
||||||
|
+
|
||||||
|
+ return res;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
/* There's no way to check accesses to destroyed windows, thus those cases are
|
||||||
|
* ignored (especially on UnmapNotify's). Other types of errors call Xlibs
|
||||||
|
* default error handler, which may call exit. */
|
||||||
|
--
|
||||||
|
2.28.0
|
||||||
|
|
179
dwm-autostart-20210120-cb3f58a.diff
Normal file
179
dwm-autostart-20210120-cb3f58a.diff
Normal file
@@ -0,0 +1,179 @@
|
|||||||
|
From 37e970479dc5d40e57fc0cbfeaa5e39941483237 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Gan Ainm <gan.ainm.riomhphost@gmail.com>
|
||||||
|
Date: Wed, 10 Jun 2020 10:59:02 +0000
|
||||||
|
Subject: [PATCH] dwm-xdgautostart-6.2.diff
|
||||||
|
|
||||||
|
===================================================================
|
||||||
|
---
|
||||||
|
dwm.1 | 23 +++++++++++++++++
|
||||||
|
dwm.c | 82 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
||||||
|
2 files changed, 105 insertions(+)
|
||||||
|
|
||||||
|
diff --git a/dwm.1 b/dwm.1
|
||||||
|
index 13b3729..9533aa6 100644
|
||||||
|
--- a/dwm.1
|
||||||
|
+++ b/dwm.1
|
||||||
|
@@ -30,6 +30,14 @@ top left corner. The tags which are applied to one or more windows are
|
||||||
|
indicated with an empty square in the top left corner.
|
||||||
|
.P
|
||||||
|
dwm draws a small border around windows to indicate the focus state.
|
||||||
|
+.P
|
||||||
|
+On start, dwm can start additional programs that may be specified in two special
|
||||||
|
+shell scripts (see the FILES section below), autostart_blocking.sh and
|
||||||
|
+autostart.sh. The former is executed first and dwm will wait for its
|
||||||
|
+termination before starting. The latter is executed in the background before
|
||||||
|
+dwm enters its handler loop.
|
||||||
|
+.P
|
||||||
|
+Either of these files may be omitted.
|
||||||
|
.SH OPTIONS
|
||||||
|
.TP
|
||||||
|
.B \-v
|
||||||
|
@@ -152,6 +160,21 @@ Toggles focused window between floating and tiled state.
|
||||||
|
.TP
|
||||||
|
.B Mod1\-Button3
|
||||||
|
Resize focused window while dragging. Tiled windows will be toggled to the floating state.
|
||||||
|
+.SH FILES
|
||||||
|
+The files containing programs to be started along with dwm are searched for in
|
||||||
|
+the following directories:
|
||||||
|
+.IP "1. $XDG_DATA_HOME/dwm"
|
||||||
|
+.IP "2. $HOME/.local/share/dwm"
|
||||||
|
+.IP "3. $HOME/.dwm"
|
||||||
|
+.P
|
||||||
|
+The first existing directory is scanned for any of the autostart files below.
|
||||||
|
+.TP 15
|
||||||
|
+autostart.sh
|
||||||
|
+This file is started as a shell background process before dwm enters its handler
|
||||||
|
+loop.
|
||||||
|
+.TP 15
|
||||||
|
+autostart_blocking.sh
|
||||||
|
+This file is started before any autostart.sh; dwm waits for its termination.
|
||||||
|
.SH CUSTOMIZATION
|
||||||
|
dwm is customized by creating a custom config.h and (re)compiling the source
|
||||||
|
code. This keeps it fast, secure and simple.
|
||||||
|
diff --git a/dwm.c b/dwm.c
|
||||||
|
index 4465af1..2156b49 100644
|
||||||
|
--- a/dwm.c
|
||||||
|
+++ b/dwm.c
|
||||||
|
@@ -29,6 +29,7 @@
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
+#include <sys/stat.h>
|
||||||
|
#include <sys/wait.h>
|
||||||
|
#include <X11/cursorfont.h>
|
||||||
|
#include <X11/keysym.h>
|
||||||
|
@@ -193,6 +194,7 @@ static void resizeclient(Client *c, int x, int y, int w, int h);
|
||||||
|
static void resizemouse(const Arg *arg);
|
||||||
|
static void restack(Monitor *m);
|
||||||
|
static void run(void);
|
||||||
|
+static void runautostart(void);
|
||||||
|
static void scan(void);
|
||||||
|
static int sendevent(Client *c, Atom proto);
|
||||||
|
static void sendmon(Client *c, Monitor *m);
|
||||||
|
@@ -235,7 +237,11 @@ static int xerrorstart(Display *dpy, XErrorEvent *ee);
|
||||||
|
static void zoom(const Arg *arg);
|
||||||
|
|
||||||
|
/* variables */
|
||||||
|
+static const char autostartblocksh[] = "autostart_blocking.sh";
|
||||||
|
+static const char autostartsh[] = "autostart.sh";
|
||||||
|
static const char broken[] = "broken";
|
||||||
|
+static const char dwmdir[] = "dwm";
|
||||||
|
+static const char localshare[] = ".local/share";
|
||||||
|
static char stext[256];
|
||||||
|
static int screen;
|
||||||
|
static int sw, sh; /* X display screen geometry width, height */
|
||||||
|
@@ -1380,6 +1386,83 @@ run(void)
|
||||||
|
handler[ev.type](&ev); /* call handler */
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+runautostart(void)
|
||||||
|
+{
|
||||||
|
+ char *pathpfx;
|
||||||
|
+ char *path;
|
||||||
|
+ char *xdgdatahome;
|
||||||
|
+ char *home;
|
||||||
|
+ struct stat sb;
|
||||||
|
+
|
||||||
|
+ if ((home = getenv("HOME")) == NULL)
|
||||||
|
+ /* this is almost impossible */
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
+ /* if $XDG_DATA_HOME is set and not empty, use $XDG_DATA_HOME/dwm,
|
||||||
|
+ * otherwise use ~/.local/share/dwm as autostart script directory
|
||||||
|
+ */
|
||||||
|
+ xdgdatahome = getenv("XDG_DATA_HOME");
|
||||||
|
+ if (xdgdatahome != NULL && *xdgdatahome != '\0') {
|
||||||
|
+ /* space for path segments, separators and nul */
|
||||||
|
+ pathpfx = ecalloc(1, strlen(xdgdatahome) + strlen(dwmdir) + 2);
|
||||||
|
+
|
||||||
|
+ if (sprintf(pathpfx, "%s/%s", xdgdatahome, dwmdir) <= 0) {
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+ } else {
|
||||||
|
+ /* space for path segments, separators and nul */
|
||||||
|
+ pathpfx = ecalloc(1, strlen(home) + strlen(localshare)
|
||||||
|
+ + strlen(dwmdir) + 3);
|
||||||
|
+
|
||||||
|
+ if (sprintf(pathpfx, "%s/%s/%s", home, localshare, dwmdir) < 0) {
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ /* check if the autostart script directory exists */
|
||||||
|
+ if (! (stat(pathpfx, &sb) == 0 && S_ISDIR(sb.st_mode))) {
|
||||||
|
+ /* the XDG conformant path does not exist or is no directory
|
||||||
|
+ * so we try ~/.dwm instead
|
||||||
|
+ */
|
||||||
|
+ char *pathpfx_new = realloc(pathpfx, strlen(home) + strlen(dwmdir) + 3);
|
||||||
|
+ if(pathpfx_new == NULL) {
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+ pathpfx = pathpfx_new;
|
||||||
|
+
|
||||||
|
+ if (sprintf(pathpfx, "%s/.%s", home, dwmdir) <= 0) {
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ /* try the blocking script first */
|
||||||
|
+ path = ecalloc(1, strlen(pathpfx) + strlen(autostartblocksh) + 2);
|
||||||
|
+ if (sprintf(path, "%s/%s", pathpfx, autostartblocksh) <= 0) {
|
||||||
|
+ free(path);
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ if (access(path, X_OK) == 0)
|
||||||
|
+ system(path);
|
||||||
|
+
|
||||||
|
+ /* now the non-blocking script */
|
||||||
|
+ if (sprintf(path, "%s/%s", pathpfx, autostartsh) <= 0) {
|
||||||
|
+ free(path);
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ if (access(path, X_OK) == 0)
|
||||||
|
+ system(strcat(path, " &"));
|
||||||
|
+
|
||||||
|
+ free(pathpfx);
|
||||||
|
+ free(path);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
scan(void)
|
||||||
|
{
|
||||||
|
@@ -2142,6 +2223,7 @@ main(int argc, char *argv[])
|
||||||
|
die("pledge");
|
||||||
|
#endif /* __OpenBSD__ */
|
||||||
|
scan();
|
||||||
|
+ runautostart();
|
||||||
|
run();
|
||||||
|
cleanup();
|
||||||
|
XCloseDisplay(dpy);
|
||||||
|
--
|
||||||
|
2.27.0
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
95
dwm-movestack-20211115-a786211.diff
Normal file
95
dwm-movestack-20211115-a786211.diff
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
From 9a4037dc0ef56f91c009317e78e9e3790dafbb58 Mon Sep 17 00:00:00 2001
|
||||||
|
From: BrunoCooper17 <BrunoCooper17@outlook.com>
|
||||||
|
Date: Mon, 15 Nov 2021 14:04:53 -0600
|
||||||
|
Subject: [PATCH] MoveStack patch
|
||||||
|
|
||||||
|
This plugin allows you to move clients around in the stack and swap them
|
||||||
|
with the master. It emulates the behavior off mod+shift+j and mod+shift+k
|
||||||
|
in Xmonad. movestack(+1) will swap the client with the current focus with
|
||||||
|
the next client. movestack(-1) will swap the client with the current focus
|
||||||
|
with the previous client.
|
||||||
|
---
|
||||||
|
config.def.h | 3 +++
|
||||||
|
movestack.c | 48 ++++++++++++++++++++++++++++++++++++++++++++++++
|
||||||
|
2 files changed, 51 insertions(+)
|
||||||
|
create mode 100644 movestack.c
|
||||||
|
|
||||||
|
diff --git a/config.def.h b/config.def.h
|
||||||
|
index a2ac963..33efa5b 100644
|
||||||
|
--- a/config.def.h
|
||||||
|
+++ b/config.def.h
|
||||||
|
@@ -60,6 +60,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 };
|
||||||
|
|
||||||
|
+#include "movestack.c"
|
||||||
|
static Key keys[] = {
|
||||||
|
/* modifier key function argument */
|
||||||
|
{ MODKEY, XK_p, spawn, {.v = dmenucmd } },
|
||||||
|
@@ -71,6 +72,8 @@ 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_j, movestack, {.i = +1 } },
|
||||||
|
+ { MODKEY|ShiftMask, XK_k, movestack, {.i = -1 } },
|
||||||
|
{ MODKEY, XK_Return, zoom, {0} },
|
||||||
|
{ MODKEY, XK_Tab, view, {0} },
|
||||||
|
{ MODKEY|ShiftMask, XK_c, killclient, {0} },
|
||||||
|
diff --git a/movestack.c b/movestack.c
|
||||||
|
new file mode 100644
|
||||||
|
index 0000000..520f4ae
|
||||||
|
--- /dev/null
|
||||||
|
+++ b/movestack.c
|
||||||
|
@@ -0,0 +1,48 @@
|
||||||
|
+void
|
||||||
|
+movestack(const Arg *arg) {
|
||||||
|
+ Client *c = NULL, *p = NULL, *pc = NULL, *i;
|
||||||
|
+
|
||||||
|
+ if(arg->i > 0) {
|
||||||
|
+ /* find the client after selmon->sel */
|
||||||
|
+ for(c = selmon->sel->next; c && (!ISVISIBLE(c) || c->isfloating); c = c->next);
|
||||||
|
+ if(!c)
|
||||||
|
+ for(c = selmon->clients; c && (!ISVISIBLE(c) || c->isfloating); c = c->next);
|
||||||
|
+
|
||||||
|
+ }
|
||||||
|
+ else {
|
||||||
|
+ /* find the client before selmon->sel */
|
||||||
|
+ for(i = selmon->clients; i != selmon->sel; i = i->next)
|
||||||
|
+ if(ISVISIBLE(i) && !i->isfloating)
|
||||||
|
+ c = i;
|
||||||
|
+ if(!c)
|
||||||
|
+ for(; i; i = i->next)
|
||||||
|
+ if(ISVISIBLE(i) && !i->isfloating)
|
||||||
|
+ c = i;
|
||||||
|
+ }
|
||||||
|
+ /* find the client before selmon->sel and c */
|
||||||
|
+ for(i = selmon->clients; i && (!p || !pc); i = i->next) {
|
||||||
|
+ if(i->next == selmon->sel)
|
||||||
|
+ p = i;
|
||||||
|
+ if(i->next == c)
|
||||||
|
+ pc = i;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ /* swap c and selmon->sel selmon->clients in the selmon->clients list */
|
||||||
|
+ if(c && c != selmon->sel) {
|
||||||
|
+ Client *temp = selmon->sel->next==c?selmon->sel:selmon->sel->next;
|
||||||
|
+ selmon->sel->next = c->next==selmon->sel?c:c->next;
|
||||||
|
+ c->next = temp;
|
||||||
|
+
|
||||||
|
+ if(p && p != c)
|
||||||
|
+ p->next = c;
|
||||||
|
+ if(pc && pc != selmon->sel)
|
||||||
|
+ pc->next = selmon->sel;
|
||||||
|
+
|
||||||
|
+ if(selmon->sel == selmon->clients)
|
||||||
|
+ selmon->clients = c;
|
||||||
|
+ else if(c == selmon->clients)
|
||||||
|
+ selmon->clients = selmon->sel;
|
||||||
|
+
|
||||||
|
+ arrange(selmon);
|
||||||
|
+ }
|
||||||
|
+}
|
||||||
|
\ No newline at end of file
|
||||||
|
--
|
||||||
|
2.33.1
|
||||||
|
|
548
dwm-msg.c
548
dwm-msg.c
@@ -1,548 +0,0 @@
|
|||||||
#include <ctype.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <inttypes.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <sys/socket.h>
|
|
||||||
#include <sys/un.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <yajl/yajl_gen.h>
|
|
||||||
|
|
||||||
#define IPC_MAGIC "DWM-IPC"
|
|
||||||
// clang-format off
|
|
||||||
#define IPC_MAGIC_ARR { 'D', 'W', 'M', '-', 'I', 'P', 'C' }
|
|
||||||
// clang-format on
|
|
||||||
#define IPC_MAGIC_LEN 7 // Not including null char
|
|
||||||
|
|
||||||
#define IPC_EVENT_TAG_CHANGE "tag_change_event"
|
|
||||||
#define IPC_EVENT_CLIENT_FOCUS_CHANGE "client_focus_change_event"
|
|
||||||
#define IPC_EVENT_LAYOUT_CHANGE "layout_change_event"
|
|
||||||
#define IPC_EVENT_MONITOR_FOCUS_CHANGE "monitor_focus_change_event"
|
|
||||||
#define IPC_EVENT_FOCUSED_TITLE_CHANGE "focused_title_change_event"
|
|
||||||
#define IPC_EVENT_FOCUSED_STATE_CHANGE "focused_state_change_event"
|
|
||||||
|
|
||||||
#define YSTR(str) yajl_gen_string(gen, (unsigned char *)str, strlen(str))
|
|
||||||
#define YINT(num) yajl_gen_integer(gen, num)
|
|
||||||
#define YDOUBLE(num) yajl_gen_double(gen, num)
|
|
||||||
#define YBOOL(v) yajl_gen_bool(gen, v)
|
|
||||||
#define YNULL() yajl_gen_null(gen)
|
|
||||||
#define YARR(body) \
|
|
||||||
{ \
|
|
||||||
yajl_gen_array_open(gen); \
|
|
||||||
body; \
|
|
||||||
yajl_gen_array_close(gen); \
|
|
||||||
}
|
|
||||||
#define YMAP(body) \
|
|
||||||
{ \
|
|
||||||
yajl_gen_map_open(gen); \
|
|
||||||
body; \
|
|
||||||
yajl_gen_map_close(gen); \
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef unsigned long Window;
|
|
||||||
|
|
||||||
const char *DEFAULT_SOCKET_PATH = "/tmp/dwm.sock";
|
|
||||||
static int sock_fd = -1;
|
|
||||||
static unsigned int ignore_reply = 0;
|
|
||||||
|
|
||||||
typedef enum IPCMessageType {
|
|
||||||
IPC_TYPE_RUN_COMMAND = 0,
|
|
||||||
IPC_TYPE_GET_MONITORS = 1,
|
|
||||||
IPC_TYPE_GET_TAGS = 2,
|
|
||||||
IPC_TYPE_GET_LAYOUTS = 3,
|
|
||||||
IPC_TYPE_GET_DWM_CLIENT = 4,
|
|
||||||
IPC_TYPE_SUBSCRIBE = 5,
|
|
||||||
IPC_TYPE_EVENT = 6
|
|
||||||
} IPCMessageType;
|
|
||||||
|
|
||||||
// Every IPC message must begin with this
|
|
||||||
typedef struct dwm_ipc_header {
|
|
||||||
uint8_t magic[IPC_MAGIC_LEN];
|
|
||||||
uint32_t size;
|
|
||||||
uint8_t type;
|
|
||||||
} __attribute((packed)) dwm_ipc_header_t;
|
|
||||||
|
|
||||||
static int
|
|
||||||
recv_message(uint8_t *msg_type, uint32_t *reply_size, uint8_t **reply)
|
|
||||||
{
|
|
||||||
uint32_t read_bytes = 0;
|
|
||||||
const int32_t to_read = sizeof(dwm_ipc_header_t);
|
|
||||||
char header[to_read];
|
|
||||||
char *walk = header;
|
|
||||||
|
|
||||||
// Try to read header
|
|
||||||
while (read_bytes < to_read) {
|
|
||||||
ssize_t n = read(sock_fd, header + read_bytes, to_read - read_bytes);
|
|
||||||
|
|
||||||
if (n == 0) {
|
|
||||||
if (read_bytes == 0) {
|
|
||||||
fprintf(stderr, "Unexpectedly reached EOF while reading header.");
|
|
||||||
fprintf(stderr,
|
|
||||||
"Read %" PRIu32 " bytes, expected %" PRIu32 " total bytes.\n",
|
|
||||||
read_bytes, to_read);
|
|
||||||
return -2;
|
|
||||||
} else {
|
|
||||||
fprintf(stderr, "Unexpectedly reached EOF while reading header.");
|
|
||||||
fprintf(stderr,
|
|
||||||
"Read %" PRIu32 " bytes, expected %" PRIu32 " total bytes.\n",
|
|
||||||
read_bytes, to_read);
|
|
||||||
return -3;
|
|
||||||
}
|
|
||||||
} else if (n == -1) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
read_bytes += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if magic string in header matches
|
|
||||||
if (memcmp(walk, IPC_MAGIC, IPC_MAGIC_LEN) != 0) {
|
|
||||||
fprintf(stderr, "Invalid magic string. Got '%.*s', expected '%s'\n",
|
|
||||||
IPC_MAGIC_LEN, walk, IPC_MAGIC);
|
|
||||||
return -3;
|
|
||||||
}
|
|
||||||
|
|
||||||
walk += IPC_MAGIC_LEN;
|
|
||||||
|
|
||||||
// Extract reply size
|
|
||||||
memcpy(reply_size, walk, sizeof(uint32_t));
|
|
||||||
walk += sizeof(uint32_t);
|
|
||||||
|
|
||||||
// Extract message type
|
|
||||||
memcpy(msg_type, walk, sizeof(uint8_t));
|
|
||||||
walk += sizeof(uint8_t);
|
|
||||||
|
|
||||||
(*reply) = malloc(*reply_size);
|
|
||||||
|
|
||||||
// Extract payload
|
|
||||||
read_bytes = 0;
|
|
||||||
while (read_bytes < *reply_size) {
|
|
||||||
ssize_t n = read(sock_fd, *reply + read_bytes, *reply_size - read_bytes);
|
|
||||||
|
|
||||||
if (n == 0) {
|
|
||||||
fprintf(stderr, "Unexpectedly reached EOF while reading payload.");
|
|
||||||
fprintf(stderr, "Read %" PRIu32 " bytes, expected %" PRIu32 " bytes.\n",
|
|
||||||
read_bytes, *reply_size);
|
|
||||||
free(*reply);
|
|
||||||
return -2;
|
|
||||||
} else if (n == -1) {
|
|
||||||
if (errno == EINTR || errno == EAGAIN) continue;
|
|
||||||
free(*reply);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
read_bytes += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
read_socket(IPCMessageType *msg_type, uint32_t *msg_size, char **msg)
|
|
||||||
{
|
|
||||||
int ret = -1;
|
|
||||||
|
|
||||||
while (ret != 0) {
|
|
||||||
ret = recv_message((uint8_t *)msg_type, msg_size, (uint8_t **)msg);
|
|
||||||
|
|
||||||
if (ret < 0) {
|
|
||||||
// Try again (non-fatal error)
|
|
||||||
if (ret == -1 && (errno == EINTR || errno == EAGAIN)) continue;
|
|
||||||
|
|
||||||
fprintf(stderr, "Error receiving response from socket. ");
|
|
||||||
fprintf(stderr, "The connection might have been lost.\n");
|
|
||||||
exit(2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static ssize_t
|
|
||||||
write_socket(const void *buf, size_t count)
|
|
||||||
{
|
|
||||||
size_t written = 0;
|
|
||||||
|
|
||||||
while (written < count) {
|
|
||||||
const ssize_t n =
|
|
||||||
write(sock_fd, ((uint8_t *)buf) + written, count - written);
|
|
||||||
|
|
||||||
if (n == -1) {
|
|
||||||
if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR)
|
|
||||||
continue;
|
|
||||||
else
|
|
||||||
return n;
|
|
||||||
}
|
|
||||||
written += n;
|
|
||||||
}
|
|
||||||
return written;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
connect_to_socket()
|
|
||||||
{
|
|
||||||
struct sockaddr_un addr;
|
|
||||||
|
|
||||||
int sock = socket(AF_UNIX, SOCK_STREAM, 0);
|
|
||||||
|
|
||||||
// Initialize struct to 0
|
|
||||||
memset(&addr, 0, sizeof(struct sockaddr_un));
|
|
||||||
|
|
||||||
addr.sun_family = AF_UNIX;
|
|
||||||
strcpy(addr.sun_path, DEFAULT_SOCKET_PATH);
|
|
||||||
|
|
||||||
connect(sock, (const struct sockaddr *)&addr, sizeof(struct sockaddr_un));
|
|
||||||
|
|
||||||
sock_fd = sock;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
send_message(IPCMessageType msg_type, uint32_t msg_size, uint8_t *msg)
|
|
||||||
{
|
|
||||||
dwm_ipc_header_t header = {
|
|
||||||
.magic = IPC_MAGIC_ARR, .size = msg_size, .type = msg_type};
|
|
||||||
|
|
||||||
size_t header_size = sizeof(dwm_ipc_header_t);
|
|
||||||
size_t total_size = header_size + msg_size;
|
|
||||||
|
|
||||||
uint8_t buffer[total_size];
|
|
||||||
|
|
||||||
// Copy header to buffer
|
|
||||||
memcpy(buffer, &header, header_size);
|
|
||||||
// Copy message to buffer
|
|
||||||
memcpy(buffer + header_size, msg, header.size);
|
|
||||||
|
|
||||||
write_socket(buffer, total_size);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
is_float(const char *s)
|
|
||||||
{
|
|
||||||
size_t len = strlen(s);
|
|
||||||
int is_dot_used = 0;
|
|
||||||
int is_minus_used = 0;
|
|
||||||
|
|
||||||
// Floats can only have one decimal point in between or digits
|
|
||||||
// Optionally, floats can also be below zero (negative)
|
|
||||||
for (int i = 0; i < len; i++) {
|
|
||||||
if (isdigit(s[i]))
|
|
||||||
continue;
|
|
||||||
else if (!is_dot_used && s[i] == '.' && i != 0 && i != len - 1) {
|
|
||||||
is_dot_used = 1;
|
|
||||||
continue;
|
|
||||||
} else if (!is_minus_used && s[i] == '-' && i == 0) {
|
|
||||||
is_minus_used = 1;
|
|
||||||
continue;
|
|
||||||
} else
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
is_unsigned_int(const char *s)
|
|
||||||
{
|
|
||||||
size_t len = strlen(s);
|
|
||||||
|
|
||||||
// Unsigned int can only have digits
|
|
||||||
for (int i = 0; i < len; i++) {
|
|
||||||
if (isdigit(s[i]))
|
|
||||||
continue;
|
|
||||||
else
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
is_signed_int(const char *s)
|
|
||||||
{
|
|
||||||
size_t len = strlen(s);
|
|
||||||
|
|
||||||
// Signed int can only have digits and a negative sign at the start
|
|
||||||
for (int i = 0; i < len; i++) {
|
|
||||||
if (isdigit(s[i]))
|
|
||||||
continue;
|
|
||||||
else if (i == 0 && s[i] == '-') {
|
|
||||||
continue;
|
|
||||||
} else
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
flush_socket_reply()
|
|
||||||
{
|
|
||||||
IPCMessageType reply_type;
|
|
||||||
uint32_t reply_size;
|
|
||||||
char *reply;
|
|
||||||
|
|
||||||
read_socket(&reply_type, &reply_size, &reply);
|
|
||||||
|
|
||||||
free(reply);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
print_socket_reply()
|
|
||||||
{
|
|
||||||
IPCMessageType reply_type;
|
|
||||||
uint32_t reply_size;
|
|
||||||
char *reply;
|
|
||||||
|
|
||||||
read_socket(&reply_type, &reply_size, &reply);
|
|
||||||
|
|
||||||
printf("%.*s\n", reply_size, reply);
|
|
||||||
fflush(stdout);
|
|
||||||
free(reply);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
run_command(const char *name, char *args[], int argc)
|
|
||||||
{
|
|
||||||
const unsigned char *msg;
|
|
||||||
size_t msg_size;
|
|
||||||
|
|
||||||
yajl_gen gen = yajl_gen_alloc(NULL);
|
|
||||||
|
|
||||||
// Message format:
|
|
||||||
// {
|
|
||||||
// "command": "<name>",
|
|
||||||
// "args": [ ... ]
|
|
||||||
// }
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("command"); YSTR(name);
|
|
||||||
YSTR("args"); YARR(
|
|
||||||
for (int i = 0; i < argc; i++) {
|
|
||||||
if (is_signed_int(args[i])) {
|
|
||||||
long long num = atoll(args[i]);
|
|
||||||
YINT(num);
|
|
||||||
} else if (is_float(args[i])) {
|
|
||||||
float num = atof(args[i]);
|
|
||||||
YDOUBLE(num);
|
|
||||||
} else {
|
|
||||||
YSTR(args[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
yajl_gen_get_buf(gen, &msg, &msg_size);
|
|
||||||
|
|
||||||
send_message(IPC_TYPE_RUN_COMMAND, msg_size, (uint8_t *)msg);
|
|
||||||
|
|
||||||
if (!ignore_reply)
|
|
||||||
print_socket_reply();
|
|
||||||
else
|
|
||||||
flush_socket_reply();
|
|
||||||
|
|
||||||
yajl_gen_free(gen);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
get_monitors()
|
|
||||||
{
|
|
||||||
send_message(IPC_TYPE_GET_MONITORS, 1, (uint8_t *)"");
|
|
||||||
print_socket_reply();
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
get_tags()
|
|
||||||
{
|
|
||||||
send_message(IPC_TYPE_GET_TAGS, 1, (uint8_t *)"");
|
|
||||||
print_socket_reply();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
get_layouts()
|
|
||||||
{
|
|
||||||
send_message(IPC_TYPE_GET_LAYOUTS, 1, (uint8_t *)"");
|
|
||||||
print_socket_reply();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
get_dwm_client(Window win)
|
|
||||||
{
|
|
||||||
const unsigned char *msg;
|
|
||||||
size_t msg_size;
|
|
||||||
|
|
||||||
yajl_gen gen = yajl_gen_alloc(NULL);
|
|
||||||
|
|
||||||
// Message format:
|
|
||||||
// {
|
|
||||||
// "client_window_id": "<win>"
|
|
||||||
// }
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("client_window_id"); YINT(win);
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
yajl_gen_get_buf(gen, &msg, &msg_size);
|
|
||||||
|
|
||||||
send_message(IPC_TYPE_GET_DWM_CLIENT, msg_size, (uint8_t *)msg);
|
|
||||||
|
|
||||||
print_socket_reply();
|
|
||||||
|
|
||||||
yajl_gen_free(gen);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
subscribe(const char *event)
|
|
||||||
{
|
|
||||||
const unsigned char *msg;
|
|
||||||
size_t msg_size;
|
|
||||||
|
|
||||||
yajl_gen gen = yajl_gen_alloc(NULL);
|
|
||||||
|
|
||||||
// Message format:
|
|
||||||
// {
|
|
||||||
// "event": "<event>",
|
|
||||||
// "action": "subscribe"
|
|
||||||
// }
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("event"); YSTR(event);
|
|
||||||
YSTR("action"); YSTR("subscribe");
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
yajl_gen_get_buf(gen, &msg, &msg_size);
|
|
||||||
|
|
||||||
send_message(IPC_TYPE_SUBSCRIBE, msg_size, (uint8_t *)msg);
|
|
||||||
|
|
||||||
if (!ignore_reply)
|
|
||||||
print_socket_reply();
|
|
||||||
else
|
|
||||||
flush_socket_reply();
|
|
||||||
|
|
||||||
yajl_gen_free(gen);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
usage_error(const char *prog_name, const char *format, ...)
|
|
||||||
{
|
|
||||||
va_list args;
|
|
||||||
va_start(args, format);
|
|
||||||
|
|
||||||
fprintf(stderr, "Error: ");
|
|
||||||
vfprintf(stderr, format, args);
|
|
||||||
fprintf(stderr, "\nusage: %s <command> [...]\n", prog_name);
|
|
||||||
fprintf(stderr, "Try '%s help'\n", prog_name);
|
|
||||||
|
|
||||||
va_end(args);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
print_usage(const char *name)
|
|
||||||
{
|
|
||||||
printf("usage: %s [options] <command> [...]\n", name);
|
|
||||||
puts("");
|
|
||||||
puts("Commands:");
|
|
||||||
puts(" run_command <name> [args...] Run an IPC command");
|
|
||||||
puts("");
|
|
||||||
puts(" get_monitors Get monitor properties");
|
|
||||||
puts("");
|
|
||||||
puts(" get_tags Get list of tags");
|
|
||||||
puts("");
|
|
||||||
puts(" get_layouts Get list of layouts");
|
|
||||||
puts("");
|
|
||||||
puts(" get_dwm_client <window_id> Get dwm client proprties");
|
|
||||||
puts("");
|
|
||||||
puts(" subscribe [events...] Subscribe to specified events");
|
|
||||||
puts(" Options: " IPC_EVENT_TAG_CHANGE ",");
|
|
||||||
puts(" " IPC_EVENT_LAYOUT_CHANGE ",");
|
|
||||||
puts(" " IPC_EVENT_CLIENT_FOCUS_CHANGE ",");
|
|
||||||
puts(" " IPC_EVENT_MONITOR_FOCUS_CHANGE ",");
|
|
||||||
puts(" " IPC_EVENT_FOCUSED_TITLE_CHANGE ",");
|
|
||||||
puts(" " IPC_EVENT_FOCUSED_STATE_CHANGE);
|
|
||||||
puts("");
|
|
||||||
puts(" help Display this message");
|
|
||||||
puts("");
|
|
||||||
puts("Options:");
|
|
||||||
puts(" --ignore-reply Don't print reply messages from");
|
|
||||||
puts(" run_command and subscribe.");
|
|
||||||
puts("");
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
main(int argc, char *argv[])
|
|
||||||
{
|
|
||||||
const char *prog_name = argv[0];
|
|
||||||
|
|
||||||
connect_to_socket();
|
|
||||||
if (sock_fd == -1) {
|
|
||||||
fprintf(stderr, "Failed to connect to socket\n");
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
int i = 1;
|
|
||||||
if (i < argc && strcmp(argv[i], "--ignore-reply") == 0) {
|
|
||||||
ignore_reply = 1;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (i >= argc) usage_error(prog_name, "Expected an argument, got none");
|
|
||||||
|
|
||||||
if (strcmp(argv[i], "help") == 0)
|
|
||||||
print_usage(prog_name);
|
|
||||||
else if (strcmp(argv[i], "run_command") == 0) {
|
|
||||||
if (++i >= argc) usage_error(prog_name, "No command specified");
|
|
||||||
// Command name
|
|
||||||
char *command = argv[i];
|
|
||||||
// Command arguments are everything after command name
|
|
||||||
char **command_args = argv + ++i;
|
|
||||||
// Number of command arguments
|
|
||||||
int command_argc = argc - i;
|
|
||||||
run_command(command, command_args, command_argc);
|
|
||||||
} else if (strcmp(argv[i], "get_monitors") == 0) {
|
|
||||||
get_monitors();
|
|
||||||
} else if (strcmp(argv[i], "get_tags") == 0) {
|
|
||||||
get_tags();
|
|
||||||
} else if (strcmp(argv[i], "get_layouts") == 0) {
|
|
||||||
get_layouts();
|
|
||||||
} else if (strcmp(argv[i], "get_dwm_client") == 0) {
|
|
||||||
if (++i < argc) {
|
|
||||||
if (is_unsigned_int(argv[i])) {
|
|
||||||
Window win = atol(argv[i]);
|
|
||||||
get_dwm_client(win);
|
|
||||||
} else
|
|
||||||
usage_error(prog_name, "Expected unsigned integer argument");
|
|
||||||
} else
|
|
||||||
usage_error(prog_name, "Expected the window id");
|
|
||||||
} else if (strcmp(argv[i], "subscribe") == 0) {
|
|
||||||
if (++i < argc) {
|
|
||||||
for (int j = i; j < argc; j++) subscribe(argv[j]);
|
|
||||||
} else
|
|
||||||
usage_error(prog_name, "Expected event name");
|
|
||||||
// Keep listening for events forever
|
|
||||||
while (1) {
|
|
||||||
print_socket_reply();
|
|
||||||
}
|
|
||||||
} else
|
|
||||||
usage_error(prog_name, "Invalid argument '%s'", argv[i]);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
763
dwm-systray-6.3.diff
Normal file
763
dwm-systray-6.3.diff
Normal file
@@ -0,0 +1,763 @@
|
|||||||
|
diff --git a/config.def.h b/config.def.h
|
||||||
|
index a2ac963..4be4c06 100644
|
||||||
|
--- a/config.def.h
|
||||||
|
+++ b/config.def.h
|
||||||
|
@@ -2,9 +2,14 @@
|
||||||
|
|
||||||
|
/* appearance */
|
||||||
|
static const unsigned int borderpx = 1; /* border pixel of windows */
|
||||||
|
-static const unsigned int snap = 32; /* snap pixel */
|
||||||
|
-static const int showbar = 1; /* 0 means no bar */
|
||||||
|
-static const int topbar = 1; /* 0 means bottom bar */
|
||||||
|
+static const unsigned int snap = 32; /* snap pixel */
|
||||||
|
+static const unsigned int systraypinning = 0; /* 0: sloppy systray follows selected monitor, >0: pin systray to monitor X */
|
||||||
|
+static const unsigned int systrayonleft = 0; /* 0: systray in the right corner, >0: systray on left of status text */
|
||||||
|
+static const unsigned int systrayspacing = 2; /* systray spacing */
|
||||||
|
+static const int systraypinningfailfirst = 1; /* 1: if pinning fails, display systray on the first monitor, False: display systray on the last monitor*/
|
||||||
|
+static const int showsystray = 1; /* 0 means no systray */
|
||||||
|
+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" };
|
||||||
|
static const char dmenufont[] = "monospace:size=10";
|
||||||
|
static const char col_gray1[] = "#222222";
|
||||||
|
@@ -101,8 +106,8 @@ static Key keys[] = {
|
||||||
|
/* click can be ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle, ClkClientWin, or ClkRootWin */
|
||||||
|
static Button buttons[] = {
|
||||||
|
/* click event mask button function argument */
|
||||||
|
- { ClkLtSymbol, 0, Button1, setlayout, {0} },
|
||||||
|
- { ClkLtSymbol, 0, Button3, setlayout, {.v = &layouts[2]} },
|
||||||
|
+ { ClkTagBar, MODKEY, Button1, tag, {0} },
|
||||||
|
+ { ClkTagBar, MODKEY, Button3, toggletag, {0} },
|
||||||
|
{ ClkWinTitle, 0, Button2, zoom, {0} },
|
||||||
|
{ ClkStatusText, 0, Button2, spawn, {.v = termcmd } },
|
||||||
|
{ ClkClientWin, MODKEY, Button1, movemouse, {0} },
|
||||||
|
diff --git a/dwm.c b/dwm.c
|
||||||
|
index a96f33c..941c1c0 100644
|
||||||
|
--- a/dwm.c
|
||||||
|
+++ b/dwm.c
|
||||||
|
@@ -57,12 +57,27 @@
|
||||||
|
#define TAGMASK ((1 << LENGTH(tags)) - 1)
|
||||||
|
#define TEXTW(X) (drw_fontset_getwidth(drw, (X)) + lrpad)
|
||||||
|
|
||||||
|
+#define SYSTEM_TRAY_REQUEST_DOCK 0
|
||||||
|
+/* XEMBED messages */
|
||||||
|
+#define XEMBED_EMBEDDED_NOTIFY 0
|
||||||
|
+#define XEMBED_WINDOW_ACTIVATE 1
|
||||||
|
+#define XEMBED_FOCUS_IN 4
|
||||||
|
+#define XEMBED_MODALITY_ON 10
|
||||||
|
+#define XEMBED_MAPPED (1 << 0)
|
||||||
|
+#define XEMBED_WINDOW_ACTIVATE 1
|
||||||
|
+#define XEMBED_WINDOW_DEACTIVATE 2
|
||||||
|
+#define VERSION_MAJOR 0
|
||||||
|
+#define VERSION_MINOR 0
|
||||||
|
+#define XEMBED_EMBEDDED_VERSION (VERSION_MAJOR << 16) | VERSION_MINOR
|
||||||
|
+
|
||||||
|
/* enums */
|
||||||
|
enum { CurNormal, CurResize, CurMove, CurLast }; /* cursor */
|
||||||
|
enum { SchemeNorm, SchemeSel }; /* color schemes */
|
||||||
|
enum { NetSupported, NetWMName, NetWMState, NetWMCheck,
|
||||||
|
+ NetSystemTray, NetSystemTrayOP, NetSystemTrayOrientation, NetSystemTrayOrientationHorz,
|
||||||
|
NetWMFullscreen, NetActiveWindow, NetWMWindowType,
|
||||||
|
NetWMWindowTypeDialog, NetClientList, NetLast }; /* EWMH atoms */
|
||||||
|
+enum { Manager, Xembed, XembedInfo, XLast }; /* Xembed atoms */
|
||||||
|
enum { WMProtocols, WMDelete, WMState, WMTakeFocus, WMLast }; /* default atoms */
|
||||||
|
enum { ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle,
|
||||||
|
ClkClientWin, ClkRootWin, ClkLast }; /* clicks */
|
||||||
|
@@ -141,6 +156,12 @@ typedef struct {
|
||||||
|
int monitor;
|
||||||
|
} Rule;
|
||||||
|
|
||||||
|
+typedef struct Systray Systray;
|
||||||
|
+struct Systray {
|
||||||
|
+ Window win;
|
||||||
|
+ Client *icons;
|
||||||
|
+};
|
||||||
|
+
|
||||||
|
/* function declarations */
|
||||||
|
static void applyrules(Client *c);
|
||||||
|
static int applysizehints(Client *c, int *x, int *y, int *w, int *h, int interact);
|
||||||
|
@@ -172,6 +193,7 @@ static void focusstack(const Arg *arg);
|
||||||
|
static Atom getatomprop(Client *c, Atom prop);
|
||||||
|
static int getrootptr(int *x, int *y);
|
||||||
|
static long getstate(Window w);
|
||||||
|
+static unsigned int getsystraywidth();
|
||||||
|
static int gettextprop(Window w, Atom atom, char *text, unsigned int size);
|
||||||
|
static void grabbuttons(Client *c, int focused);
|
||||||
|
static void grabkeys(void);
|
||||||
|
@@ -189,13 +211,16 @@ static void pop(Client *);
|
||||||
|
static void propertynotify(XEvent *e);
|
||||||
|
static void quit(const Arg *arg);
|
||||||
|
static Monitor *recttomon(int x, int y, int w, int h);
|
||||||
|
+static void removesystrayicon(Client *i);
|
||||||
|
static void resize(Client *c, int x, int y, int w, int h, int interact);
|
||||||
|
+static void resizebarwin(Monitor *m);
|
||||||
|
static void resizeclient(Client *c, int x, int y, int w, int h);
|
||||||
|
static void resizemouse(const Arg *arg);
|
||||||
|
+static void resizerequest(XEvent *e);
|
||||||
|
static void restack(Monitor *m);
|
||||||
|
static void run(void);
|
||||||
|
static void scan(void);
|
||||||
|
-static int sendevent(Client *c, Atom proto);
|
||||||
|
+static int sendevent(Window w, Atom proto, int m, long d0, long d1, long d2, long d3, long d4);
|
||||||
|
static void sendmon(Client *c, Monitor *m);
|
||||||
|
static void setclientstate(Client *c, long state);
|
||||||
|
static void setfocus(Client *c);
|
||||||
|
@@ -207,6 +232,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 Monitor *systraytomon(Monitor *m);
|
||||||
|
static void tag(const Arg *arg);
|
||||||
|
static void tagmon(const Arg *arg);
|
||||||
|
static void tile(Monitor *);
|
||||||
|
@@ -224,18 +250,23 @@ static int updategeom(void);
|
||||||
|
static void updatenumlockmask(void);
|
||||||
|
static void updatesizehints(Client *c);
|
||||||
|
static void updatestatus(void);
|
||||||
|
+static void updatesystray(void);
|
||||||
|
+static void updatesystrayicongeom(Client *i, int w, int h);
|
||||||
|
+static void updatesystrayiconstate(Client *i, XPropertyEvent *ev);
|
||||||
|
static void updatetitle(Client *c);
|
||||||
|
static void updatewindowtype(Client *c);
|
||||||
|
static void updatewmhints(Client *c);
|
||||||
|
static void view(const Arg *arg);
|
||||||
|
static Client *wintoclient(Window w);
|
||||||
|
static Monitor *wintomon(Window w);
|
||||||
|
+static Client *wintosystrayicon(Window w);
|
||||||
|
static int xerror(Display *dpy, XErrorEvent *ee);
|
||||||
|
static int xerrordummy(Display *dpy, XErrorEvent *ee);
|
||||||
|
static int xerrorstart(Display *dpy, XErrorEvent *ee);
|
||||||
|
static void zoom(const Arg *arg);
|
||||||
|
|
||||||
|
/* variables */
|
||||||
|
+static Systray *systray = NULL;
|
||||||
|
static const char broken[] = "broken";
|
||||||
|
static char stext[256];
|
||||||
|
static int screen;
|
||||||
|
@@ -258,9 +289,10 @@ static void (*handler[LASTEvent]) (XEvent *) = {
|
||||||
|
[MapRequest] = maprequest,
|
||||||
|
[MotionNotify] = motionnotify,
|
||||||
|
[PropertyNotify] = propertynotify,
|
||||||
|
+ [ResizeRequest] = resizerequest,
|
||||||
|
[UnmapNotify] = unmapnotify
|
||||||
|
};
|
||||||
|
-static Atom wmatom[WMLast], netatom[NetLast];
|
||||||
|
+static Atom wmatom[WMLast], netatom[NetLast], xatom[XLast];
|
||||||
|
static int running = 1;
|
||||||
|
static Cur *cursor[CurLast];
|
||||||
|
static Clr **scheme;
|
||||||
|
@@ -440,7 +472,7 @@ buttonpress(XEvent *e)
|
||||||
|
arg.ui = 1 << i;
|
||||||
|
} else if (ev->x < x + blw)
|
||||||
|
click = ClkLtSymbol;
|
||||||
|
- else if (ev->x > selmon->ww - (int)TEXTW(stext))
|
||||||
|
+ else if (ev->x > selmon->ww - (int)TEXTW(stext) - getsystraywidth())
|
||||||
|
click = ClkStatusText;
|
||||||
|
else
|
||||||
|
click = ClkWinTitle;
|
||||||
|
@@ -483,7 +515,14 @@ cleanup(void)
|
||||||
|
XUngrabKey(dpy, AnyKey, AnyModifier, root);
|
||||||
|
while (mons)
|
||||||
|
cleanupmon(mons);
|
||||||
|
- for (i = 0; i < CurLast; i++)
|
||||||
|
+
|
||||||
|
+ if (showsystray) {
|
||||||
|
+ XUnmapWindow(dpy, systray->win);
|
||||||
|
+ XDestroyWindow(dpy, systray->win);
|
||||||
|
+ free(systray);
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ for (i = 0; i < CurLast; i++)
|
||||||
|
drw_cur_free(drw, cursor[i]);
|
||||||
|
for (i = 0; i < LENGTH(colors); i++)
|
||||||
|
free(scheme[i]);
|
||||||
|
@@ -513,9 +552,58 @@ cleanupmon(Monitor *mon)
|
||||||
|
void
|
||||||
|
clientmessage(XEvent *e)
|
||||||
|
{
|
||||||
|
+ XWindowAttributes wa;
|
||||||
|
+ XSetWindowAttributes swa;
|
||||||
|
XClientMessageEvent *cme = &e->xclient;
|
||||||
|
Client *c = wintoclient(cme->window);
|
||||||
|
|
||||||
|
+ if (showsystray && cme->window == systray->win && cme->message_type == netatom[NetSystemTrayOP]) {
|
||||||
|
+ /* add systray icons */
|
||||||
|
+ if (cme->data.l[1] == SYSTEM_TRAY_REQUEST_DOCK) {
|
||||||
|
+ if (!(c = (Client *)calloc(1, sizeof(Client))))
|
||||||
|
+ die("fatal: could not malloc() %u bytes\n", sizeof(Client));
|
||||||
|
+ if (!(c->win = cme->data.l[2])) {
|
||||||
|
+ free(c);
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+ c->mon = selmon;
|
||||||
|
+ c->next = systray->icons;
|
||||||
|
+ systray->icons = c;
|
||||||
|
+ if (!XGetWindowAttributes(dpy, c->win, &wa)) {
|
||||||
|
+ /* use sane defaults */
|
||||||
|
+ wa.width = bh;
|
||||||
|
+ wa.height = bh;
|
||||||
|
+ wa.border_width = 0;
|
||||||
|
+ }
|
||||||
|
+ c->x = c->oldx = c->y = c->oldy = 0;
|
||||||
|
+ c->w = c->oldw = wa.width;
|
||||||
|
+ c->h = c->oldh = wa.height;
|
||||||
|
+ c->oldbw = wa.border_width;
|
||||||
|
+ c->bw = 0;
|
||||||
|
+ c->isfloating = True;
|
||||||
|
+ /* reuse tags field as mapped status */
|
||||||
|
+ c->tags = 1;
|
||||||
|
+ updatesizehints(c);
|
||||||
|
+ updatesystrayicongeom(c, wa.width, wa.height);
|
||||||
|
+ XAddToSaveSet(dpy, c->win);
|
||||||
|
+ XSelectInput(dpy, c->win, StructureNotifyMask | PropertyChangeMask | ResizeRedirectMask);
|
||||||
|
+ XReparentWindow(dpy, c->win, systray->win, 0, 0);
|
||||||
|
+ /* use parents background color */
|
||||||
|
+ swa.background_pixel = scheme[SchemeNorm][ColBg].pixel;
|
||||||
|
+ XChangeWindowAttributes(dpy, c->win, CWBackPixel, &swa);
|
||||||
|
+ sendevent(c->win, netatom[Xembed], StructureNotifyMask, CurrentTime, XEMBED_EMBEDDED_NOTIFY, 0 , systray->win, XEMBED_EMBEDDED_VERSION);
|
||||||
|
+ /* FIXME not sure if I have to send these events, too */
|
||||||
|
+ sendevent(c->win, netatom[Xembed], StructureNotifyMask, CurrentTime, XEMBED_FOCUS_IN, 0 , systray->win, XEMBED_EMBEDDED_VERSION);
|
||||||
|
+ sendevent(c->win, netatom[Xembed], StructureNotifyMask, CurrentTime, XEMBED_WINDOW_ACTIVATE, 0 , systray->win, XEMBED_EMBEDDED_VERSION);
|
||||||
|
+ sendevent(c->win, netatom[Xembed], StructureNotifyMask, CurrentTime, XEMBED_MODALITY_ON, 0 , systray->win, XEMBED_EMBEDDED_VERSION);
|
||||||
|
+ XSync(dpy, False);
|
||||||
|
+ resizebarwin(selmon);
|
||||||
|
+ updatesystray();
|
||||||
|
+ setclientstate(c, NormalState);
|
||||||
|
+ }
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
if (!c)
|
||||||
|
return;
|
||||||
|
if (cme->message_type == netatom[NetWMState]) {
|
||||||
|
@@ -568,7 +656,7 @@ configurenotify(XEvent *e)
|
||||||
|
for (c = m->clients; c; c = c->next)
|
||||||
|
if (c->isfullscreen)
|
||||||
|
resizeclient(c, m->mx, m->my, m->mw, m->mh);
|
||||||
|
- XMoveResizeWindow(dpy, m->barwin, m->wx, m->by, m->ww, bh);
|
||||||
|
+ resizebarwin(m);
|
||||||
|
}
|
||||||
|
focus(NULL);
|
||||||
|
arrange(NULL);
|
||||||
|
@@ -653,6 +741,11 @@ destroynotify(XEvent *e)
|
||||||
|
|
||||||
|
if ((c = wintoclient(ev->window)))
|
||||||
|
unmanage(c, 1);
|
||||||
|
+ else if ((c = wintosystrayicon(ev->window))) {
|
||||||
|
+ removesystrayicon(c);
|
||||||
|
+ resizebarwin(selmon);
|
||||||
|
+ updatesystray();
|
||||||
|
+ }
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -696,7 +789,7 @@ dirtomon(int dir)
|
||||||
|
void
|
||||||
|
drawbar(Monitor *m)
|
||||||
|
{
|
||||||
|
- int x, w, tw = 0;
|
||||||
|
+ int x, w, tw = 0, stw = 0;
|
||||||
|
int boxs = drw->fonts->h / 9;
|
||||||
|
int boxw = drw->fonts->h / 6 + 2;
|
||||||
|
unsigned int i, occ = 0, urg = 0;
|
||||||
|
@@ -705,13 +798,17 @@ drawbar(Monitor *m)
|
||||||
|
if (!m->showbar)
|
||||||
|
return;
|
||||||
|
|
||||||
|
+ if(showsystray && m == systraytomon(m) && !systrayonleft)
|
||||||
|
+ stw = getsystraywidth();
|
||||||
|
+
|
||||||
|
/* 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]);
|
||||||
|
- tw = TEXTW(stext) - lrpad + 2; /* 2px right padding */
|
||||||
|
- drw_text(drw, m->ww - tw, 0, tw, bh, 0, stext, 0);
|
||||||
|
+ tw = TEXTW(stext) - lrpad / 2 + 2; /* 2px extra right padding */
|
||||||
|
+ drw_text(drw, m->ww - tw - stw, 0, tw, bh, lrpad / 2 - 2, stext, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
+ resizebarwin(m);
|
||||||
|
for (c = m->clients; c; c = c->next) {
|
||||||
|
occ |= c->tags;
|
||||||
|
if (c->isurgent)
|
||||||
|
@@ -732,7 +829,7 @@ drawbar(Monitor *m)
|
||||||
|
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 ((w = m->ww - tw - stw - x) > 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);
|
||||||
|
@@ -743,7 +840,7 @@ drawbar(Monitor *m)
|
||||||
|
drw_rect(drw, x, 0, w, bh, 1, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
- drw_map(drw, m->barwin, 0, 0, m->ww, bh);
|
||||||
|
+ drw_map(drw, m->barwin, 0, 0, m->ww - stw, bh);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -780,8 +877,11 @@ expose(XEvent *e)
|
||||||
|
Monitor *m;
|
||||||
|
XExposeEvent *ev = &e->xexpose;
|
||||||
|
|
||||||
|
- if (ev->count == 0 && (m = wintomon(ev->window)))
|
||||||
|
+ if (ev->count == 0 && (m = wintomon(ev->window))) {
|
||||||
|
drawbar(m);
|
||||||
|
+ if (m == selmon)
|
||||||
|
+ updatesystray();
|
||||||
|
+ }
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -867,9 +967,17 @@ getatomprop(Client *c, Atom prop)
|
||||||
|
unsigned char *p = NULL;
|
||||||
|
Atom da, atom = None;
|
||||||
|
|
||||||
|
- if (XGetWindowProperty(dpy, c->win, prop, 0L, sizeof atom, False, XA_ATOM,
|
||||||
|
+ /* FIXME getatomprop should return the number of items and a pointer to
|
||||||
|
+ * the stored data instead of this workaround */
|
||||||
|
+ Atom req = XA_ATOM;
|
||||||
|
+ if (prop == xatom[XembedInfo])
|
||||||
|
+ req = xatom[XembedInfo];
|
||||||
|
+
|
||||||
|
+ if (XGetWindowProperty(dpy, c->win, prop, 0L, sizeof atom, False, req,
|
||||||
|
&da, &di, &dl, &dl, &p) == Success && p) {
|
||||||
|
atom = *(Atom *)p;
|
||||||
|
+ if (da == xatom[XembedInfo] && dl == 2)
|
||||||
|
+ atom = ((Atom *)p)[1];
|
||||||
|
XFree(p);
|
||||||
|
}
|
||||||
|
return atom;
|
||||||
|
@@ -903,6 +1011,16 @@ getstate(Window w)
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
+unsigned int
|
||||||
|
+getsystraywidth()
|
||||||
|
+{
|
||||||
|
+ unsigned int w = 0;
|
||||||
|
+ Client *i;
|
||||||
|
+ if(showsystray)
|
||||||
|
+ for(i = systray->icons; i; w += i->w + systrayspacing, i = i->next) ;
|
||||||
|
+ return w ? w + systrayspacing : 1;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
int
|
||||||
|
gettextprop(Window w, Atom atom, char *text, unsigned int size)
|
||||||
|
{
|
||||||
|
@@ -1007,7 +1125,8 @@ killclient(const Arg *arg)
|
||||||
|
{
|
||||||
|
if (!selmon->sel)
|
||||||
|
return;
|
||||||
|
- if (!sendevent(selmon->sel, wmatom[WMDelete])) {
|
||||||
|
+
|
||||||
|
+ if (!sendevent(selmon->sel->win, wmatom[WMDelete], NoEventMask, wmatom[WMDelete], CurrentTime, 0 , 0, 0)) {
|
||||||
|
XGrabServer(dpy);
|
||||||
|
XSetErrorHandler(xerrordummy);
|
||||||
|
XSetCloseDownMode(dpy, DestroyAll);
|
||||||
|
@@ -1096,6 +1215,13 @@ maprequest(XEvent *e)
|
||||||
|
static XWindowAttributes wa;
|
||||||
|
XMapRequestEvent *ev = &e->xmaprequest;
|
||||||
|
|
||||||
|
+ Client *i;
|
||||||
|
+ if ((i = wintosystrayicon(ev->window))) {
|
||||||
|
+ sendevent(i->win, netatom[Xembed], StructureNotifyMask, CurrentTime, XEMBED_WINDOW_ACTIVATE, 0, systray->win, XEMBED_EMBEDDED_VERSION);
|
||||||
|
+ resizebarwin(selmon);
|
||||||
|
+ updatesystray();
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
if (!XGetWindowAttributes(dpy, ev->window, &wa))
|
||||||
|
return;
|
||||||
|
if (wa.override_redirect)
|
||||||
|
@@ -1219,7 +1345,18 @@ propertynotify(XEvent *e)
|
||||||
|
Window trans;
|
||||||
|
XPropertyEvent *ev = &e->xproperty;
|
||||||
|
|
||||||
|
- if ((ev->window == root) && (ev->atom == XA_WM_NAME))
|
||||||
|
+ if ((c = wintosystrayicon(ev->window))) {
|
||||||
|
+ if (ev->atom == XA_WM_NORMAL_HINTS) {
|
||||||
|
+ updatesizehints(c);
|
||||||
|
+ updatesystrayicongeom(c, c->w, c->h);
|
||||||
|
+ }
|
||||||
|
+ else
|
||||||
|
+ updatesystrayiconstate(c, ev);
|
||||||
|
+ resizebarwin(selmon);
|
||||||
|
+ updatesystray();
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ if ((ev->window == root) && (ev->atom == XA_WM_NAME))
|
||||||
|
updatestatus();
|
||||||
|
else if (ev->state == PropertyDelete)
|
||||||
|
return; /* ignore */
|
||||||
|
@@ -1269,6 +1406,19 @@ recttomon(int x, int y, int w, int h)
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+removesystrayicon(Client *i)
|
||||||
|
+{
|
||||||
|
+ Client **ii;
|
||||||
|
+
|
||||||
|
+ if (!showsystray || !i)
|
||||||
|
+ return;
|
||||||
|
+ for (ii = &systray->icons; *ii && *ii != i; ii = &(*ii)->next);
|
||||||
|
+ if (ii)
|
||||||
|
+ *ii = i->next;
|
||||||
|
+ free(i);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
resize(Client *c, int x, int y, int w, int h, int interact)
|
||||||
|
{
|
||||||
|
@@ -1276,6 +1426,14 @@ resize(Client *c, int x, int y, int w, int h, int interact)
|
||||||
|
resizeclient(c, x, y, w, h);
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+resizebarwin(Monitor *m) {
|
||||||
|
+ unsigned int w = m->ww;
|
||||||
|
+ if (showsystray && m == systraytomon(m) && !systrayonleft)
|
||||||
|
+ w -= getsystraywidth();
|
||||||
|
+ XMoveResizeWindow(dpy, m->barwin, m->wx, m->by, w, bh);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
resizeclient(Client *c, int x, int y, int w, int h)
|
||||||
|
{
|
||||||
|
@@ -1348,6 +1506,19 @@ resizemouse(const Arg *arg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
+void
|
||||||
|
+resizerequest(XEvent *e)
|
||||||
|
+{
|
||||||
|
+ XResizeRequestEvent *ev = &e->xresizerequest;
|
||||||
|
+ Client *i;
|
||||||
|
+
|
||||||
|
+ if ((i = wintosystrayicon(ev->window))) {
|
||||||
|
+ updatesystrayicongeom(i, ev->width, ev->height);
|
||||||
|
+ resizebarwin(selmon);
|
||||||
|
+ updatesystray();
|
||||||
|
+ }
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
restack(Monitor *m)
|
||||||
|
{
|
||||||
|
@@ -1437,26 +1608,37 @@ setclientstate(Client *c, long state)
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
-sendevent(Client *c, Atom proto)
|
||||||
|
+sendevent(Window w, Atom proto, int mask, long d0, long d1, long d2, long d3, long d4)
|
||||||
|
{
|
||||||
|
int n;
|
||||||
|
- Atom *protocols;
|
||||||
|
+ Atom *protocols, mt;
|
||||||
|
int exists = 0;
|
||||||
|
XEvent ev;
|
||||||
|
|
||||||
|
- if (XGetWMProtocols(dpy, c->win, &protocols, &n)) {
|
||||||
|
- while (!exists && n--)
|
||||||
|
- exists = protocols[n] == proto;
|
||||||
|
- XFree(protocols);
|
||||||
|
+ if (proto == wmatom[WMTakeFocus] || proto == wmatom[WMDelete]) {
|
||||||
|
+ mt = wmatom[WMProtocols];
|
||||||
|
+ if (XGetWMProtocols(dpy, w, &protocols, &n)) {
|
||||||
|
+ while (!exists && n--)
|
||||||
|
+ exists = protocols[n] == proto;
|
||||||
|
+ XFree(protocols);
|
||||||
|
+ }
|
||||||
|
}
|
||||||
|
+ else {
|
||||||
|
+ exists = True;
|
||||||
|
+ mt = proto;
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
if (exists) {
|
||||||
|
ev.type = ClientMessage;
|
||||||
|
- ev.xclient.window = c->win;
|
||||||
|
- ev.xclient.message_type = wmatom[WMProtocols];
|
||||||
|
+ ev.xclient.window = w;
|
||||||
|
+ ev.xclient.message_type = mt;
|
||||||
|
ev.xclient.format = 32;
|
||||||
|
- ev.xclient.data.l[0] = proto;
|
||||||
|
- ev.xclient.data.l[1] = CurrentTime;
|
||||||
|
- XSendEvent(dpy, c->win, False, NoEventMask, &ev);
|
||||||
|
+ ev.xclient.data.l[0] = d0;
|
||||||
|
+ ev.xclient.data.l[1] = d1;
|
||||||
|
+ ev.xclient.data.l[2] = d2;
|
||||||
|
+ ev.xclient.data.l[3] = d3;
|
||||||
|
+ ev.xclient.data.l[4] = d4;
|
||||||
|
+ XSendEvent(dpy, w, False, mask, &ev);
|
||||||
|
}
|
||||||
|
return exists;
|
||||||
|
}
|
||||||
|
@@ -1470,7 +1652,7 @@ setfocus(Client *c)
|
||||||
|
XA_WINDOW, 32, PropModeReplace,
|
||||||
|
(unsigned char *) &(c->win), 1);
|
||||||
|
}
|
||||||
|
- sendevent(c, wmatom[WMTakeFocus]);
|
||||||
|
+ sendevent(c->win, wmatom[WMTakeFocus], NoEventMask, wmatom[WMTakeFocus], CurrentTime, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -1558,15 +1740,22 @@ setup(void)
|
||||||
|
wmatom[WMState] = XInternAtom(dpy, "WM_STATE", False);
|
||||||
|
wmatom[WMTakeFocus] = XInternAtom(dpy, "WM_TAKE_FOCUS", False);
|
||||||
|
netatom[NetActiveWindow] = XInternAtom(dpy, "_NET_ACTIVE_WINDOW", False);
|
||||||
|
- netatom[NetSupported] = XInternAtom(dpy, "_NET_SUPPORTED", False);
|
||||||
|
- netatom[NetWMName] = XInternAtom(dpy, "_NET_WM_NAME", False);
|
||||||
|
+ netatom[NetSupported] = XInternAtom(dpy, "_NET_SUPPORTED", False);
|
||||||
|
+ netatom[NetSystemTray] = XInternAtom(dpy, "_NET_SYSTEM_TRAY_S0", False);
|
||||||
|
+ netatom[NetSystemTrayOP] = XInternAtom(dpy, "_NET_SYSTEM_TRAY_OPCODE", False);
|
||||||
|
+ netatom[NetSystemTrayOrientation] = XInternAtom(dpy, "_NET_SYSTEM_TRAY_ORIENTATION", False);
|
||||||
|
+ netatom[NetSystemTrayOrientationHorz] = XInternAtom(dpy, "_NET_SYSTEM_TRAY_ORIENTATION_HORZ", False);
|
||||||
|
+ netatom[NetWMName] = XInternAtom(dpy, "_NET_WM_NAME", False);
|
||||||
|
netatom[NetWMState] = XInternAtom(dpy, "_NET_WM_STATE", False);
|
||||||
|
netatom[NetWMCheck] = XInternAtom(dpy, "_NET_SUPPORTING_WM_CHECK", False);
|
||||||
|
netatom[NetWMFullscreen] = XInternAtom(dpy, "_NET_WM_STATE_FULLSCREEN", False);
|
||||||
|
netatom[NetWMWindowType] = XInternAtom(dpy, "_NET_WM_WINDOW_TYPE", False);
|
||||||
|
netatom[NetWMWindowTypeDialog] = XInternAtom(dpy, "_NET_WM_WINDOW_TYPE_DIALOG", False);
|
||||||
|
netatom[NetClientList] = XInternAtom(dpy, "_NET_CLIENT_LIST", False);
|
||||||
|
- /* init cursors */
|
||||||
|
+ xatom[Manager] = XInternAtom(dpy, "MANAGER", False);
|
||||||
|
+ xatom[Xembed] = XInternAtom(dpy, "_XEMBED", False);
|
||||||
|
+ xatom[XembedInfo] = XInternAtom(dpy, "_XEMBED_INFO", False);
|
||||||
|
+ /* init cursors */
|
||||||
|
cursor[CurNormal] = drw_cur_create(drw, XC_left_ptr);
|
||||||
|
cursor[CurResize] = drw_cur_create(drw, XC_sizing);
|
||||||
|
cursor[CurMove] = drw_cur_create(drw, XC_fleur);
|
||||||
|
@@ -1574,6 +1763,8 @@ setup(void)
|
||||||
|
scheme = ecalloc(LENGTH(colors), sizeof(Clr *));
|
||||||
|
for (i = 0; i < LENGTH(colors); i++)
|
||||||
|
scheme[i] = drw_scm_create(drw, colors[i], 3);
|
||||||
|
+ /* init system tray */
|
||||||
|
+ updatesystray();
|
||||||
|
/* init bars */
|
||||||
|
updatebars();
|
||||||
|
updatestatus();
|
||||||
|
@@ -1707,7 +1898,18 @@ togglebar(const Arg *arg)
|
||||||
|
{
|
||||||
|
selmon->showbar = !selmon->showbar;
|
||||||
|
updatebarpos(selmon);
|
||||||
|
- XMoveResizeWindow(dpy, selmon->barwin, selmon->wx, selmon->by, selmon->ww, bh);
|
||||||
|
+ resizebarwin(selmon);
|
||||||
|
+ if (showsystray) {
|
||||||
|
+ XWindowChanges wc;
|
||||||
|
+ if (!selmon->showbar)
|
||||||
|
+ wc.y = -bh;
|
||||||
|
+ else if (selmon->showbar) {
|
||||||
|
+ wc.y = 0;
|
||||||
|
+ if (!selmon->topbar)
|
||||||
|
+ wc.y = selmon->mh - bh;
|
||||||
|
+ }
|
||||||
|
+ XConfigureWindow(dpy, systray->win, CWY, &wc);
|
||||||
|
+ }
|
||||||
|
arrange(selmon);
|
||||||
|
}
|
||||||
|
|
||||||
|
@@ -1802,11 +2004,18 @@ unmapnotify(XEvent *e)
|
||||||
|
else
|
||||||
|
unmanage(c, 0);
|
||||||
|
}
|
||||||
|
+ else if ((c = wintosystrayicon(ev->window))) {
|
||||||
|
+ /* KLUDGE! sometimes icons occasionally unmap their windows, but do
|
||||||
|
+ * _not_ destroy them. We map those windows back */
|
||||||
|
+ XMapRaised(dpy, c->win);
|
||||||
|
+ updatesystray();
|
||||||
|
+ }
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
updatebars(void)
|
||||||
|
{
|
||||||
|
+ unsigned int w;
|
||||||
|
Monitor *m;
|
||||||
|
XSetWindowAttributes wa = {
|
||||||
|
.override_redirect = True,
|
||||||
|
@@ -1817,10 +2026,15 @@ updatebars(void)
|
||||||
|
for (m = mons; m; m = m->next) {
|
||||||
|
if (m->barwin)
|
||||||
|
continue;
|
||||||
|
- m->barwin = XCreateWindow(dpy, root, m->wx, m->by, m->ww, bh, 0, DefaultDepth(dpy, screen),
|
||||||
|
+ w = m->ww;
|
||||||
|
+ if (showsystray && m == systraytomon(m))
|
||||||
|
+ w -= getsystraywidth();
|
||||||
|
+ m->barwin = XCreateWindow(dpy, root, m->wx, m->by, w, bh, 0, DefaultDepth(dpy, screen),
|
||||||
|
CopyFromParent, DefaultVisual(dpy, screen),
|
||||||
|
CWOverrideRedirect|CWBackPixmap|CWEventMask, &wa);
|
||||||
|
XDefineCursor(dpy, m->barwin, cursor[CurNormal]->cursor);
|
||||||
|
+ if (showsystray && m == systraytomon(m))
|
||||||
|
+ XMapRaised(dpy, systray->win);
|
||||||
|
XMapRaised(dpy, m->barwin);
|
||||||
|
XSetClassHint(dpy, m->barwin, &ch);
|
||||||
|
}
|
||||||
|
@@ -1996,6 +2210,125 @@ updatestatus(void)
|
||||||
|
if (!gettextprop(root, XA_WM_NAME, stext, sizeof(stext)))
|
||||||
|
strcpy(stext, "dwm-"VERSION);
|
||||||
|
drawbar(selmon);
|
||||||
|
+ updatesystray();
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+
|
||||||
|
+void
|
||||||
|
+updatesystrayicongeom(Client *i, int w, int h)
|
||||||
|
+{
|
||||||
|
+ if (i) {
|
||||||
|
+ i->h = bh;
|
||||||
|
+ if (w == h)
|
||||||
|
+ i->w = bh;
|
||||||
|
+ else if (h == bh)
|
||||||
|
+ i->w = w;
|
||||||
|
+ else
|
||||||
|
+ i->w = (int) ((float)bh * ((float)w / (float)h));
|
||||||
|
+ applysizehints(i, &(i->x), &(i->y), &(i->w), &(i->h), False);
|
||||||
|
+ /* force icons into the systray dimensions if they don't want to */
|
||||||
|
+ if (i->h > bh) {
|
||||||
|
+ if (i->w == i->h)
|
||||||
|
+ i->w = bh;
|
||||||
|
+ else
|
||||||
|
+ i->w = (int) ((float)bh * ((float)i->w / (float)i->h));
|
||||||
|
+ i->h = bh;
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+void
|
||||||
|
+updatesystrayiconstate(Client *i, XPropertyEvent *ev)
|
||||||
|
+{
|
||||||
|
+ long flags;
|
||||||
|
+ int code = 0;
|
||||||
|
+
|
||||||
|
+ if (!showsystray || !i || ev->atom != xatom[XembedInfo] ||
|
||||||
|
+ !(flags = getatomprop(i, xatom[XembedInfo])))
|
||||||
|
+ return;
|
||||||
|
+
|
||||||
|
+ if (flags & XEMBED_MAPPED && !i->tags) {
|
||||||
|
+ i->tags = 1;
|
||||||
|
+ code = XEMBED_WINDOW_ACTIVATE;
|
||||||
|
+ XMapRaised(dpy, i->win);
|
||||||
|
+ setclientstate(i, NormalState);
|
||||||
|
+ }
|
||||||
|
+ else if (!(flags & XEMBED_MAPPED) && i->tags) {
|
||||||
|
+ i->tags = 0;
|
||||||
|
+ code = XEMBED_WINDOW_DEACTIVATE;
|
||||||
|
+ XUnmapWindow(dpy, i->win);
|
||||||
|
+ setclientstate(i, WithdrawnState);
|
||||||
|
+ }
|
||||||
|
+ else
|
||||||
|
+ return;
|
||||||
|
+ sendevent(i->win, xatom[Xembed], StructureNotifyMask, CurrentTime, code, 0,
|
||||||
|
+ systray->win, XEMBED_EMBEDDED_VERSION);
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+void
|
||||||
|
+updatesystray(void)
|
||||||
|
+{
|
||||||
|
+ XSetWindowAttributes wa;
|
||||||
|
+ XWindowChanges wc;
|
||||||
|
+ Client *i;
|
||||||
|
+ Monitor *m = systraytomon(NULL);
|
||||||
|
+ unsigned int x = m->mx + m->mw;
|
||||||
|
+ unsigned int sw = TEXTW(stext) - lrpad + systrayspacing;
|
||||||
|
+ unsigned int w = 1;
|
||||||
|
+
|
||||||
|
+ if (!showsystray)
|
||||||
|
+ return;
|
||||||
|
+ if (systrayonleft)
|
||||||
|
+ x -= sw + lrpad / 2;
|
||||||
|
+ if (!systray) {
|
||||||
|
+ /* init systray */
|
||||||
|
+ if (!(systray = (Systray *)calloc(1, sizeof(Systray))))
|
||||||
|
+ die("fatal: could not malloc() %u bytes\n", sizeof(Systray));
|
||||||
|
+ systray->win = XCreateSimpleWindow(dpy, root, x, m->by, w, bh, 0, 0, scheme[SchemeSel][ColBg].pixel);
|
||||||
|
+ wa.event_mask = ButtonPressMask | ExposureMask;
|
||||||
|
+ wa.override_redirect = True;
|
||||||
|
+ wa.background_pixel = scheme[SchemeNorm][ColBg].pixel;
|
||||||
|
+ XSelectInput(dpy, systray->win, SubstructureNotifyMask);
|
||||||
|
+ XChangeProperty(dpy, systray->win, netatom[NetSystemTrayOrientation], XA_CARDINAL, 32,
|
||||||
|
+ PropModeReplace, (unsigned char *)&netatom[NetSystemTrayOrientationHorz], 1);
|
||||||
|
+ XChangeWindowAttributes(dpy, systray->win, CWEventMask|CWOverrideRedirect|CWBackPixel, &wa);
|
||||||
|
+ XMapRaised(dpy, systray->win);
|
||||||
|
+ XSetSelectionOwner(dpy, netatom[NetSystemTray], systray->win, CurrentTime);
|
||||||
|
+ if (XGetSelectionOwner(dpy, netatom[NetSystemTray]) == systray->win) {
|
||||||
|
+ sendevent(root, xatom[Manager], StructureNotifyMask, CurrentTime, netatom[NetSystemTray], systray->win, 0, 0);
|
||||||
|
+ XSync(dpy, False);
|
||||||
|
+ }
|
||||||
|
+ else {
|
||||||
|
+ fprintf(stderr, "dwm: unable to obtain system tray.\n");
|
||||||
|
+ free(systray);
|
||||||
|
+ systray = NULL;
|
||||||
|
+ return;
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+ for (w = 0, i = systray->icons; i; i = i->next) {
|
||||||
|
+ /* make sure the background color stays the same */
|
||||||
|
+ wa.background_pixel = scheme[SchemeNorm][ColBg].pixel;
|
||||||
|
+ XChangeWindowAttributes(dpy, i->win, CWBackPixel, &wa);
|
||||||
|
+ XMapRaised(dpy, i->win);
|
||||||
|
+ w += systrayspacing;
|
||||||
|
+ i->x = w;
|
||||||
|
+ XMoveResizeWindow(dpy, i->win, i->x, 0, i->w, i->h);
|
||||||
|
+ w += i->w;
|
||||||
|
+ if (i->mon != m)
|
||||||
|
+ i->mon = m;
|
||||||
|
+ }
|
||||||
|
+ w = w ? w + systrayspacing : 1;
|
||||||
|
+ x -= w;
|
||||||
|
+ XMoveResizeWindow(dpy, systray->win, x, m->by, w, bh);
|
||||||
|
+ wc.x = x; wc.y = m->by; wc.width = w; wc.height = bh;
|
||||||
|
+ wc.stack_mode = Above; wc.sibling = m->barwin;
|
||||||
|
+ XConfigureWindow(dpy, systray->win, CWX|CWY|CWWidth|CWHeight|CWSibling|CWStackMode, &wc);
|
||||||
|
+ XMapWindow(dpy, systray->win);
|
||||||
|
+ XMapSubwindows(dpy, systray->win);
|
||||||
|
+ /* redraw background */
|
||||||
|
+ XSetForeground(dpy, drw->gc, scheme[SchemeNorm][ColBg].pixel);
|
||||||
|
+ XFillRectangle(dpy, systray->win, drw->gc, 0, 0, w, bh);
|
||||||
|
+ XSync(dpy, False);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
@@ -2063,6 +2396,16 @@ wintoclient(Window w)
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
+Client *
|
||||||
|
+wintosystrayicon(Window w) {
|
||||||
|
+ Client *i = NULL;
|
||||||
|
+
|
||||||
|
+ if (!showsystray || !w)
|
||||||
|
+ return i;
|
||||||
|
+ for (i = systray->icons; i && i->win != w; i = i->next) ;
|
||||||
|
+ return i;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
Monitor *
|
||||||
|
wintomon(Window w)
|
||||||
|
{
|
||||||
|
@@ -2116,6 +2459,22 @@ xerrorstart(Display *dpy, XErrorEvent *ee)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
+Monitor *
|
||||||
|
+systraytomon(Monitor *m) {
|
||||||
|
+ Monitor *t;
|
||||||
|
+ int i, n;
|
||||||
|
+ if(!systraypinning) {
|
||||||
|
+ if(!m)
|
||||||
|
+ return selmon;
|
||||||
|
+ return m == selmon ? m : NULL;
|
||||||
|
+ }
|
||||||
|
+ for(n = 1, t = mons; t && t->next; n++, t = t->next) ;
|
||||||
|
+ for(i = 1, t = mons; t && t->next && i < systraypinning; i++, t = t->next) ;
|
||||||
|
+ if(systraypinningfailfirst && n < systraypinning)
|
||||||
|
+ return mons;
|
||||||
|
+ return t;
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
void
|
||||||
|
zoom(const Arg *arg)
|
||||||
|
{
|
23
dwm.1
23
dwm.1
@@ -30,6 +30,14 @@ top left corner. The tags which are applied to one or more windows are
|
|||||||
indicated with an empty square in the top left corner.
|
indicated with an empty square in the top left corner.
|
||||||
.P
|
.P
|
||||||
dwm draws a small border around windows to indicate the focus state.
|
dwm draws a small border around windows to indicate the focus state.
|
||||||
|
.P
|
||||||
|
On start, dwm can start additional programs that may be specified in two special
|
||||||
|
shell scripts (see the FILES section below), autostart_blocking.sh and
|
||||||
|
autostart.sh. The former is executed first and dwm will wait for its
|
||||||
|
termination before starting. The latter is executed in the background before
|
||||||
|
dwm enters its handler loop.
|
||||||
|
.P
|
||||||
|
Either of these files may be omitted.
|
||||||
.SH OPTIONS
|
.SH OPTIONS
|
||||||
.TP
|
.TP
|
||||||
.B \-v
|
.B \-v
|
||||||
@@ -152,6 +160,21 @@ Toggles focused window between floating and tiled state.
|
|||||||
.TP
|
.TP
|
||||||
.B Mod1\-Button3
|
.B Mod1\-Button3
|
||||||
Resize focused window while dragging. Tiled windows will be toggled to the floating state.
|
Resize focused window while dragging. Tiled windows will be toggled to the floating state.
|
||||||
|
.SH FILES
|
||||||
|
The files containing programs to be started along with dwm are searched for in
|
||||||
|
the following directories:
|
||||||
|
.IP "1. $XDG_DATA_HOME/dwm"
|
||||||
|
.IP "2. $HOME/.local/share/dwm"
|
||||||
|
.IP "3. $HOME/.dwm"
|
||||||
|
.P
|
||||||
|
The first existing directory is scanned for any of the autostart files below.
|
||||||
|
.TP 15
|
||||||
|
autostart.sh
|
||||||
|
This file is started as a shell background process before dwm enters its handler
|
||||||
|
loop.
|
||||||
|
.TP 15
|
||||||
|
autostart_blocking.sh
|
||||||
|
This file is started before any autostart.sh; dwm waits for its termination.
|
||||||
.SH CUSTOMIZATION
|
.SH CUSTOMIZATION
|
||||||
dwm is customized by creating a custom config.h and (re)compiling the source
|
dwm is customized by creating a custom config.h and (re)compiling the source
|
||||||
code. This keeps it fast, secure and simple.
|
code. This keeps it fast, secure and simple.
|
||||||
|
320
ipc.h
320
ipc.h
@@ -1,320 +0,0 @@
|
|||||||
#ifndef IPC_H_
|
|
||||||
#define IPC_H_
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <sys/epoll.h>
|
|
||||||
#include <yajl/yajl_gen.h>
|
|
||||||
|
|
||||||
#include "IPCClient.h"
|
|
||||||
|
|
||||||
// clang-format off
|
|
||||||
#define IPC_MAGIC "DWM-IPC"
|
|
||||||
#define IPC_MAGIC_ARR { 'D', 'W', 'M', '-', 'I', 'P', 'C'}
|
|
||||||
#define IPC_MAGIC_LEN 7 // Not including null char
|
|
||||||
|
|
||||||
#define IPCCOMMAND(FUNC, ARGC, TYPES) \
|
|
||||||
{ #FUNC, {FUNC }, ARGC, (ArgType[ARGC])TYPES }
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
typedef enum IPCMessageType {
|
|
||||||
IPC_TYPE_RUN_COMMAND = 0,
|
|
||||||
IPC_TYPE_GET_MONITORS = 1,
|
|
||||||
IPC_TYPE_GET_TAGS = 2,
|
|
||||||
IPC_TYPE_GET_LAYOUTS = 3,
|
|
||||||
IPC_TYPE_GET_DWM_CLIENT = 4,
|
|
||||||
IPC_TYPE_SUBSCRIBE = 5,
|
|
||||||
IPC_TYPE_EVENT = 6
|
|
||||||
} IPCMessageType;
|
|
||||||
|
|
||||||
typedef enum IPCEvent {
|
|
||||||
IPC_EVENT_TAG_CHANGE = 1 << 0,
|
|
||||||
IPC_EVENT_CLIENT_FOCUS_CHANGE = 1 << 1,
|
|
||||||
IPC_EVENT_LAYOUT_CHANGE = 1 << 2,
|
|
||||||
IPC_EVENT_MONITOR_FOCUS_CHANGE = 1 << 3,
|
|
||||||
IPC_EVENT_FOCUSED_TITLE_CHANGE = 1 << 4,
|
|
||||||
IPC_EVENT_FOCUSED_STATE_CHANGE = 1 << 5
|
|
||||||
} IPCEvent;
|
|
||||||
|
|
||||||
typedef enum IPCSubscriptionAction {
|
|
||||||
IPC_ACTION_UNSUBSCRIBE = 0,
|
|
||||||
IPC_ACTION_SUBSCRIBE = 1
|
|
||||||
} IPCSubscriptionAction;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Every IPC packet starts with this structure
|
|
||||||
*/
|
|
||||||
typedef struct dwm_ipc_header {
|
|
||||||
uint8_t magic[IPC_MAGIC_LEN];
|
|
||||||
uint32_t size;
|
|
||||||
uint8_t type;
|
|
||||||
} __attribute((packed)) dwm_ipc_header_t;
|
|
||||||
|
|
||||||
typedef enum ArgType {
|
|
||||||
ARG_TYPE_NONE = 0,
|
|
||||||
ARG_TYPE_UINT = 1,
|
|
||||||
ARG_TYPE_SINT = 2,
|
|
||||||
ARG_TYPE_FLOAT = 3,
|
|
||||||
ARG_TYPE_PTR = 4,
|
|
||||||
ARG_TYPE_STR = 5
|
|
||||||
} ArgType;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An IPCCommand function can have either of these function signatures
|
|
||||||
*/
|
|
||||||
typedef union ArgFunction {
|
|
||||||
void (*single_param)(const Arg *);
|
|
||||||
void (*array_param)(const Arg *, int);
|
|
||||||
} ArgFunction;
|
|
||||||
|
|
||||||
typedef struct IPCCommand {
|
|
||||||
char *name;
|
|
||||||
ArgFunction func;
|
|
||||||
unsigned int argc;
|
|
||||||
ArgType *arg_types;
|
|
||||||
} IPCCommand;
|
|
||||||
|
|
||||||
typedef struct IPCParsedCommand {
|
|
||||||
char *name;
|
|
||||||
Arg *args;
|
|
||||||
ArgType *arg_types;
|
|
||||||
unsigned int argc;
|
|
||||||
} IPCParsedCommand;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Initialize the IPC socket and the IPC module
|
|
||||||
*
|
|
||||||
* @param socket_path Path to create the socket at
|
|
||||||
* @param epoll_fd File descriptor for epoll
|
|
||||||
* @param commands Address of IPCCommands array defined in config.h
|
|
||||||
* @param commands_len Length of commands[] array
|
|
||||||
*
|
|
||||||
* @return int The file descriptor of the socket if it was successfully created,
|
|
||||||
* -1 otherwise
|
|
||||||
*/
|
|
||||||
int ipc_init(const char *socket_path, const int p_epoll_fd,
|
|
||||||
IPCCommand commands[], const int commands_len);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Uninitialize the socket and module. Free allocated memory and restore static
|
|
||||||
* variables to their state before ipc_init
|
|
||||||
*/
|
|
||||||
void ipc_cleanup();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the file descriptor of the IPC socket
|
|
||||||
*
|
|
||||||
* @return int File descriptor of IPC socket, -1 if socket not created.
|
|
||||||
*/
|
|
||||||
int ipc_get_sock_fd();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get address to IPCClient with specified file descriptor
|
|
||||||
*
|
|
||||||
* @param fd File descriptor of IPC Client
|
|
||||||
*
|
|
||||||
* @return Address to IPCClient with specified file descriptor, -1 otherwise
|
|
||||||
*/
|
|
||||||
IPCClient *ipc_get_client(int fd);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if an IPC client exists with the specified file descriptor
|
|
||||||
*
|
|
||||||
* @param fd File descriptor
|
|
||||||
*
|
|
||||||
* @return int 1 if client exists, 0 otherwise
|
|
||||||
*/
|
|
||||||
int ipc_is_client_registered(int fd);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Disconnect an IPCClient from the socket and remove the client from the list
|
|
||||||
* of known connected clients
|
|
||||||
*
|
|
||||||
* @param c Address of IPCClient
|
|
||||||
*
|
|
||||||
* @return 0 if the client's file descriptor was closed successfully, the
|
|
||||||
* result of executing close() on the file descriptor otherwise.
|
|
||||||
*/
|
|
||||||
int ipc_drop_client(IPCClient *c);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Accept an IPC Client requesting to connect to the socket and add it to the
|
|
||||||
* list of clients
|
|
||||||
*
|
|
||||||
* @return File descriptor of new client, -1 on error
|
|
||||||
*/
|
|
||||||
int ipc_accept_client();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Read an incoming message from an accepted IPC client
|
|
||||||
*
|
|
||||||
* @param c Address of IPCClient
|
|
||||||
* @param msg_type Address to IPCMessageType variable which will be assigned
|
|
||||||
* the message type of the received message
|
|
||||||
* @param msg_size Address to uint32_t variable which will be assigned the size
|
|
||||||
* of the received message
|
|
||||||
* @param msg Address to char* variable which will be assigned the address of
|
|
||||||
* the received message. This must be freed using free().
|
|
||||||
*
|
|
||||||
* @return 0 on success, -1 on error reading message, -2 if reading the message
|
|
||||||
* resulted in EAGAIN, EINTR, or EWOULDBLOCK.
|
|
||||||
*/
|
|
||||||
int ipc_read_client(IPCClient *c, IPCMessageType *msg_type, uint32_t *msg_size,
|
|
||||||
char **msg);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Write any pending buffer of the client to the client's socket
|
|
||||||
*
|
|
||||||
* @param c Client whose buffer to write
|
|
||||||
*
|
|
||||||
* @return Number of bytes written >= 0, -1 otherwise. errno will still be set
|
|
||||||
* from the write operation.
|
|
||||||
*/
|
|
||||||
ssize_t ipc_write_client(IPCClient *c);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Prepare a message in the specified client's buffer.
|
|
||||||
*
|
|
||||||
* @param c Client to prepare message for
|
|
||||||
* @param msg_type Type of message to prepare
|
|
||||||
* @param msg_size Size of the message in bytes. Should not exceed
|
|
||||||
* MAX_MESSAGE_SIZE
|
|
||||||
* @param msg Message to prepare (not including header). This pointer can be
|
|
||||||
* freed after the function invocation.
|
|
||||||
*/
|
|
||||||
void ipc_prepare_send_message(IPCClient *c, const IPCMessageType msg_type,
|
|
||||||
const uint32_t msg_size, const char *msg);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Prepare an error message in the specified client's buffer
|
|
||||||
*
|
|
||||||
* @param c Client to prepare message for
|
|
||||||
* @param msg_type Type of message
|
|
||||||
* @param format Format string following vsprintf
|
|
||||||
* @param ... Arguments for format string
|
|
||||||
*/
|
|
||||||
void ipc_prepare_reply_failure(IPCClient *c, IPCMessageType msg_type,
|
|
||||||
const char *format, ...);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Prepare a success message in the specified client's buffer
|
|
||||||
*
|
|
||||||
* @param c Client to prepare message for
|
|
||||||
* @param msg_type Type of message
|
|
||||||
*/
|
|
||||||
void ipc_prepare_reply_success(IPCClient *c, IPCMessageType msg_type);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a tag_change_event to all subscribers. Should be called only when there
|
|
||||||
* has been a tag state change.
|
|
||||||
*
|
|
||||||
* @param mon_num The index of the monitor (Monitor.num property)
|
|
||||||
* @param old_state The old tag state
|
|
||||||
* @param new_state The new (now current) tag state
|
|
||||||
*/
|
|
||||||
void ipc_tag_change_event(const int mon_num, TagState old_state,
|
|
||||||
TagState new_state);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a client_focus_change_event to all subscribers. Should be called only
|
|
||||||
* when the client focus changes.
|
|
||||||
*
|
|
||||||
* @param mon_num The index of the monitor (Monitor.num property)
|
|
||||||
* @param old_client The old DWM client selection (Monitor.oldsel)
|
|
||||||
* @param new_client The new (now current) DWM client selection
|
|
||||||
*/
|
|
||||||
void ipc_client_focus_change_event(const int mon_num, Client *old_client,
|
|
||||||
Client *new_client);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a layout_change_event to all subscribers. Should be called only
|
|
||||||
* when there has been a layout change.
|
|
||||||
*
|
|
||||||
* @param mon_num The index of the monitor (Monitor.num property)
|
|
||||||
* @param old_symbol The old layout symbol
|
|
||||||
* @param old_layout Address to the old Layout
|
|
||||||
* @param new_symbol The new (now current) layout symbol
|
|
||||||
* @param new_layout Address to the new Layout
|
|
||||||
*/
|
|
||||||
void ipc_layout_change_event(const int mon_num, const char *old_symbol,
|
|
||||||
const Layout *old_layout, const char *new_symbol,
|
|
||||||
const Layout *new_layout);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a monitor_focus_change_event to all subscribers. Should be called only
|
|
||||||
* when the monitor focus changes.
|
|
||||||
*
|
|
||||||
* @param last_mon_num The index of the previously selected monitor
|
|
||||||
* @param new_mon_num The index of the newly selected monitor
|
|
||||||
*/
|
|
||||||
void ipc_monitor_focus_change_event(const int last_mon_num,
|
|
||||||
const int new_mon_num);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a focused_title_change_event to all subscribers. Should only be called
|
|
||||||
* if a selected client has a title change.
|
|
||||||
*
|
|
||||||
* @param mon_num Index of the client's monitor
|
|
||||||
* @param client_id Window XID of client
|
|
||||||
* @param old_name Old name of the client window
|
|
||||||
* @param new_name New name of the client window
|
|
||||||
*/
|
|
||||||
void ipc_focused_title_change_event(const int mon_num, const Window client_id,
|
|
||||||
const char *old_name, const char *new_name);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send a focused_state_change_event to all subscribers. Should only be called
|
|
||||||
* if a selected client has a state change.
|
|
||||||
*
|
|
||||||
* @param mon_num Index of the client's monitor
|
|
||||||
* @param client_id Window XID of client
|
|
||||||
* @param old_state Old state of the client
|
|
||||||
* @param new_state New state of the client
|
|
||||||
*/
|
|
||||||
void ipc_focused_state_change_event(const int mon_num, const Window client_id,
|
|
||||||
const ClientState *old_state,
|
|
||||||
const ClientState *new_state);
|
|
||||||
/**
|
|
||||||
* Check to see if an event has occured and call the *_change_event functions
|
|
||||||
* accordingly
|
|
||||||
*
|
|
||||||
* @param mons Address of Monitor pointing to start of linked list
|
|
||||||
* @param lastselmon Address of pointer to previously selected monitor
|
|
||||||
* @param selmon Address of selected Monitor
|
|
||||||
*/
|
|
||||||
void ipc_send_events(Monitor *mons, Monitor **lastselmon, Monitor *selmon);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle an epoll event caused by a registered IPC client. Read, process, and
|
|
||||||
* handle any received messages from clients. Write pending buffer to client if
|
|
||||||
* the client is ready to receive messages. Drop clients that have sent an
|
|
||||||
* EPOLLHUP.
|
|
||||||
*
|
|
||||||
* @param ev Associated epoll event returned by epoll_wait
|
|
||||||
* @param mons Address of Monitor pointing to start of linked list
|
|
||||||
* @param selmon Address of selected Monitor
|
|
||||||
* @param lastselmon Address of pointer to previously selected monitor
|
|
||||||
* @param tags Array of tag names
|
|
||||||
* @param tags_len Length of tags array
|
|
||||||
* @param layouts Array of available layouts
|
|
||||||
* @param layouts_len Length of layouts array
|
|
||||||
*
|
|
||||||
* @return 0 if event was successfully handled, -1 on any error receiving
|
|
||||||
* or handling incoming messages or unhandled epoll event.
|
|
||||||
*/
|
|
||||||
int ipc_handle_client_epoll_event(struct epoll_event *ev, Monitor *mons,
|
|
||||||
Monitor **lastselmon, Monitor *selmon,
|
|
||||||
const char *tags[], const int tags_len,
|
|
||||||
const Layout *layouts, const int layouts_len);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle an epoll event caused by the IPC socket. This function only handles an
|
|
||||||
* EPOLLIN event indicating a new client requesting to connect to the socket.
|
|
||||||
*
|
|
||||||
* @param ev Associated epoll event returned by epoll_wait
|
|
||||||
*
|
|
||||||
* @return 0, if the event was successfully handled, -1 if not an EPOLLIN event
|
|
||||||
* or if a new IPC client connection request could not be accepted.
|
|
||||||
*/
|
|
||||||
int ipc_handle_socket_epoll_event(struct epoll_event *ev);
|
|
||||||
|
|
||||||
#endif /* IPC_H_ */
|
|
48
movestack.c
Normal file
48
movestack.c
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
void
|
||||||
|
movestack(const Arg *arg) {
|
||||||
|
Client *c = NULL, *p = NULL, *pc = NULL, *i;
|
||||||
|
|
||||||
|
if(arg->i > 0) {
|
||||||
|
/* find the client after selmon->sel */
|
||||||
|
for(c = selmon->sel->next; c && (!ISVISIBLE(c) || c->isfloating); c = c->next);
|
||||||
|
if(!c)
|
||||||
|
for(c = selmon->clients; c && (!ISVISIBLE(c) || c->isfloating); c = c->next);
|
||||||
|
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
/* find the client before selmon->sel */
|
||||||
|
for(i = selmon->clients; i != selmon->sel; i = i->next)
|
||||||
|
if(ISVISIBLE(i) && !i->isfloating)
|
||||||
|
c = i;
|
||||||
|
if(!c)
|
||||||
|
for(; i; i = i->next)
|
||||||
|
if(ISVISIBLE(i) && !i->isfloating)
|
||||||
|
c = i;
|
||||||
|
}
|
||||||
|
/* find the client before selmon->sel and c */
|
||||||
|
for(i = selmon->clients; i && (!p || !pc); i = i->next) {
|
||||||
|
if(i->next == selmon->sel)
|
||||||
|
p = i;
|
||||||
|
if(i->next == c)
|
||||||
|
pc = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* swap c and selmon->sel selmon->clients in the selmon->clients list */
|
||||||
|
if(c && c != selmon->sel) {
|
||||||
|
Client *temp = selmon->sel->next==c?selmon->sel:selmon->sel->next;
|
||||||
|
selmon->sel->next = c->next==selmon->sel?c:c->next;
|
||||||
|
c->next = temp;
|
||||||
|
|
||||||
|
if(p && p != c)
|
||||||
|
p->next = c;
|
||||||
|
if(pc && pc != selmon->sel)
|
||||||
|
pc->next = selmon->sel;
|
||||||
|
|
||||||
|
if(selmon->sel == selmon->clients)
|
||||||
|
selmon->clients = c;
|
||||||
|
else if(c == selmon->clients)
|
||||||
|
selmon->clients = selmon->sel;
|
||||||
|
|
||||||
|
arrange(selmon);
|
||||||
|
}
|
||||||
|
}
|
135
util.c
135
util.c
@@ -3,8 +3,6 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <errno.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
|
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
|
||||||
@@ -35,136 +33,3 @@ die(const char *fmt, ...) {
|
|||||||
|
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
|
||||||
normalizepath(const char *path, char **normal)
|
|
||||||
{
|
|
||||||
size_t len = strlen(path);
|
|
||||||
*normal = (char *)malloc((len + 1) * sizeof(char));
|
|
||||||
const char *walk = path;
|
|
||||||
const char *match;
|
|
||||||
size_t newlen = 0;
|
|
||||||
|
|
||||||
while ((match = strchr(walk, '/'))) {
|
|
||||||
// Copy everything between match and walk
|
|
||||||
strncpy(*normal + newlen, walk, match - walk);
|
|
||||||
newlen += match - walk;
|
|
||||||
walk += match - walk;
|
|
||||||
|
|
||||||
// Skip all repeating slashes
|
|
||||||
while (*walk == '/')
|
|
||||||
walk++;
|
|
||||||
|
|
||||||
// If not last character in path
|
|
||||||
if (walk != path + len)
|
|
||||||
(*normal)[newlen++] = '/';
|
|
||||||
}
|
|
||||||
|
|
||||||
(*normal)[newlen++] = '\0';
|
|
||||||
|
|
||||||
// Copy remaining path
|
|
||||||
strcat(*normal, walk);
|
|
||||||
newlen += strlen(walk);
|
|
||||||
|
|
||||||
*normal = (char *)realloc(*normal, newlen * sizeof(char));
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
parentdir(const char *path, char **parent)
|
|
||||||
{
|
|
||||||
char *normal;
|
|
||||||
char *walk;
|
|
||||||
|
|
||||||
normalizepath(path, &normal);
|
|
||||||
|
|
||||||
// Pointer to last '/'
|
|
||||||
if (!(walk = strrchr(normal, '/'))) {
|
|
||||||
free(normal);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get path up to last '/'
|
|
||||||
size_t len = walk - normal;
|
|
||||||
*parent = (char *)malloc((len + 1) * sizeof(char));
|
|
||||||
|
|
||||||
// Copy path up to last '/'
|
|
||||||
strncpy(*parent, normal, len);
|
|
||||||
// Add null char
|
|
||||||
(*parent)[len] = '\0';
|
|
||||||
|
|
||||||
free(normal);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
mkdirp(const char *path)
|
|
||||||
{
|
|
||||||
char *normal;
|
|
||||||
char *walk;
|
|
||||||
size_t normallen;
|
|
||||||
|
|
||||||
normalizepath(path, &normal);
|
|
||||||
normallen = strlen(normal);
|
|
||||||
walk = normal;
|
|
||||||
|
|
||||||
while (walk < normal + normallen + 1) {
|
|
||||||
// Get length from walk to next /
|
|
||||||
size_t n = strcspn(walk, "/");
|
|
||||||
|
|
||||||
// Skip path /
|
|
||||||
if (n == 0) {
|
|
||||||
walk++;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length of current path segment
|
|
||||||
size_t curpathlen = walk - normal + n;
|
|
||||||
char curpath[curpathlen + 1];
|
|
||||||
struct stat s;
|
|
||||||
|
|
||||||
// Copy path segment to stat
|
|
||||||
strncpy(curpath, normal, curpathlen);
|
|
||||||
strcpy(curpath + curpathlen, "");
|
|
||||||
int res = stat(curpath, &s);
|
|
||||||
|
|
||||||
if (res < 0) {
|
|
||||||
if (errno == ENOENT) {
|
|
||||||
DEBUG("Making directory %s\n", curpath);
|
|
||||||
if (mkdir(curpath, 0700) < 0) {
|
|
||||||
fprintf(stderr, "Failed to make directory %s\n", curpath);
|
|
||||||
perror("");
|
|
||||||
free(normal);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
fprintf(stderr, "Error statting directory %s\n", curpath);
|
|
||||||
perror("");
|
|
||||||
free(normal);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Continue to next path segment
|
|
||||||
walk += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
free(normal);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
nullterminate(char **str, size_t *len)
|
|
||||||
{
|
|
||||||
if ((*str)[*len - 1] == '\0')
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
(*len)++;
|
|
||||||
*str = (char*)realloc(*str, *len * sizeof(char));
|
|
||||||
(*str)[*len - 1] = '\0';
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
10
util.h
10
util.h
@@ -4,15 +4,5 @@
|
|||||||
#define MIN(A, B) ((A) < (B) ? (A) : (B))
|
#define MIN(A, B) ((A) < (B) ? (A) : (B))
|
||||||
#define BETWEEN(X, A, B) ((A) <= (X) && (X) <= (B))
|
#define BETWEEN(X, A, B) ((A) <= (X) && (X) <= (B))
|
||||||
|
|
||||||
#ifdef _DEBUG
|
|
||||||
#define DEBUG(...) fprintf(stderr, __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define DEBUG(...)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void die(const char *fmt, ...);
|
void die(const char *fmt, ...);
|
||||||
void *ecalloc(size_t nmemb, size_t size);
|
void *ecalloc(size_t nmemb, size_t size);
|
||||||
int normalizepath(const char *path, char **normal);
|
|
||||||
int mkdirp(const char *path);
|
|
||||||
int parentdir(const char *path, char **parent);
|
|
||||||
int nullterminate(char **str, size_t *len);
|
|
||||||
|
351
yajl_dumps.c
351
yajl_dumps.c
@@ -1,351 +0,0 @@
|
|||||||
#include "yajl_dumps.h"
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_tag(yajl_gen gen, const char *name, const int tag_mask)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("bit_mask"); YINT(tag_mask);
|
|
||||||
YSTR("name"); YSTR(name);
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_tags(yajl_gen gen, const char *tags[], int tags_len)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YARR(
|
|
||||||
for (int i = 0; i < tags_len; i++)
|
|
||||||
dump_tag(gen, tags[i], 1 << i);
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_client(yajl_gen gen, Client *c)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("name"); YSTR(c->name);
|
|
||||||
YSTR("tags"); YINT(c->tags);
|
|
||||||
YSTR("window_id"); YINT(c->win);
|
|
||||||
YSTR("monitor_number"); YINT(c->mon->num);
|
|
||||||
|
|
||||||
YSTR("geometry"); YMAP(
|
|
||||||
YSTR("current"); YMAP (
|
|
||||||
YSTR("x"); YINT(c->x);
|
|
||||||
YSTR("y"); YINT(c->y);
|
|
||||||
YSTR("width"); YINT(c->w);
|
|
||||||
YSTR("height"); YINT(c->h);
|
|
||||||
)
|
|
||||||
YSTR("old"); YMAP(
|
|
||||||
YSTR("x"); YINT(c->oldx);
|
|
||||||
YSTR("y"); YINT(c->oldy);
|
|
||||||
YSTR("width"); YINT(c->oldw);
|
|
||||||
YSTR("height"); YINT(c->oldh);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("size_hints"); YMAP(
|
|
||||||
YSTR("base"); YMAP(
|
|
||||||
YSTR("width"); YINT(c->basew);
|
|
||||||
YSTR("height"); YINT(c->baseh);
|
|
||||||
)
|
|
||||||
YSTR("step"); YMAP(
|
|
||||||
YSTR("width"); YINT(c->incw);
|
|
||||||
YSTR("height"); YINT(c->inch);
|
|
||||||
)
|
|
||||||
YSTR("max"); YMAP(
|
|
||||||
YSTR("width"); YINT(c->maxw);
|
|
||||||
YSTR("height"); YINT(c->maxh);
|
|
||||||
)
|
|
||||||
YSTR("min"); YMAP(
|
|
||||||
YSTR("width"); YINT(c->minw);
|
|
||||||
YSTR("height"); YINT(c->minh);
|
|
||||||
)
|
|
||||||
YSTR("aspect_ratio"); YMAP(
|
|
||||||
YSTR("min"); YDOUBLE(c->mina);
|
|
||||||
YSTR("max"); YDOUBLE(c->maxa);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("border_width"); YMAP(
|
|
||||||
YSTR("current"); YINT(c->bw);
|
|
||||||
YSTR("old"); YINT(c->oldbw);
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("states"); YMAP(
|
|
||||||
YSTR("is_fixed"); YBOOL(c->isfixed);
|
|
||||||
YSTR("is_floating"); YBOOL(c->isfloating);
|
|
||||||
YSTR("is_urgent"); YBOOL(c->isurgent);
|
|
||||||
YSTR("never_focus"); YBOOL(c->neverfocus);
|
|
||||||
YSTR("old_state"); YBOOL(c->oldstate);
|
|
||||||
YSTR("is_fullscreen"); YBOOL(c->isfullscreen);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_monitor(yajl_gen gen, Monitor *mon, int is_selected)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("master_factor"); YDOUBLE(mon->mfact);
|
|
||||||
YSTR("num_master"); YINT(mon->nmaster);
|
|
||||||
YSTR("num"); YINT(mon->num);
|
|
||||||
YSTR("is_selected"); YBOOL(is_selected);
|
|
||||||
|
|
||||||
YSTR("monitor_geometry"); YMAP(
|
|
||||||
YSTR("x"); YINT(mon->mx);
|
|
||||||
YSTR("y"); YINT(mon->my);
|
|
||||||
YSTR("width"); YINT(mon->mw);
|
|
||||||
YSTR("height"); YINT(mon->mh);
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("window_geometry"); YMAP(
|
|
||||||
YSTR("x"); YINT(mon->wx);
|
|
||||||
YSTR("y"); YINT(mon->wy);
|
|
||||||
YSTR("width"); YINT(mon->ww);
|
|
||||||
YSTR("height"); YINT(mon->wh);
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("tagset"); YMAP(
|
|
||||||
YSTR("current"); YINT(mon->tagset[mon->seltags]);
|
|
||||||
YSTR("old"); YINT(mon->tagset[mon->seltags ^ 1]);
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("tag_state"); dump_tag_state(gen, mon->tagstate);
|
|
||||||
|
|
||||||
YSTR("clients"); YMAP(
|
|
||||||
YSTR("selected"); YINT(mon->sel ? mon->sel->win : 0);
|
|
||||||
YSTR("stack"); YARR(
|
|
||||||
for (Client* c = mon->stack; c; c = c->snext)
|
|
||||||
YINT(c->win);
|
|
||||||
)
|
|
||||||
YSTR("all"); YARR(
|
|
||||||
for (Client* c = mon->clients; c; c = c->next)
|
|
||||||
YINT(c->win);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("layout"); YMAP(
|
|
||||||
YSTR("symbol"); YMAP(
|
|
||||||
YSTR("current"); YSTR(mon->ltsymbol);
|
|
||||||
YSTR("old"); YSTR(mon->lastltsymbol);
|
|
||||||
)
|
|
||||||
YSTR("address"); YMAP(
|
|
||||||
YSTR("current"); YINT((uintptr_t)mon->lt[mon->sellt]);
|
|
||||||
YSTR("old"); YINT((uintptr_t)mon->lt[mon->sellt ^ 1]);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
YSTR("bar"); YMAP(
|
|
||||||
YSTR("y"); YINT(mon->by);
|
|
||||||
YSTR("is_shown"); YBOOL(mon->showbar);
|
|
||||||
YSTR("is_top"); YBOOL(mon->topbar);
|
|
||||||
YSTR("window_id"); YINT(mon->barwin);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_monitors(yajl_gen gen, Monitor *mons, Monitor *selmon)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YARR(
|
|
||||||
for (Monitor *mon = mons; mon; mon = mon->next) {
|
|
||||||
if (mon == selmon)
|
|
||||||
dump_monitor(gen, mon, 1);
|
|
||||||
else
|
|
||||||
dump_monitor(gen, mon, 0);
|
|
||||||
}
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_layouts(yajl_gen gen, const Layout layouts[], const int layouts_len)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YARR(
|
|
||||||
for (int i = 0; i < layouts_len; i++) {
|
|
||||||
YMAP(
|
|
||||||
// Check for a NULL pointer. The cycle layouts patch adds an entry at
|
|
||||||
// the end of the layouts array with a NULL pointer for the symbol
|
|
||||||
YSTR("symbol"); YSTR((layouts[i].symbol ? layouts[i].symbol : ""));
|
|
||||||
YSTR("address"); YINT((uintptr_t)(layouts + i));
|
|
||||||
)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_tag_state(yajl_gen gen, TagState state)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("selected"); YINT(state.selected);
|
|
||||||
YSTR("occupied"); YINT(state.occupied);
|
|
||||||
YSTR("urgent"); YINT(state.urgent);
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_tag_event(yajl_gen gen, int mon_num, TagState old_state,
|
|
||||||
TagState new_state)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("tag_change_event"); YMAP(
|
|
||||||
YSTR("monitor_number"); YINT(mon_num);
|
|
||||||
YSTR("old_state"); dump_tag_state(gen, old_state);
|
|
||||||
YSTR("new_state"); dump_tag_state(gen, new_state);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_client_focus_change_event(yajl_gen gen, Client *old_client,
|
|
||||||
Client *new_client, int mon_num)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("client_focus_change_event"); YMAP(
|
|
||||||
YSTR("monitor_number"); YINT(mon_num);
|
|
||||||
YSTR("old_win_id"); old_client == NULL ? YNULL() : YINT(old_client->win);
|
|
||||||
YSTR("new_win_id"); new_client == NULL ? YNULL() : YINT(new_client->win);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_layout_change_event(yajl_gen gen, const int mon_num,
|
|
||||||
const char *old_symbol, const Layout *old_layout,
|
|
||||||
const char *new_symbol, const Layout *new_layout)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("layout_change_event"); YMAP(
|
|
||||||
YSTR("monitor_number"); YINT(mon_num);
|
|
||||||
YSTR("old_symbol"); YSTR(old_symbol);
|
|
||||||
YSTR("old_address"); YINT((uintptr_t)old_layout);
|
|
||||||
YSTR("new_symbol"); YSTR(new_symbol);
|
|
||||||
YSTR("new_address"); YINT((uintptr_t)new_layout);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_monitor_focus_change_event(yajl_gen gen, const int last_mon_num,
|
|
||||||
const int new_mon_num)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("monitor_focus_change_event"); YMAP(
|
|
||||||
YSTR("old_monitor_number"); YINT(last_mon_num);
|
|
||||||
YSTR("new_monitor_number"); YINT(new_mon_num);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_focused_title_change_event(yajl_gen gen, const int mon_num,
|
|
||||||
const Window client_id, const char *old_name,
|
|
||||||
const char *new_name)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("focused_title_change_event"); YMAP(
|
|
||||||
YSTR("monitor_number"); YINT(mon_num);
|
|
||||||
YSTR("client_window_id"); YINT(client_id);
|
|
||||||
YSTR("old_name"); YSTR(old_name);
|
|
||||||
YSTR("new_name"); YSTR(new_name);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_client_state(yajl_gen gen, const ClientState *state)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("old_state"); YBOOL(state->oldstate);
|
|
||||||
YSTR("is_fixed"); YBOOL(state->isfixed);
|
|
||||||
YSTR("is_floating"); YBOOL(state->isfloating);
|
|
||||||
YSTR("is_fullscreen"); YBOOL(state->isfullscreen);
|
|
||||||
YSTR("is_urgent"); YBOOL(state->isurgent);
|
|
||||||
YSTR("never_focus"); YBOOL(state->neverfocus);
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_focused_state_change_event(yajl_gen gen, const int mon_num,
|
|
||||||
const Window client_id,
|
|
||||||
const ClientState *old_state,
|
|
||||||
const ClientState *new_state)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("focused_state_change_event"); YMAP(
|
|
||||||
YSTR("monitor_number"); YINT(mon_num);
|
|
||||||
YSTR("client_window_id"); YINT(client_id);
|
|
||||||
YSTR("old_state"); dump_client_state(gen, old_state);
|
|
||||||
YSTR("new_state"); dump_client_state(gen, new_state);
|
|
||||||
)
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
dump_error_message(yajl_gen gen, const char *reason)
|
|
||||||
{
|
|
||||||
// clang-format off
|
|
||||||
YMAP(
|
|
||||||
YSTR("result"); YSTR("error");
|
|
||||||
YSTR("reason"); YSTR(reason);
|
|
||||||
)
|
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
65
yajl_dumps.h
65
yajl_dumps.h
@@ -1,65 +0,0 @@
|
|||||||
#ifndef YAJL_DUMPS_H_
|
|
||||||
#define YAJL_DUMPS_H_
|
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
#include <yajl/yajl_gen.h>
|
|
||||||
|
|
||||||
#define YSTR(str) yajl_gen_string(gen, (unsigned char *)str, strlen(str))
|
|
||||||
#define YINT(num) yajl_gen_integer(gen, num)
|
|
||||||
#define YDOUBLE(num) yajl_gen_double(gen, num)
|
|
||||||
#define YBOOL(v) yajl_gen_bool(gen, v)
|
|
||||||
#define YNULL() yajl_gen_null(gen)
|
|
||||||
#define YARR(body) \
|
|
||||||
{ \
|
|
||||||
yajl_gen_array_open(gen); \
|
|
||||||
body; \
|
|
||||||
yajl_gen_array_close(gen); \
|
|
||||||
}
|
|
||||||
#define YMAP(body) \
|
|
||||||
{ \
|
|
||||||
yajl_gen_map_open(gen); \
|
|
||||||
body; \
|
|
||||||
yajl_gen_map_close(gen); \
|
|
||||||
}
|
|
||||||
|
|
||||||
int dump_tag(yajl_gen gen, const char *name, const int tag_mask);
|
|
||||||
|
|
||||||
int dump_tags(yajl_gen gen, const char *tags[], int tags_len);
|
|
||||||
|
|
||||||
int dump_client(yajl_gen gen, Client *c);
|
|
||||||
|
|
||||||
int dump_monitor(yajl_gen gen, Monitor *mon, int is_selected);
|
|
||||||
|
|
||||||
int dump_monitors(yajl_gen gen, Monitor *mons, Monitor *selmon);
|
|
||||||
|
|
||||||
int dump_layouts(yajl_gen gen, const Layout layouts[], const int layouts_len);
|
|
||||||
|
|
||||||
int dump_tag_state(yajl_gen gen, TagState state);
|
|
||||||
|
|
||||||
int dump_tag_event(yajl_gen gen, int mon_num, TagState old_state,
|
|
||||||
TagState new_state);
|
|
||||||
|
|
||||||
int dump_client_focus_change_event(yajl_gen gen, Client *old_client,
|
|
||||||
Client *new_client, int mon_num);
|
|
||||||
|
|
||||||
int dump_layout_change_event(yajl_gen gen, const int mon_num,
|
|
||||||
const char *old_symbol, const Layout *old_layout,
|
|
||||||
const char *new_symbol, const Layout *new_layout);
|
|
||||||
|
|
||||||
int dump_monitor_focus_change_event(yajl_gen gen, const int last_mon_num,
|
|
||||||
const int new_mon_num);
|
|
||||||
|
|
||||||
int dump_focused_title_change_event(yajl_gen gen, const int mon_num,
|
|
||||||
const Window client_id,
|
|
||||||
const char *old_name, const char *new_name);
|
|
||||||
|
|
||||||
int dump_client_state(yajl_gen gen, const ClientState *state);
|
|
||||||
|
|
||||||
int dump_focused_state_change_event(yajl_gen gen, const int mon_num,
|
|
||||||
const Window client_id,
|
|
||||||
const ClientState *old_state,
|
|
||||||
const ClientState *new_state);
|
|
||||||
|
|
||||||
int dump_error_message(yajl_gen gen, const char *reason);
|
|
||||||
|
|
||||||
#endif // YAJL_DUMPS_H_
|
|
Reference in New Issue
Block a user