From 565d3851e5cf089904ead67f8298fbfeee38add5 Mon Sep 17 00:00:00 2001 From: audy Date: Thu, 1 Jun 2023 23:23:30 -0600 Subject: [PATCH] init --- dmenu/LICENSE | 30 + dmenu/Makefile | 64 ++ dmenu/README | 24 + dmenu/arg.h | 49 + dmenu/config.def.h | 23 + dmenu/config.h | 23 + dmenu/config.mk | 31 + dmenu/dmenu.1 | 194 ++++ dmenu/dmenu.c | 782 ++++++++++++++++ dmenu/dmenu_path | 13 + dmenu/dmenu_run | 2 + dmenu/drw.c | 436 +++++++++ dmenu/drw.h | 57 ++ dmenu/stest.1 | 90 ++ dmenu/stest.c | 109 +++ dmenu/util.c | 35 + dmenu/util.h | 8 + dwm/LICENSE | 37 + dwm/Makefile | 51 ++ dwm/README | 48 + dwm/config.def.h | 116 +++ dwm/config.h | 122 +++ dwm/config.mk | 38 + dwm/drw.c | 436 +++++++++ dwm/drw.h | 57 ++ dwm/drw.o | Bin 0 -> 10440 bytes dwm/dwm | Bin 0 -> 66736 bytes dwm/dwm.c | 2152 ++++++++++++++++++++++++++++++++++++++++++++ dwm/dwm.o | Bin 0 -> 56728 bytes dwm/dwm.png | Bin 0 -> 373 bytes dwm/transient.c | 42 + dwm/util.c | 35 + dwm/util.h | 8 + dwm/util.o | Bin 0 -> 2272 bytes 34 files changed, 5112 insertions(+) create mode 100644 dmenu/LICENSE create mode 100644 dmenu/Makefile create mode 100644 dmenu/README create mode 100644 dmenu/arg.h create mode 100644 dmenu/config.def.h create mode 100644 dmenu/config.h create mode 100644 dmenu/config.mk create mode 100644 dmenu/dmenu.1 create mode 100644 dmenu/dmenu.c create mode 100755 dmenu/dmenu_path create mode 100755 dmenu/dmenu_run create mode 100644 dmenu/drw.c create mode 100644 dmenu/drw.h create mode 100644 dmenu/stest.1 create mode 100644 dmenu/stest.c create mode 100644 dmenu/util.c create mode 100644 dmenu/util.h create mode 100644 dwm/LICENSE create mode 100644 dwm/Makefile create mode 100644 dwm/README create mode 100644 dwm/config.def.h create mode 100644 dwm/config.h create mode 100644 dwm/config.mk create mode 100644 dwm/drw.c create mode 100644 dwm/drw.h create mode 100644 dwm/drw.o create mode 100755 dwm/dwm create mode 100644 dwm/dwm.c create mode 100644 dwm/dwm.o create mode 100644 dwm/dwm.png create mode 100644 dwm/transient.c create mode 100644 dwm/util.c create mode 100644 dwm/util.h create mode 100644 dwm/util.o diff --git a/dmenu/LICENSE b/dmenu/LICENSE new file mode 100644 index 0000000..3afd28e --- /dev/null +++ b/dmenu/LICENSE @@ -0,0 +1,30 @@ +MIT/X Consortium License + +© 2006-2019 Anselm R Garbe +© 2006-2008 Sander van Dijk +© 2006-2007 Michał Janeczek +© 2007 Kris Maglione +© 2009 Gottox +© 2009 Markus Schnalke +© 2009 Evan Gates +© 2010-2012 Connor Lane Smith +© 2014-2020 Hiltjo Posthuma +© 2015-2019 Quentin Rameau + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the "Software"), +to deal in the Software without restriction, including without limitation +the rights to use, copy, modify, merge, publish, distribute, sublicense, +and/or sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. diff --git a/dmenu/Makefile b/dmenu/Makefile new file mode 100644 index 0000000..a03a95c --- /dev/null +++ b/dmenu/Makefile @@ -0,0 +1,64 @@ +# dmenu - dynamic menu +# See LICENSE file for copyright and license details. + +include config.mk + +SRC = drw.c dmenu.c stest.c util.c +OBJ = $(SRC:.c=.o) + +all: options dmenu stest + +options: + @echo dmenu build options: + @echo "CFLAGS = $(CFLAGS)" + @echo "LDFLAGS = $(LDFLAGS)" + @echo "CC = $(CC)" + +.c.o: + $(CC) -c $(CFLAGS) $< + +config.h: + cp config.def.h $@ + +$(OBJ): arg.h config.h config.mk drw.h + +dmenu: dmenu.o drw.o util.o + $(CC) -o $@ dmenu.o drw.o util.o $(LDFLAGS) + +stest: stest.o + $(CC) -o $@ stest.o $(LDFLAGS) + +clean: + rm -f dmenu stest $(OBJ) dmenu-$(VERSION).tar.gz + +dist: clean + mkdir -p dmenu-$(VERSION) + cp LICENSE Makefile README arg.h config.def.h config.mk dmenu.1\ + drw.h util.h dmenu_path dmenu_run stest.1 $(SRC)\ + dmenu-$(VERSION) + tar -cf dmenu-$(VERSION).tar dmenu-$(VERSION) + gzip dmenu-$(VERSION).tar + rm -rf dmenu-$(VERSION) + +install: all + mkdir -p $(DESTDIR)$(PREFIX)/bin + cp -f dmenu dmenu_path dmenu_run stest $(DESTDIR)$(PREFIX)/bin + chmod 755 $(DESTDIR)$(PREFIX)/bin/dmenu + chmod 755 $(DESTDIR)$(PREFIX)/bin/dmenu_path + chmod 755 $(DESTDIR)$(PREFIX)/bin/dmenu_run + chmod 755 $(DESTDIR)$(PREFIX)/bin/stest + mkdir -p $(DESTDIR)$(MANPREFIX)/man1 + sed "s/VERSION/$(VERSION)/g" < dmenu.1 > $(DESTDIR)$(MANPREFIX)/man1/dmenu.1 + sed "s/VERSION/$(VERSION)/g" < stest.1 > $(DESTDIR)$(MANPREFIX)/man1/stest.1 + chmod 644 $(DESTDIR)$(MANPREFIX)/man1/dmenu.1 + chmod 644 $(DESTDIR)$(MANPREFIX)/man1/stest.1 + +uninstall: + rm -f $(DESTDIR)$(PREFIX)/bin/dmenu\ + $(DESTDIR)$(PREFIX)/bin/dmenu_path\ + $(DESTDIR)$(PREFIX)/bin/dmenu_run\ + $(DESTDIR)$(PREFIX)/bin/stest\ + $(DESTDIR)$(MANPREFIX)/man1/dmenu.1\ + $(DESTDIR)$(MANPREFIX)/man1/stest.1 + +.PHONY: all options clean dist install uninstall diff --git a/dmenu/README b/dmenu/README new file mode 100644 index 0000000..a8fcdfe --- /dev/null +++ b/dmenu/README @@ -0,0 +1,24 @@ +dmenu - dynamic menu +==================== +dmenu is an efficient dynamic menu for X. + + +Requirements +------------ +In order to build dmenu you need the Xlib header files. + + +Installation +------------ +Edit config.mk to match your local setup (dmenu is installed into +the /usr/local namespace by default). + +Afterwards enter the following command to build and install dmenu +(if necessary as root): + + make clean install + + +Running dmenu +------------- +See the man page for details. diff --git a/dmenu/arg.h b/dmenu/arg.h new file mode 100644 index 0000000..e94e02b --- /dev/null +++ b/dmenu/arg.h @@ -0,0 +1,49 @@ +/* + * Copy me if you can. + * by 20h + */ + +#ifndef ARG_H__ +#define ARG_H__ + +extern char *argv0; + +/* use main(int argc, char *argv[]) */ +#define ARGBEGIN for (argv0 = *argv, argv++, argc--;\ + argv[0] && argv[0][0] == '-'\ + && argv[0][1];\ + argc--, argv++) {\ + char argc_;\ + char **argv_;\ + int brk_;\ + if (argv[0][1] == '-' && argv[0][2] == '\0') {\ + argv++;\ + argc--;\ + break;\ + }\ + for (brk_ = 0, argv[0]++, argv_ = argv;\ + argv[0][0] && !brk_;\ + argv[0]++) {\ + if (argv_ != argv)\ + break;\ + argc_ = argv[0][0];\ + switch (argc_) + +#define ARGEND }\ + } + +#define ARGC() argc_ + +#define EARGF(x) ((argv[0][1] == '\0' && argv[1] == NULL)?\ + ((x), abort(), (char *)0) :\ + (brk_ = 1, (argv[0][1] != '\0')?\ + (&argv[0][1]) :\ + (argc--, argv++, argv[0]))) + +#define ARGF() ((argv[0][1] == '\0' && argv[1] == NULL)?\ + (char *)0 :\ + (brk_ = 1, (argv[0][1] != '\0')?\ + (&argv[0][1]) :\ + (argc--, argv++, argv[0]))) + +#endif diff --git a/dmenu/config.def.h b/dmenu/config.def.h new file mode 100644 index 0000000..1edb647 --- /dev/null +++ b/dmenu/config.def.h @@ -0,0 +1,23 @@ +/* See LICENSE file for copyright and license details. */ +/* Default settings; can be overriden by command line. */ + +static int topbar = 1; /* -b option; if 0, dmenu appears at bottom */ +/* -fn option overrides fonts[0]; default X11 font or font set */ +static const char *fonts[] = { + "monospace:size=10" +}; +static const char *prompt = NULL; /* -p option; prompt to the left of input field */ +static const char *colors[SchemeLast][2] = { + /* fg bg */ + [SchemeNorm] = { "#bbbbbb", "#222222" }, + [SchemeSel] = { "#eeeeee", "#005577" }, + [SchemeOut] = { "#000000", "#00ffff" }, +}; +/* -l option; if nonzero, dmenu uses vertical list with given number of lines */ +static unsigned int lines = 0; + +/* + * Characters not considered part of a word while deleting words + * for example: " /?\"&[]" + */ +static const char worddelimiters[] = " "; diff --git a/dmenu/config.h b/dmenu/config.h new file mode 100644 index 0000000..1edb647 --- /dev/null +++ b/dmenu/config.h @@ -0,0 +1,23 @@ +/* See LICENSE file for copyright and license details. */ +/* Default settings; can be overriden by command line. */ + +static int topbar = 1; /* -b option; if 0, dmenu appears at bottom */ +/* -fn option overrides fonts[0]; default X11 font or font set */ +static const char *fonts[] = { + "monospace:size=10" +}; +static const char *prompt = NULL; /* -p option; prompt to the left of input field */ +static const char *colors[SchemeLast][2] = { + /* fg bg */ + [SchemeNorm] = { "#bbbbbb", "#222222" }, + [SchemeSel] = { "#eeeeee", "#005577" }, + [SchemeOut] = { "#000000", "#00ffff" }, +}; +/* -l option; if nonzero, dmenu uses vertical list with given number of lines */ +static unsigned int lines = 0; + +/* + * Characters not considered part of a word while deleting words + * for example: " /?\"&[]" + */ +static const char worddelimiters[] = " "; diff --git a/dmenu/config.mk b/dmenu/config.mk new file mode 100644 index 0000000..05d5a3e --- /dev/null +++ b/dmenu/config.mk @@ -0,0 +1,31 @@ +# dmenu version +VERSION = 5.0 + +# paths +PREFIX = /usr/local +MANPREFIX = $(PREFIX)/share/man + +X11INC = /usr/X11R6/include +X11LIB = /usr/X11R6/lib + +# Xinerama, comment if you don't want it +XINERAMALIBS = -lXinerama +XINERAMAFLAGS = -DXINERAMA + +# freetype +FREETYPELIBS = -lfontconfig -lXft +FREETYPEINC = /usr/include/freetype2 +# OpenBSD (uncomment) +#FREETYPEINC = $(X11INC)/freetype2 + +# includes and libs +INCS = -I$(X11INC) -I$(FREETYPEINC) +LIBS = -L$(X11LIB) -lX11 $(XINERAMALIBS) $(FREETYPELIBS) + +# flags +CPPFLAGS = -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_XOPEN_SOURCE=700 -D_POSIX_C_SOURCE=200809L -DVERSION=\"$(VERSION)\" $(XINERAMAFLAGS) +CFLAGS = -std=c99 -pedantic -Wall -Os $(INCS) $(CPPFLAGS) +LDFLAGS = $(LIBS) + +# compiler and linker +CC = cc diff --git a/dmenu/dmenu.1 b/dmenu/dmenu.1 new file mode 100644 index 0000000..323f93c --- /dev/null +++ b/dmenu/dmenu.1 @@ -0,0 +1,194 @@ +.TH DMENU 1 dmenu\-VERSION +.SH NAME +dmenu \- dynamic menu +.SH SYNOPSIS +.B dmenu +.RB [ \-bfiv ] +.RB [ \-l +.IR lines ] +.RB [ \-m +.IR monitor ] +.RB [ \-p +.IR prompt ] +.RB [ \-fn +.IR font ] +.RB [ \-nb +.IR color ] +.RB [ \-nf +.IR color ] +.RB [ \-sb +.IR color ] +.RB [ \-sf +.IR color ] +.RB [ \-w +.IR windowid ] +.P +.BR dmenu_run " ..." +.SH DESCRIPTION +.B dmenu +is a dynamic menu for X, which reads a list of newline\-separated items from +stdin. When the user selects an item and presses Return, their choice is printed +to stdout and dmenu terminates. Entering text will narrow the items to those +matching the tokens in the input. +.P +.B dmenu_run +is a script used by +.IR dwm (1) +which lists programs in the user's $PATH and runs the result in their $SHELL. +.SH OPTIONS +.TP +.B \-b +dmenu appears at the bottom of the screen. +.TP +.B \-f +dmenu grabs the keyboard before reading stdin if not reading from a tty. This +is faster, but will lock up X until stdin reaches end\-of\-file. +.TP +.B \-i +dmenu matches menu items case insensitively. +.TP +.BI \-l " lines" +dmenu lists items vertically, with the given number of lines. +.TP +.BI \-m " monitor" +dmenu is displayed on the monitor number supplied. Monitor numbers are starting +from 0. +.TP +.BI \-p " prompt" +defines the prompt to be displayed to the left of the input field. +.TP +.BI \-fn " font" +defines the font or font set used. +.TP +.BI \-nb " color" +defines the normal background color. +.IR #RGB , +.IR #RRGGBB , +and X color names are supported. +.TP +.BI \-nf " color" +defines the normal foreground color. +.TP +.BI \-sb " color" +defines the selected background color. +.TP +.BI \-sf " color" +defines the selected foreground color. +.TP +.B \-v +prints version information to stdout, then exits. +.TP +.BI \-w " windowid" +embed into windowid. +.SH USAGE +dmenu is completely controlled by the keyboard. Items are selected using the +arrow keys, page up, page down, home, and end. +.TP +.B Tab +Copy the selected item to the input field. +.TP +.B Return +Confirm selection. Prints the selected item to stdout and exits, returning +success. +.TP +.B Ctrl-Return +Confirm selection. Prints the selected item to stdout and continues. +.TP +.B Shift\-Return +Confirm input. Prints the input text to stdout and exits, returning success. +.TP +.B Escape +Exit without selecting an item, returning failure. +.TP +.B Ctrl-Left +Move cursor to the start of the current word +.TP +.B Ctrl-Right +Move cursor to the end of the current word +.TP +.B C\-a +Home +.TP +.B C\-b +Left +.TP +.B C\-c +Escape +.TP +.B C\-d +Delete +.TP +.B C\-e +End +.TP +.B C\-f +Right +.TP +.B C\-g +Escape +.TP +.B C\-h +Backspace +.TP +.B C\-i +Tab +.TP +.B C\-j +Return +.TP +.B C\-J +Shift-Return +.TP +.B C\-k +Delete line right +.TP +.B C\-m +Return +.TP +.B C\-M +Shift-Return +.TP +.B C\-n +Down +.TP +.B C\-p +Up +.TP +.B C\-u +Delete line left +.TP +.B C\-w +Delete word left +.TP +.B C\-y +Paste from primary X selection +.TP +.B C\-Y +Paste from X clipboard +.TP +.B M\-b +Move cursor to the start of the current word +.TP +.B M\-f +Move cursor to the end of the current word +.TP +.B M\-g +Home +.TP +.B M\-G +End +.TP +.B M\-h +Up +.TP +.B M\-j +Page down +.TP +.B M\-k +Page up +.TP +.B M\-l +Down +.SH SEE ALSO +.IR dwm (1), +.IR stest (1) diff --git a/dmenu/dmenu.c b/dmenu/dmenu.c new file mode 100644 index 0000000..98507d9 --- /dev/null +++ b/dmenu/dmenu.c @@ -0,0 +1,782 @@ +/* See LICENSE file for copyright and license details. */ +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#ifdef XINERAMA +#include +#endif +#include + +#include "drw.h" +#include "util.h" + +/* macros */ +#define INTERSECT(x,y,w,h,r) (MAX(0, MIN((x)+(w),(r).x_org+(r).width) - MAX((x),(r).x_org)) \ + * MAX(0, MIN((y)+(h),(r).y_org+(r).height) - MAX((y),(r).y_org))) +#define LENGTH(X) (sizeof X / sizeof X[0]) +#define TEXTW(X) (drw_fontset_getwidth(drw, (X)) + lrpad) + +/* enums */ +enum { SchemeNorm, SchemeSel, SchemeOut, SchemeLast }; /* color schemes */ + +struct item { + char *text; + struct item *left, *right; + int out; +}; + +static char text[BUFSIZ] = ""; +static char *embed; +static int bh, mw, mh; +static int inputw = 0, promptw; +static int lrpad; /* sum of left and right padding */ +static size_t cursor; +static struct item *items = NULL; +static struct item *matches, *matchend; +static struct item *prev, *curr, *next, *sel; +static int mon = -1, screen; + +static Atom clip, utf8; +static Display *dpy; +static Window root, parentwin, win; +static XIC xic; + +static Drw *drw; +static Clr *scheme[SchemeLast]; + +#include "config.h" + +static int (*fstrncmp)(const char *, const char *, size_t) = strncmp; +static char *(*fstrstr)(const char *, const char *) = strstr; + +static void +appenditem(struct item *item, struct item **list, struct item **last) +{ + if (*last) + (*last)->right = item; + else + *list = item; + + item->left = *last; + item->right = NULL; + *last = item; +} + +static void +calcoffsets(void) +{ + int i, n; + + if (lines > 0) + n = lines * bh; + else + n = mw - (promptw + inputw + TEXTW("<") + TEXTW(">")); + /* calculate which items will begin the next page and previous page */ + for (i = 0, next = curr; next; next = next->right) + if ((i += (lines > 0) ? bh : MIN(TEXTW(next->text), n)) > n) + break; + for (i = 0, prev = curr; prev && prev->left; prev = prev->left) + if ((i += (lines > 0) ? bh : MIN(TEXTW(prev->left->text), n)) > n) + break; +} + +static void +cleanup(void) +{ + size_t i; + + XUngrabKey(dpy, AnyKey, AnyModifier, root); + for (i = 0; i < SchemeLast; i++) + free(scheme[i]); + drw_free(drw); + XSync(dpy, False); + XCloseDisplay(dpy); +} + +static char * +cistrstr(const char *s, const char *sub) +{ + size_t len; + + for (len = strlen(sub); *s; s++) + if (!strncasecmp(s, sub, len)) + return (char *)s; + return NULL; +} + +static int +drawitem(struct item *item, int x, int y, int w) +{ + if (item == sel) + drw_setscheme(drw, scheme[SchemeSel]); + else if (item->out) + drw_setscheme(drw, scheme[SchemeOut]); + else + drw_setscheme(drw, scheme[SchemeNorm]); + + return drw_text(drw, x, y, w, bh, lrpad / 2, item->text, 0); +} + +static void +drawmenu(void) +{ + unsigned int curpos; + struct item *item; + int x = 0, y = 0, w; + + drw_setscheme(drw, scheme[SchemeNorm]); + drw_rect(drw, 0, 0, mw, mh, 1, 1); + + if (prompt && *prompt) { + drw_setscheme(drw, scheme[SchemeSel]); + x = drw_text(drw, x, 0, promptw, bh, lrpad / 2, prompt, 0); + } + /* draw input field */ + w = (lines > 0 || !matches) ? mw - x : inputw; + drw_setscheme(drw, scheme[SchemeNorm]); + drw_text(drw, x, 0, w, bh, lrpad / 2, text, 0); + + curpos = TEXTW(text) - TEXTW(&text[cursor]); + if ((curpos += lrpad / 2 - 1) < w) { + drw_setscheme(drw, scheme[SchemeNorm]); + drw_rect(drw, x + curpos, 2, 2, bh - 4, 1, 0); + } + + if (lines > 0) { + /* draw vertical list */ + for (item = curr; item != next; item = item->right) + drawitem(item, x, y += bh, mw - x); + } else if (matches) { + /* draw horizontal list */ + x += inputw; + w = TEXTW("<"); + if (curr->left) { + drw_setscheme(drw, scheme[SchemeNorm]); + drw_text(drw, x, 0, w, bh, lrpad / 2, "<", 0); + } + x += w; + for (item = curr; item != next; item = item->right) + x = drawitem(item, x, 0, MIN(TEXTW(item->text), mw - x - TEXTW(">"))); + if (next) { + w = TEXTW(">"); + drw_setscheme(drw, scheme[SchemeNorm]); + drw_text(drw, mw - w, 0, w, bh, lrpad / 2, ">", 0); + } + } + drw_map(drw, win, 0, 0, mw, mh); +} + +static void +grabfocus(void) +{ + struct timespec ts = { .tv_sec = 0, .tv_nsec = 10000000 }; + Window focuswin; + int i, revertwin; + + for (i = 0; i < 100; ++i) { + XGetInputFocus(dpy, &focuswin, &revertwin); + if (focuswin == win) + return; + XSetInputFocus(dpy, win, RevertToParent, CurrentTime); + nanosleep(&ts, NULL); + } + die("cannot grab focus"); +} + +static void +grabkeyboard(void) +{ + struct timespec ts = { .tv_sec = 0, .tv_nsec = 1000000 }; + int i; + + if (embed) + return; + /* try to grab keyboard, we may have to wait for another process to ungrab */ + for (i = 0; i < 1000; i++) { + if (XGrabKeyboard(dpy, DefaultRootWindow(dpy), True, GrabModeAsync, + GrabModeAsync, CurrentTime) == GrabSuccess) + return; + nanosleep(&ts, NULL); + } + die("cannot grab keyboard"); +} + +static void +match(void) +{ + static char **tokv = NULL; + static int tokn = 0; + + char buf[sizeof text], *s; + int i, tokc = 0; + size_t len, textsize; + struct item *item, *lprefix, *lsubstr, *prefixend, *substrend; + + strcpy(buf, text); + /* separate input text into tokens to be matched individually */ + for (s = strtok(buf, " "); s; tokv[tokc - 1] = s, s = strtok(NULL, " ")) + if (++tokc > tokn && !(tokv = realloc(tokv, ++tokn * sizeof *tokv))) + die("cannot realloc %u bytes:", tokn * sizeof *tokv); + len = tokc ? strlen(tokv[0]) : 0; + + matches = lprefix = lsubstr = matchend = prefixend = substrend = NULL; + textsize = strlen(text) + 1; + for (item = items; item && item->text; item++) { + for (i = 0; i < tokc; i++) + if (!fstrstr(item->text, tokv[i])) + break; + if (i != tokc) /* not all tokens match */ + continue; + /* exact matches go first, then prefixes, then substrings */ + if (!tokc || !fstrncmp(text, item->text, textsize)) + appenditem(item, &matches, &matchend); + else if (!fstrncmp(tokv[0], item->text, len)) + appenditem(item, &lprefix, &prefixend); + else + appenditem(item, &lsubstr, &substrend); + } + if (lprefix) { + if (matches) { + matchend->right = lprefix; + lprefix->left = matchend; + } else + matches = lprefix; + matchend = prefixend; + } + if (lsubstr) { + if (matches) { + matchend->right = lsubstr; + lsubstr->left = matchend; + } else + matches = lsubstr; + matchend = substrend; + } + curr = sel = matches; + calcoffsets(); +} + +static void +insert(const char *str, ssize_t n) +{ + if (strlen(text) + n > sizeof text - 1) + return; + /* move existing text out of the way, insert new text, and update cursor */ + memmove(&text[cursor + n], &text[cursor], sizeof text - cursor - MAX(n, 0)); + if (n > 0) + memcpy(&text[cursor], str, n); + cursor += n; + match(); +} + +static size_t +nextrune(int inc) +{ + ssize_t n; + + /* return location of next utf8 rune in the given direction (+1 or -1) */ + for (n = cursor + inc; n + inc >= 0 && (text[n] & 0xc0) == 0x80; n += inc) + ; + return n; +} + +static void +movewordedge(int dir) +{ + if (dir < 0) { /* move cursor to the start of the word*/ + while (cursor > 0 && strchr(worddelimiters, text[nextrune(-1)])) + cursor = nextrune(-1); + while (cursor > 0 && !strchr(worddelimiters, text[nextrune(-1)])) + cursor = nextrune(-1); + } else { /* move cursor to the end of the word */ + while (text[cursor] && strchr(worddelimiters, text[cursor])) + cursor = nextrune(+1); + while (text[cursor] && !strchr(worddelimiters, text[cursor])) + cursor = nextrune(+1); + } +} + +static void +keypress(XKeyEvent *ev) +{ + char buf[32]; + int len; + KeySym ksym; + Status status; + + len = XmbLookupString(xic, ev, buf, sizeof buf, &ksym, &status); + switch (status) { + default: /* XLookupNone, XBufferOverflow */ + return; + case XLookupChars: + goto insert; + case XLookupKeySym: + case XLookupBoth: + break; + } + + if (ev->state & ControlMask) { + switch(ksym) { + case XK_a: ksym = XK_Home; break; + case XK_b: ksym = XK_Left; break; + case XK_c: ksym = XK_Escape; break; + case XK_d: ksym = XK_Delete; break; + case XK_e: ksym = XK_End; break; + case XK_f: ksym = XK_Right; break; + case XK_g: ksym = XK_Escape; break; + case XK_h: ksym = XK_BackSpace; break; + case XK_i: ksym = XK_Tab; break; + case XK_j: /* fallthrough */ + case XK_J: /* fallthrough */ + case XK_m: /* fallthrough */ + case XK_M: ksym = XK_Return; ev->state &= ~ControlMask; break; + case XK_n: ksym = XK_Down; break; + case XK_p: ksym = XK_Up; break; + + case XK_k: /* delete right */ + text[cursor] = '\0'; + match(); + break; + case XK_u: /* delete left */ + insert(NULL, 0 - cursor); + break; + case XK_w: /* delete word */ + while (cursor > 0 && strchr(worddelimiters, text[nextrune(-1)])) + insert(NULL, nextrune(-1) - cursor); + while (cursor > 0 && !strchr(worddelimiters, text[nextrune(-1)])) + insert(NULL, nextrune(-1) - cursor); + break; + case XK_y: /* paste selection */ + case XK_Y: + XConvertSelection(dpy, (ev->state & ShiftMask) ? clip : XA_PRIMARY, + utf8, utf8, win, CurrentTime); + return; + case XK_Left: + case XK_KP_Left: + movewordedge(-1); + goto draw; + case XK_Right: + case XK_KP_Right: + movewordedge(+1); + goto draw; + case XK_Return: + case XK_KP_Enter: + break; + case XK_bracketleft: + cleanup(); + exit(1); + default: + return; + } + } else if (ev->state & Mod1Mask) { + switch(ksym) { + case XK_b: + movewordedge(-1); + goto draw; + case XK_f: + movewordedge(+1); + goto draw; + case XK_g: ksym = XK_Home; break; + case XK_G: ksym = XK_End; break; + case XK_h: ksym = XK_Up; break; + case XK_j: ksym = XK_Next; break; + case XK_k: ksym = XK_Prior; break; + case XK_l: ksym = XK_Down; break; + default: + return; + } + } + + switch(ksym) { + default: +insert: + if (!iscntrl(*buf)) + insert(buf, len); + break; + case XK_Delete: + case XK_KP_Delete: + if (text[cursor] == '\0') + return; + cursor = nextrune(+1); + /* fallthrough */ + case XK_BackSpace: + if (cursor == 0) + return; + insert(NULL, nextrune(-1) - cursor); + break; + case XK_End: + case XK_KP_End: + if (text[cursor] != '\0') { + cursor = strlen(text); + break; + } + if (next) { + /* jump to end of list and position items in reverse */ + curr = matchend; + calcoffsets(); + curr = prev; + calcoffsets(); + while (next && (curr = curr->right)) + calcoffsets(); + } + sel = matchend; + break; + case XK_Escape: + cleanup(); + exit(1); + case XK_Home: + case XK_KP_Home: + if (sel == matches) { + cursor = 0; + break; + } + sel = curr = matches; + calcoffsets(); + break; + case XK_Left: + case XK_KP_Left: + if (cursor > 0 && (!sel || !sel->left || lines > 0)) { + cursor = nextrune(-1); + break; + } + if (lines > 0) + return; + /* fallthrough */ + case XK_Up: + case XK_KP_Up: + if (sel && sel->left && (sel = sel->left)->right == curr) { + curr = prev; + calcoffsets(); + } + break; + case XK_Next: + case XK_KP_Next: + if (!next) + return; + sel = curr = next; + calcoffsets(); + break; + case XK_Prior: + case XK_KP_Prior: + if (!prev) + return; + sel = curr = prev; + calcoffsets(); + break; + case XK_Return: + case XK_KP_Enter: + puts((sel && !(ev->state & ShiftMask)) ? sel->text : text); + if (!(ev->state & ControlMask)) { + cleanup(); + exit(0); + } + if (sel) + sel->out = 1; + break; + case XK_Right: + case XK_KP_Right: + if (text[cursor] != '\0') { + cursor = nextrune(+1); + break; + } + if (lines > 0) + return; + /* fallthrough */ + case XK_Down: + case XK_KP_Down: + if (sel && sel->right && (sel = sel->right) == next) { + curr = next; + calcoffsets(); + } + break; + case XK_Tab: + if (!sel) + return; + strncpy(text, sel->text, sizeof text - 1); + text[sizeof text - 1] = '\0'; + cursor = strlen(text); + match(); + break; + } + +draw: + drawmenu(); +} + +static void +paste(void) +{ + char *p, *q; + int di; + unsigned long dl; + Atom da; + + /* we have been given the current selection, now insert it into input */ + if (XGetWindowProperty(dpy, win, utf8, 0, (sizeof text / 4) + 1, False, + utf8, &da, &di, &dl, &dl, (unsigned char **)&p) + == Success && p) { + insert(p, (q = strchr(p, '\n')) ? q - p : (ssize_t)strlen(p)); + XFree(p); + } + drawmenu(); +} + +static void +readstdin(void) +{ + char buf[sizeof text], *p; + size_t i, imax = 0, size = 0; + unsigned int tmpmax = 0; + + /* read each line from stdin and add it to the item list */ + for (i = 0; fgets(buf, sizeof buf, stdin); i++) { + if (i + 1 >= size / sizeof *items) + if (!(items = realloc(items, (size += BUFSIZ)))) + die("cannot realloc %u bytes:", size); + if ((p = strchr(buf, '\n'))) + *p = '\0'; + if (!(items[i].text = strdup(buf))) + die("cannot strdup %u bytes:", strlen(buf) + 1); + items[i].out = 0; + drw_font_getexts(drw->fonts, buf, strlen(buf), &tmpmax, NULL); + if (tmpmax > inputw) { + inputw = tmpmax; + imax = i; + } + } + if (items) + items[i].text = NULL; + inputw = items ? TEXTW(items[imax].text) : 0; + lines = MIN(lines, i); +} + +static void +run(void) +{ + XEvent ev; + + while (!XNextEvent(dpy, &ev)) { + if (XFilterEvent(&ev, win)) + continue; + switch(ev.type) { + case DestroyNotify: + if (ev.xdestroywindow.window != win) + break; + cleanup(); + exit(1); + case Expose: + if (ev.xexpose.count == 0) + drw_map(drw, win, 0, 0, mw, mh); + break; + case FocusIn: + /* regrab focus from parent window */ + if (ev.xfocus.window != win) + grabfocus(); + break; + case KeyPress: + keypress(&ev.xkey); + break; + case SelectionNotify: + if (ev.xselection.property == utf8) + paste(); + break; + case VisibilityNotify: + if (ev.xvisibility.state != VisibilityUnobscured) + XRaiseWindow(dpy, win); + break; + } + } +} + +static void +setup(void) +{ + int x, y, i, j; + unsigned int du; + XSetWindowAttributes swa; + XIM xim; + Window w, dw, *dws; + XWindowAttributes wa; + XClassHint ch = {"dmenu", "dmenu"}; +#ifdef XINERAMA + XineramaScreenInfo *info; + Window pw; + int a, di, n, area = 0; +#endif + /* init appearance */ + for (j = 0; j < SchemeLast; j++) + scheme[j] = drw_scm_create(drw, colors[j], 2); + + clip = XInternAtom(dpy, "CLIPBOARD", False); + utf8 = XInternAtom(dpy, "UTF8_STRING", False); + + /* calculate menu geometry */ + bh = drw->fonts->h + 2; + lines = MAX(lines, 0); + mh = (lines + 1) * bh; +#ifdef XINERAMA + i = 0; + if (parentwin == root && (info = XineramaQueryScreens(dpy, &n))) { + XGetInputFocus(dpy, &w, &di); + if (mon >= 0 && mon < n) + i = mon; + else if (w != root && w != PointerRoot && w != None) { + /* find top-level window containing current input focus */ + do { + if (XQueryTree(dpy, (pw = w), &dw, &w, &dws, &du) && dws) + XFree(dws); + } while (w != root && w != pw); + /* find xinerama screen with which the window intersects most */ + if (XGetWindowAttributes(dpy, pw, &wa)) + for (j = 0; j < n; j++) + if ((a = INTERSECT(wa.x, wa.y, wa.width, wa.height, info[j])) > area) { + area = a; + i = j; + } + } + /* no focused window is on screen, so use pointer location instead */ + if (mon < 0 && !area && XQueryPointer(dpy, root, &dw, &dw, &x, &y, &di, &di, &du)) + for (i = 0; i < n; i++) + if (INTERSECT(x, y, 1, 1, info[i])) + break; + + x = info[i].x_org; + y = info[i].y_org + (topbar ? 0 : info[i].height - mh); + mw = info[i].width; + XFree(info); + } else +#endif + { + if (!XGetWindowAttributes(dpy, parentwin, &wa)) + die("could not get embedding window attributes: 0x%lx", + parentwin); + x = 0; + y = topbar ? 0 : wa.height - mh; + mw = wa.width; + } + promptw = (prompt && *prompt) ? TEXTW(prompt) - lrpad / 4 : 0; + inputw = MIN(inputw, mw/3); + match(); + + /* create menu window */ + swa.override_redirect = True; + swa.background_pixel = scheme[SchemeNorm][ColBg].pixel; + swa.event_mask = ExposureMask | KeyPressMask | VisibilityChangeMask; + win = XCreateWindow(dpy, parentwin, x, y, mw, mh, 0, + CopyFromParent, CopyFromParent, CopyFromParent, + CWOverrideRedirect | CWBackPixel | CWEventMask, &swa); + XSetClassHint(dpy, win, &ch); + + + /* input methods */ + if ((xim = XOpenIM(dpy, NULL, NULL, NULL)) == NULL) + die("XOpenIM failed: could not open input device"); + + xic = XCreateIC(xim, XNInputStyle, XIMPreeditNothing | XIMStatusNothing, + XNClientWindow, win, XNFocusWindow, win, NULL); + + XMapRaised(dpy, win); + if (embed) { + XSelectInput(dpy, parentwin, FocusChangeMask | SubstructureNotifyMask); + if (XQueryTree(dpy, parentwin, &dw, &w, &dws, &du) && dws) { + for (i = 0; i < du && dws[i] != win; ++i) + XSelectInput(dpy, dws[i], FocusChangeMask); + XFree(dws); + } + grabfocus(); + } + drw_resize(drw, mw, mh); + drawmenu(); +} + +static void +usage(void) +{ + fputs("usage: dmenu [-bfiv] [-l lines] [-p prompt] [-fn font] [-m monitor]\n" + " [-nb color] [-nf color] [-sb color] [-sf color] [-w windowid]\n", stderr); + exit(1); +} + +int +main(int argc, char *argv[]) +{ + XWindowAttributes wa; + int i, fast = 0; + + for (i = 1; i < argc; i++) + /* these options take no arguments */ + if (!strcmp(argv[i], "-v")) { /* prints version information */ + puts("dmenu-"VERSION); + exit(0); + } else if (!strcmp(argv[i], "-b")) /* appears at the bottom of the screen */ + topbar = 0; + else if (!strcmp(argv[i], "-f")) /* grabs keyboard before reading stdin */ + fast = 1; + else if (!strcmp(argv[i], "-i")) { /* case-insensitive item matching */ + fstrncmp = strncasecmp; + fstrstr = cistrstr; + } else if (i + 1 == argc) + usage(); + /* these options take one argument */ + else if (!strcmp(argv[i], "-l")) /* number of lines in vertical list */ + lines = atoi(argv[++i]); + else if (!strcmp(argv[i], "-m")) + mon = atoi(argv[++i]); + else if (!strcmp(argv[i], "-p")) /* adds prompt to left of input field */ + prompt = argv[++i]; + else if (!strcmp(argv[i], "-fn")) /* font or font set */ + fonts[0] = argv[++i]; + else if (!strcmp(argv[i], "-nb")) /* normal background color */ + colors[SchemeNorm][ColBg] = argv[++i]; + else if (!strcmp(argv[i], "-nf")) /* normal foreground color */ + colors[SchemeNorm][ColFg] = argv[++i]; + else if (!strcmp(argv[i], "-sb")) /* selected background color */ + colors[SchemeSel][ColBg] = argv[++i]; + else if (!strcmp(argv[i], "-sf")) /* selected foreground color */ + colors[SchemeSel][ColFg] = argv[++i]; + else if (!strcmp(argv[i], "-w")) /* embedding window id */ + embed = argv[++i]; + else + usage(); + + if (!setlocale(LC_CTYPE, "") || !XSupportsLocale()) + fputs("warning: no locale support\n", stderr); + if (!(dpy = XOpenDisplay(NULL))) + die("cannot open display"); + screen = DefaultScreen(dpy); + root = RootWindow(dpy, screen); + if (!embed || !(parentwin = strtol(embed, NULL, 0))) + parentwin = root; + if (!XGetWindowAttributes(dpy, parentwin, &wa)) + die("could not get embedding window attributes: 0x%lx", + parentwin); + drw = drw_create(dpy, screen, root, wa.width, wa.height); + if (!drw_fontset_create(drw, fonts, LENGTH(fonts))) + die("no fonts could be loaded."); + lrpad = drw->fonts->h; + +#ifdef __OpenBSD__ + if (pledge("stdio rpath", NULL) == -1) + die("pledge"); +#endif + + if (fast && !isatty(0)) { + grabkeyboard(); + readstdin(); + } else { + readstdin(); + grabkeyboard(); + } + setup(); + run(); + + return 1; /* unreachable */ +} diff --git a/dmenu/dmenu_path b/dmenu/dmenu_path new file mode 100755 index 0000000..3a7cda7 --- /dev/null +++ b/dmenu/dmenu_path @@ -0,0 +1,13 @@ +#!/bin/sh + +cachedir="${XDG_CACHE_HOME:-"$HOME/.cache"}" +cache="$cachedir/dmenu_run" + +[ ! -e "$cachedir" ] && mkdir -p "$cachedir" + +IFS=: +if stest -dqr -n "$cache" $PATH; then + stest -flx $PATH | sort -u | tee "$cache" +else + cat "$cache" +fi diff --git a/dmenu/dmenu_run b/dmenu/dmenu_run new file mode 100755 index 0000000..834ede5 --- /dev/null +++ b/dmenu/dmenu_run @@ -0,0 +1,2 @@ +#!/bin/sh +dmenu_path | dmenu "$@" | ${SHELL:-"/bin/sh"} & diff --git a/dmenu/drw.c b/dmenu/drw.c new file mode 100644 index 0000000..4cdbcbe --- /dev/null +++ b/dmenu/drw.c @@ -0,0 +1,436 @@ +/* See LICENSE file for copyright and license details. */ +#include +#include +#include +#include +#include + +#include "drw.h" +#include "util.h" + +#define UTF_INVALID 0xFFFD +#define UTF_SIZ 4 + +static const unsigned char utfbyte[UTF_SIZ + 1] = {0x80, 0, 0xC0, 0xE0, 0xF0}; +static const unsigned char utfmask[UTF_SIZ + 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; +static const long utfmin[UTF_SIZ + 1] = { 0, 0, 0x80, 0x800, 0x10000}; +static const long utfmax[UTF_SIZ + 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; + +static long +utf8decodebyte(const char c, size_t *i) +{ + for (*i = 0; *i < (UTF_SIZ + 1); ++(*i)) + if (((unsigned char)c & utfmask[*i]) == utfbyte[*i]) + return (unsigned char)c & ~utfmask[*i]; + return 0; +} + +static size_t +utf8validate(long *u, size_t i) +{ + if (!BETWEEN(*u, utfmin[i], utfmax[i]) || BETWEEN(*u, 0xD800, 0xDFFF)) + *u = UTF_INVALID; + for (i = 1; *u > utfmax[i]; ++i) + ; + return i; +} + +static size_t +utf8decode(const char *c, long *u, size_t clen) +{ + size_t i, j, len, type; + long udecoded; + + *u = UTF_INVALID; + if (!clen) + return 0; + udecoded = utf8decodebyte(c[0], &len); + if (!BETWEEN(len, 1, UTF_SIZ)) + return 1; + for (i = 1, j = 1; i < clen && j < len; ++i, ++j) { + udecoded = (udecoded << 6) | utf8decodebyte(c[i], &type); + if (type) + return j; + } + if (j < len) + return 0; + *u = udecoded; + utf8validate(u, len); + + return len; +} + +Drw * +drw_create(Display *dpy, int screen, Window root, unsigned int w, unsigned int h) +{ + Drw *drw = ecalloc(1, sizeof(Drw)); + + drw->dpy = dpy; + drw->screen = screen; + drw->root = root; + drw->w = w; + drw->h = h; + drw->drawable = XCreatePixmap(dpy, root, w, h, DefaultDepth(dpy, screen)); + drw->gc = XCreateGC(dpy, root, 0, NULL); + XSetLineAttributes(dpy, drw->gc, 1, LineSolid, CapButt, JoinMiter); + + return drw; +} + +void +drw_resize(Drw *drw, unsigned int w, unsigned int h) +{ + if (!drw) + return; + + drw->w = w; + drw->h = h; + if (drw->drawable) + XFreePixmap(drw->dpy, drw->drawable); + drw->drawable = XCreatePixmap(drw->dpy, drw->root, w, h, DefaultDepth(drw->dpy, drw->screen)); +} + +void +drw_free(Drw *drw) +{ + XFreePixmap(drw->dpy, drw->drawable); + XFreeGC(drw->dpy, drw->gc); + drw_fontset_free(drw->fonts); + free(drw); +} + +/* This function is an implementation detail. Library users should use + * drw_fontset_create instead. + */ +static Fnt * +xfont_create(Drw *drw, const char *fontname, FcPattern *fontpattern) +{ + Fnt *font; + XftFont *xfont = NULL; + FcPattern *pattern = NULL; + + if (fontname) { + /* Using the pattern found at font->xfont->pattern does not yield the + * same substitution results as using the pattern returned by + * FcNameParse; using the latter results in the desired fallback + * behaviour whereas the former just results in missing-character + * rectangles being drawn, at least with some fonts. */ + if (!(xfont = XftFontOpenName(drw->dpy, drw->screen, fontname))) { + fprintf(stderr, "error, cannot load font from name: '%s'\n", fontname); + return NULL; + } + if (!(pattern = FcNameParse((FcChar8 *) fontname))) { + fprintf(stderr, "error, cannot parse font name to pattern: '%s'\n", fontname); + XftFontClose(drw->dpy, xfont); + return NULL; + } + } else if (fontpattern) { + if (!(xfont = XftFontOpenPattern(drw->dpy, fontpattern))) { + fprintf(stderr, "error, cannot load font from pattern.\n"); + return NULL; + } + } else { + die("no font specified."); + } + + /* Do not allow using color fonts. This is a workaround for a BadLength + * error from Xft with color glyphs. Modelled on the Xterm workaround. See + * https://bugzilla.redhat.com/show_bug.cgi?id=1498269 + * https://lists.suckless.org/dev/1701/30932.html + * https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=916349 + * and lots more all over the internet. + */ + FcBool iscol; + if(FcPatternGetBool(xfont->pattern, FC_COLOR, 0, &iscol) == FcResultMatch && iscol) { + XftFontClose(drw->dpy, xfont); + return NULL; + } + + font = ecalloc(1, sizeof(Fnt)); + font->xfont = xfont; + font->pattern = pattern; + font->h = xfont->ascent + xfont->descent; + font->dpy = drw->dpy; + + return font; +} + +static void +xfont_free(Fnt *font) +{ + if (!font) + return; + if (font->pattern) + FcPatternDestroy(font->pattern); + XftFontClose(font->dpy, font->xfont); + free(font); +} + +Fnt* +drw_fontset_create(Drw* drw, const char *fonts[], size_t fontcount) +{ + Fnt *cur, *ret = NULL; + size_t i; + + if (!drw || !fonts) + return NULL; + + for (i = 1; i <= fontcount; i++) { + if ((cur = xfont_create(drw, fonts[fontcount - i], NULL))) { + cur->next = ret; + ret = cur; + } + } + return (drw->fonts = ret); +} + +void +drw_fontset_free(Fnt *font) +{ + if (font) { + drw_fontset_free(font->next); + xfont_free(font); + } +} + +void +drw_clr_create(Drw *drw, Clr *dest, const char *clrname) +{ + if (!drw || !dest || !clrname) + return; + + if (!XftColorAllocName(drw->dpy, DefaultVisual(drw->dpy, drw->screen), + DefaultColormap(drw->dpy, drw->screen), + clrname, dest)) + die("error, cannot allocate color '%s'", clrname); +} + +/* Wrapper to create color schemes. The caller has to call free(3) on the + * returned color scheme when done using it. */ +Clr * +drw_scm_create(Drw *drw, const char *clrnames[], size_t clrcount) +{ + size_t i; + Clr *ret; + + /* need at least two colors for a scheme */ + if (!drw || !clrnames || clrcount < 2 || !(ret = ecalloc(clrcount, sizeof(XftColor)))) + return NULL; + + for (i = 0; i < clrcount; i++) + drw_clr_create(drw, &ret[i], clrnames[i]); + return ret; +} + +void +drw_setfontset(Drw *drw, Fnt *set) +{ + if (drw) + drw->fonts = set; +} + +void +drw_setscheme(Drw *drw, Clr *scm) +{ + if (drw) + drw->scheme = scm; +} + +void +drw_rect(Drw *drw, int x, int y, unsigned int w, unsigned int h, int filled, int invert) +{ + if (!drw || !drw->scheme) + return; + XSetForeground(drw->dpy, drw->gc, invert ? drw->scheme[ColBg].pixel : drw->scheme[ColFg].pixel); + if (filled) + XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h); + else + XDrawRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w - 1, h - 1); +} + +int +drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lpad, const char *text, int invert) +{ + char buf[1024]; + int ty; + unsigned int ew; + XftDraw *d = NULL; + Fnt *usedfont, *curfont, *nextfont; + size_t i, len; + int utf8strlen, utf8charlen, render = x || y || w || h; + long utf8codepoint = 0; + const char *utf8str; + FcCharSet *fccharset; + FcPattern *fcpattern; + FcPattern *match; + XftResult result; + int charexists = 0; + + if (!drw || (render && !drw->scheme) || !text || !drw->fonts) + return 0; + + if (!render) { + w = ~w; + } else { + XSetForeground(drw->dpy, drw->gc, drw->scheme[invert ? ColFg : ColBg].pixel); + XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h); + d = XftDrawCreate(drw->dpy, drw->drawable, + DefaultVisual(drw->dpy, drw->screen), + DefaultColormap(drw->dpy, drw->screen)); + x += lpad; + w -= lpad; + } + + usedfont = drw->fonts; + while (1) { + utf8strlen = 0; + utf8str = text; + nextfont = NULL; + while (*text) { + utf8charlen = utf8decode(text, &utf8codepoint, UTF_SIZ); + for (curfont = drw->fonts; curfont; curfont = curfont->next) { + charexists = charexists || XftCharExists(drw->dpy, curfont->xfont, utf8codepoint); + if (charexists) { + if (curfont == usedfont) { + utf8strlen += utf8charlen; + text += utf8charlen; + } else { + nextfont = curfont; + } + break; + } + } + + if (!charexists || nextfont) + break; + else + charexists = 0; + } + + if (utf8strlen) { + drw_font_getexts(usedfont, utf8str, utf8strlen, &ew, NULL); + /* shorten text if necessary */ + for (len = MIN(utf8strlen, sizeof(buf) - 1); len && ew > w; len--) + drw_font_getexts(usedfont, utf8str, len, &ew, NULL); + + if (len) { + memcpy(buf, utf8str, len); + buf[len] = '\0'; + if (len < utf8strlen) + for (i = len; i && i > len - 3; buf[--i] = '.') + ; /* NOP */ + + if (render) { + ty = y + (h - usedfont->h) / 2 + usedfont->xfont->ascent; + XftDrawStringUtf8(d, &drw->scheme[invert ? ColBg : ColFg], + usedfont->xfont, x, ty, (XftChar8 *)buf, len); + } + x += ew; + w -= ew; + } + } + + if (!*text) { + break; + } else if (nextfont) { + charexists = 0; + usedfont = nextfont; + } else { + /* Regardless of whether or not a fallback font is found, the + * character must be drawn. */ + charexists = 1; + + fccharset = FcCharSetCreate(); + FcCharSetAddChar(fccharset, utf8codepoint); + + if (!drw->fonts->pattern) { + /* Refer to the comment in xfont_create for more information. */ + die("the first font in the cache must be loaded from a font string."); + } + + fcpattern = FcPatternDuplicate(drw->fonts->pattern); + FcPatternAddCharSet(fcpattern, FC_CHARSET, fccharset); + FcPatternAddBool(fcpattern, FC_SCALABLE, FcTrue); + FcPatternAddBool(fcpattern, FC_COLOR, FcFalse); + + FcConfigSubstitute(NULL, fcpattern, FcMatchPattern); + FcDefaultSubstitute(fcpattern); + match = XftFontMatch(drw->dpy, drw->screen, fcpattern, &result); + + FcCharSetDestroy(fccharset); + FcPatternDestroy(fcpattern); + + if (match) { + usedfont = xfont_create(drw, NULL, match); + if (usedfont && XftCharExists(drw->dpy, usedfont->xfont, utf8codepoint)) { + for (curfont = drw->fonts; curfont->next; curfont = curfont->next) + ; /* NOP */ + curfont->next = usedfont; + } else { + xfont_free(usedfont); + usedfont = drw->fonts; + } + } + } + } + if (d) + XftDrawDestroy(d); + + return x + (render ? w : 0); +} + +void +drw_map(Drw *drw, Window win, int x, int y, unsigned int w, unsigned int h) +{ + if (!drw) + return; + + XCopyArea(drw->dpy, drw->drawable, win, drw->gc, x, y, w, h, x, y); + XSync(drw->dpy, False); +} + +unsigned int +drw_fontset_getwidth(Drw *drw, const char *text) +{ + if (!drw || !drw->fonts || !text) + return 0; + return drw_text(drw, 0, 0, 0, 0, 0, text, 0); +} + +void +drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h) +{ + XGlyphInfo ext; + + if (!font || !text) + return; + + XftTextExtentsUtf8(font->dpy, font->xfont, (XftChar8 *)text, len, &ext); + if (w) + *w = ext.xOff; + if (h) + *h = font->h; +} + +Cur * +drw_cur_create(Drw *drw, int shape) +{ + Cur *cur; + + if (!drw || !(cur = ecalloc(1, sizeof(Cur)))) + return NULL; + + cur->cursor = XCreateFontCursor(drw->dpy, shape); + + return cur; +} + +void +drw_cur_free(Drw *drw, Cur *cursor) +{ + if (!cursor) + return; + + XFreeCursor(drw->dpy, cursor->cursor); + free(cursor); +} diff --git a/dmenu/drw.h b/dmenu/drw.h new file mode 100644 index 0000000..4c67419 --- /dev/null +++ b/dmenu/drw.h @@ -0,0 +1,57 @@ +/* See LICENSE file for copyright and license details. */ + +typedef struct { + Cursor cursor; +} Cur; + +typedef struct Fnt { + Display *dpy; + unsigned int h; + XftFont *xfont; + FcPattern *pattern; + struct Fnt *next; +} Fnt; + +enum { ColFg, ColBg }; /* Clr scheme index */ +typedef XftColor Clr; + +typedef struct { + unsigned int w, h; + Display *dpy; + int screen; + Window root; + Drawable drawable; + GC gc; + Clr *scheme; + Fnt *fonts; +} Drw; + +/* Drawable abstraction */ +Drw *drw_create(Display *dpy, int screen, Window win, unsigned int w, unsigned int h); +void drw_resize(Drw *drw, unsigned int w, unsigned int h); +void drw_free(Drw *drw); + +/* Fnt abstraction */ +Fnt *drw_fontset_create(Drw* drw, const char *fonts[], size_t fontcount); +void drw_fontset_free(Fnt* set); +unsigned int drw_fontset_getwidth(Drw *drw, const char *text); +void drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h); + +/* Colorscheme abstraction */ +void drw_clr_create(Drw *drw, Clr *dest, const char *clrname); +Clr *drw_scm_create(Drw *drw, const char *clrnames[], size_t clrcount); + +/* Cursor abstraction */ +Cur *drw_cur_create(Drw *drw, int shape); +void drw_cur_free(Drw *drw, Cur *cursor); + +/* Drawing context manipulation */ +void drw_setfontset(Drw *drw, Fnt *set); +void drw_setscheme(Drw *drw, Clr *scm); + +/* Drawing functions */ +void drw_rect(Drw *drw, int x, int y, unsigned int w, unsigned int h, int filled, int invert); +int drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lpad, const char *text, int invert); + +/* Map functions */ +void drw_map(Drw *drw, Window win, int x, int y, unsigned int w, unsigned int h); diff --git a/dmenu/stest.1 b/dmenu/stest.1 new file mode 100644 index 0000000..2667d8a --- /dev/null +++ b/dmenu/stest.1 @@ -0,0 +1,90 @@ +.TH STEST 1 dmenu\-VERSION +.SH NAME +stest \- filter a list of files by properties +.SH SYNOPSIS +.B stest +.RB [ -abcdefghlpqrsuwx ] +.RB [ -n +.IR file ] +.RB [ -o +.IR file ] +.RI [ file ...] +.SH DESCRIPTION +.B stest +takes a list of files and filters by the files' properties, analogous to +.IR test (1). +Files which pass all tests are printed to stdout. If no files are given, stest +reads files from stdin. +.SH OPTIONS +.TP +.B \-a +Test hidden files. +.TP +.B \-b +Test that files are block specials. +.TP +.B \-c +Test that files are character specials. +.TP +.B \-d +Test that files are directories. +.TP +.B \-e +Test that files exist. +.TP +.B \-f +Test that files are regular files. +.TP +.B \-g +Test that files have their set-group-ID flag set. +.TP +.B \-h +Test that files are symbolic links. +.TP +.B \-l +Test the contents of a directory given as an argument. +.TP +.BI \-n " file" +Test that files are newer than +.IR file . +.TP +.BI \-o " file" +Test that files are older than +.IR file . +.TP +.B \-p +Test that files are named pipes. +.TP +.B \-q +No files are printed, only the exit status is returned. +.TP +.B \-r +Test that files are readable. +.TP +.B \-s +Test that files are not empty. +.TP +.B \-u +Test that files have their set-user-ID flag set. +.TP +.B \-v +Invert the sense of tests, only failing files pass. +.TP +.B \-w +Test that files are writable. +.TP +.B \-x +Test that files are executable. +.SH EXIT STATUS +.TP +.B 0 +At least one file passed all tests. +.TP +.B 1 +No files passed all tests. +.TP +.B 2 +An error occurred. +.SH SEE ALSO +.IR dmenu (1), +.IR test (1) diff --git a/dmenu/stest.c b/dmenu/stest.c new file mode 100644 index 0000000..e27d3a5 --- /dev/null +++ b/dmenu/stest.c @@ -0,0 +1,109 @@ +/* See LICENSE file for copyright and license details. */ +#include + +#include +#include +#include +#include +#include +#include + +#include "arg.h" +char *argv0; + +#define FLAG(x) (flag[(x)-'a']) + +static void test(const char *, const char *); +static void usage(void); + +static int match = 0; +static int flag[26]; +static struct stat old, new; + +static void +test(const char *path, const char *name) +{ + struct stat st, ln; + + if ((!stat(path, &st) && (FLAG('a') || name[0] != '.') /* hidden files */ + && (!FLAG('b') || S_ISBLK(st.st_mode)) /* block special */ + && (!FLAG('c') || S_ISCHR(st.st_mode)) /* character special */ + && (!FLAG('d') || S_ISDIR(st.st_mode)) /* directory */ + && (!FLAG('e') || access(path, F_OK) == 0) /* exists */ + && (!FLAG('f') || S_ISREG(st.st_mode)) /* regular file */ + && (!FLAG('g') || st.st_mode & S_ISGID) /* set-group-id flag */ + && (!FLAG('h') || (!lstat(path, &ln) && S_ISLNK(ln.st_mode))) /* symbolic link */ + && (!FLAG('n') || st.st_mtime > new.st_mtime) /* newer than file */ + && (!FLAG('o') || st.st_mtime < old.st_mtime) /* older than file */ + && (!FLAG('p') || S_ISFIFO(st.st_mode)) /* named pipe */ + && (!FLAG('r') || access(path, R_OK) == 0) /* readable */ + && (!FLAG('s') || st.st_size > 0) /* not empty */ + && (!FLAG('u') || st.st_mode & S_ISUID) /* set-user-id flag */ + && (!FLAG('w') || access(path, W_OK) == 0) /* writable */ + && (!FLAG('x') || access(path, X_OK) == 0)) != FLAG('v')) { /* executable */ + if (FLAG('q')) + exit(0); + match = 1; + puts(name); + } +} + +static void +usage(void) +{ + fprintf(stderr, "usage: %s [-abcdefghlpqrsuvwx] " + "[-n file] [-o file] [file...]\n", argv0); + exit(2); /* like test(1) return > 1 on error */ +} + +int +main(int argc, char *argv[]) +{ + struct dirent *d; + char path[PATH_MAX], *line = NULL, *file; + size_t linesiz = 0; + ssize_t n; + DIR *dir; + int r; + + ARGBEGIN { + case 'n': /* newer than file */ + case 'o': /* older than file */ + file = EARGF(usage()); + if (!(FLAG(ARGC()) = !stat(file, (ARGC() == 'n' ? &new : &old)))) + perror(file); + break; + default: + /* miscellaneous operators */ + if (strchr("abcdefghlpqrsuvwx", ARGC())) + FLAG(ARGC()) = 1; + else + usage(); /* unknown flag */ + } ARGEND; + + if (!argc) { + /* read list from stdin */ + while ((n = getline(&line, &linesiz, stdin)) > 0) { + if (line[n - 1] == '\n') + line[n - 1] = '\0'; + test(line, line); + } + free(line); + } else { + for (; argc; argc--, argv++) { + if (FLAG('l') && (dir = opendir(*argv))) { + /* test directory contents */ + while ((d = readdir(dir))) { + r = snprintf(path, sizeof path, "%s/%s", + *argv, d->d_name); + if (r >= 0 && (size_t)r < sizeof path) + test(path, d->d_name); + } + closedir(dir); + } else { + test(*argv, *argv); + } + } + } + return match ? 0 : 1; +} diff --git a/dmenu/util.c b/dmenu/util.c new file mode 100644 index 0000000..fe044fc --- /dev/null +++ b/dmenu/util.c @@ -0,0 +1,35 @@ +/* See LICENSE file for copyright and license details. */ +#include +#include +#include +#include + +#include "util.h" + +void * +ecalloc(size_t nmemb, size_t size) +{ + void *p; + + if (!(p = calloc(nmemb, size))) + die("calloc:"); + return p; +} + +void +die(const char *fmt, ...) { + va_list ap; + + va_start(ap, fmt); + vfprintf(stderr, fmt, ap); + va_end(ap); + + if (fmt[0] && fmt[strlen(fmt)-1] == ':') { + fputc(' ', stderr); + perror(NULL); + } else { + fputc('\n', stderr); + } + + exit(1); +} diff --git a/dmenu/util.h b/dmenu/util.h new file mode 100644 index 0000000..f633b51 --- /dev/null +++ b/dmenu/util.h @@ -0,0 +1,8 @@ +/* See LICENSE file for copyright and license details. */ + +#define MAX(A, B) ((A) > (B) ? (A) : (B)) +#define MIN(A, B) ((A) < (B) ? (A) : (B)) +#define BETWEEN(X, A, B) ((A) <= (X) && (X) <= (B)) + +void die(const char *fmt, ...); +void *ecalloc(size_t nmemb, size_t size); diff --git a/dwm/LICENSE b/dwm/LICENSE new file mode 100644 index 0000000..d221f09 --- /dev/null +++ b/dwm/LICENSE @@ -0,0 +1,37 @@ +MIT/X Consortium License + +© 2006-2019 Anselm R Garbe +© 2006-2009 Jukka Salmi +© 2006-2007 Sander van Dijk +© 2007-2011 Peter Hartlich +© 2007-2009 Szabolcs Nagy +© 2007-2009 Christof Musik +© 2007-2009 Premysl Hruby +© 2007-2008 Enno Gottox Boland +© 2008 Martin Hurton +© 2008 Neale Pickett +© 2009 Mate Nagy +© 2010-2016 Hiltjo Posthuma +© 2010-2012 Connor Lane Smith +© 2011 Christoph Lohmann <20h@r-36.net> +© 2015-2016 Quentin Rameau +© 2015-2016 Eric Pruitt +© 2016-2017 Markus Teich + +Permission is hereby granted, free of charge, to any person obtaining a +copy of this software and associated documentation files (the "Software"), +to deal in the Software without restriction, including without limitation +the rights to use, copy, modify, merge, publish, distribute, sublicense, +and/or sell copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. diff --git a/dwm/Makefile b/dwm/Makefile new file mode 100644 index 0000000..77bcbc0 --- /dev/null +++ b/dwm/Makefile @@ -0,0 +1,51 @@ +# dwm - dynamic window manager +# See LICENSE file for copyright and license details. + +include config.mk + +SRC = drw.c dwm.c util.c +OBJ = ${SRC:.c=.o} + +all: options dwm + +options: + @echo dwm build options: + @echo "CFLAGS = ${CFLAGS}" + @echo "LDFLAGS = ${LDFLAGS}" + @echo "CC = ${CC}" + +.c.o: + ${CC} -c ${CFLAGS} $< + +${OBJ}: config.h config.mk + +config.h: + cp config.def.h $@ + +dwm: ${OBJ} + ${CC} -o $@ ${OBJ} ${LDFLAGS} + +clean: + rm -f dwm ${OBJ} dwm-${VERSION}.tar.gz + +dist: clean + mkdir -p dwm-${VERSION} + cp -R LICENSE Makefile README config.def.h config.mk\ + dwm.1 drw.h util.h ${SRC} dwm.png transient.c dwm-${VERSION} + tar -cf dwm-${VERSION}.tar dwm-${VERSION} + gzip dwm-${VERSION}.tar + rm -rf dwm-${VERSION} + +install: all + mkdir -p ${DESTDIR}${PREFIX}/bin + cp -f dwm ${DESTDIR}${PREFIX}/bin + chmod 755 ${DESTDIR}${PREFIX}/bin/dwm + mkdir -p ${DESTDIR}${MANPREFIX}/man1 + sed "s/VERSION/${VERSION}/g" < dwm.1 > ${DESTDIR}${MANPREFIX}/man1/dwm.1 + chmod 644 ${DESTDIR}${MANPREFIX}/man1/dwm.1 + +uninstall: + rm -f ${DESTDIR}${PREFIX}/bin/dwm\ + ${DESTDIR}${MANPREFIX}/man1/dwm.1 + +.PHONY: all options clean dist install uninstall diff --git a/dwm/README b/dwm/README new file mode 100644 index 0000000..95d4fd0 --- /dev/null +++ b/dwm/README @@ -0,0 +1,48 @@ +dwm - dynamic window manager +============================ +dwm is an extremely fast, small, and dynamic window manager for X. + + +Requirements +------------ +In order to build dwm you need the Xlib header files. + + +Installation +------------ +Edit config.mk to match your local setup (dwm is installed into +the /usr/local namespace by default). + +Afterwards enter the following command to build and install dwm (if +necessary as root): + + make clean install + + +Running dwm +----------- +Add the following line to your .xinitrc to start dwm using startx: + + exec dwm + +In order to connect dwm to a specific display, make sure that +the DISPLAY environment variable is set correctly, e.g.: + + DISPLAY=foo.bar:1 exec dwm + +(This will start dwm on display :1 of the host foo.bar.) + +In order to display status info in the bar, you can do something +like this in your .xinitrc: + + while xsetroot -name "`date` `uptime | sed 's/.*,//'`" + do + sleep 1 + done & + exec dwm + + +Configuration +------------- +The configuration of dwm is done by creating a custom config.h +and (re)compiling the source code. diff --git a/dwm/config.def.h b/dwm/config.def.h new file mode 100644 index 0000000..a2ac963 --- /dev/null +++ b/dwm/config.def.h @@ -0,0 +1,116 @@ +/* See LICENSE file for copyright and license details. */ + +/* 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 char *fonts[] = { "monospace:size=10" }; +static const char dmenufont[] = "monospace:size=10"; +static const char col_gray1[] = "#222222"; +static const char col_gray2[] = "#444444"; +static const char col_gray3[] = "#bbbbbb"; +static const char col_gray4[] = "#eeeeee"; +static const char col_cyan[] = "#005577"; +static const char *colors[][3] = { + /* fg bg border */ + [SchemeNorm] = { col_gray3, col_gray1, col_gray2 }, + [SchemeSel] = { col_gray4, col_cyan, col_cyan }, +}; + +/* tagging */ +static const char *tags[] = { "1", "2", "3", "4", "5", "6", "7", "8", "9" }; + +static const Rule rules[] = { + /* xprop(1): + * WM_CLASS(STRING) = instance, class + * WM_NAME(STRING) = title + */ + /* class instance title tags mask isfloating monitor */ + { "Gimp", NULL, NULL, 0, 1, -1 }, + { "Firefox", NULL, NULL, 1 << 8, 0, -1 }, +}; + +/* layout(s) */ +static const float mfact = 0.55; /* factor of master area size [0.05..0.95] */ +static const int nmaster = 1; /* number of clients in master area */ +static const int resizehints = 1; /* 1 means respect size hints in tiled resizals */ +static const int lockfullscreen = 1; /* 1 will force focus on the fullscreen window */ + +static const Layout layouts[] = { + /* symbol arrange function */ + { "[]=", tile }, /* first entry is default */ + { "><>", NULL }, /* no layout function means floating behavior */ + { "[M]", monocle }, +}; + +/* key definitions */ +#define MODKEY Mod1Mask +#define TAGKEYS(KEY,TAG) \ + { MODKEY, KEY, view, {.ui = 1 << TAG} }, \ + { MODKEY|ControlMask, KEY, toggleview, {.ui = 1 << TAG} }, \ + { MODKEY|ShiftMask, KEY, tag, {.ui = 1 << TAG} }, \ + { MODKEY|ControlMask|ShiftMask, KEY, toggletag, {.ui = 1 << TAG} }, + +/* helper for spawning shell commands in the pre dwm-5.0 fashion */ +#define SHCMD(cmd) { .v = (const char*[]){ "/bin/sh", "-c", cmd, NULL } } + +/* commands */ +static char dmenumon[2] = "0"; /* component of dmenucmd, manipulated in spawn() */ +static const char *dmenucmd[] = { "dmenu_run", "-m", dmenumon, "-fn", dmenufont, "-nb", col_gray1, "-nf", col_gray3, "-sb", col_cyan, "-sf", col_gray4, NULL }; +static const char *termcmd[] = { "st", NULL }; + +static Key keys[] = { + /* modifier key function argument */ + { MODKEY, XK_p, spawn, {.v = dmenucmd } }, + { MODKEY|ShiftMask, XK_Return, spawn, {.v = termcmd } }, + { MODKEY, XK_b, togglebar, {0} }, + { MODKEY, XK_j, focusstack, {.i = +1 } }, + { MODKEY, XK_k, focusstack, {.i = -1 } }, + { MODKEY, XK_i, incnmaster, {.i = +1 } }, + { MODKEY, XK_d, incnmaster, {.i = -1 } }, + { MODKEY, XK_h, setmfact, {.f = -0.05} }, + { MODKEY, XK_l, setmfact, {.f = +0.05} }, + { MODKEY, XK_Return, zoom, {0} }, + { MODKEY, XK_Tab, view, {0} }, + { MODKEY|ShiftMask, XK_c, killclient, {0} }, + { MODKEY, XK_t, setlayout, {.v = &layouts[0]} }, + { MODKEY, XK_f, setlayout, {.v = &layouts[1]} }, + { MODKEY, XK_m, setlayout, {.v = &layouts[2]} }, + { MODKEY, XK_space, setlayout, {0} }, + { MODKEY|ShiftMask, XK_space, togglefloating, {0} }, + { MODKEY, XK_0, view, {.ui = ~0 } }, + { MODKEY|ShiftMask, XK_0, tag, {.ui = ~0 } }, + { MODKEY, XK_comma, focusmon, {.i = -1 } }, + { MODKEY, XK_period, focusmon, {.i = +1 } }, + { MODKEY|ShiftMask, XK_comma, tagmon, {.i = -1 } }, + { MODKEY|ShiftMask, XK_period, tagmon, {.i = +1 } }, + TAGKEYS( XK_1, 0) + TAGKEYS( XK_2, 1) + TAGKEYS( XK_3, 2) + TAGKEYS( XK_4, 3) + TAGKEYS( XK_5, 4) + TAGKEYS( XK_6, 5) + TAGKEYS( XK_7, 6) + TAGKEYS( XK_8, 7) + TAGKEYS( XK_9, 8) + { MODKEY|ShiftMask, XK_q, quit, {0} }, +}; + +/* button definitions */ +/* 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]} }, + { ClkWinTitle, 0, Button2, zoom, {0} }, + { ClkStatusText, 0, Button2, spawn, {.v = termcmd } }, + { ClkClientWin, MODKEY, Button1, movemouse, {0} }, + { ClkClientWin, MODKEY, Button2, togglefloating, {0} }, + { ClkClientWin, MODKEY, Button3, resizemouse, {0} }, + { ClkTagBar, 0, Button1, view, {0} }, + { ClkTagBar, 0, Button3, toggleview, {0} }, + { ClkTagBar, MODKEY, Button1, tag, {0} }, + { ClkTagBar, MODKEY, Button3, toggletag, {0} }, +}; + diff --git a/dwm/config.h b/dwm/config.h new file mode 100644 index 0000000..b28ead7 --- /dev/null +++ b/dwm/config.h @@ -0,0 +1,122 @@ +/* See LICENSE file for copyright and license details. */ + +/* 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 char *fonts[] = { "monospace:size=10" }; +static const char dmenufont[] = "monospace:size=10"; +static const char col_gray1[] = "#222222"; +static const char col_gray2[] = "#444444"; +static const char col_gray3[] = "#bbbbbb"; +static const char col_gray4[] = "#eeeeee"; +static const char col_cyan[] = "#550077"; /* purple */ +static const char *colors[][3] = { + /* fg bg border */ + [SchemeNorm] = { col_gray3, col_gray1, col_gray2 }, + [SchemeSel] = { col_gray4, col_cyan, col_cyan }, +}; + +/* tagging */ +static const char *tags[] = { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F" }; + +static const Rule rules[] = { + /* xprop(1): + * WM_CLASS(STRING) = instance, class + * WM_NAME(STRING) = title + */ + /* class instance title tags mask isfloating monitor */ + { "Gimp", NULL, NULL, 0, 1, -1 }, + { "Firefox", NULL, NULL, 1 << 8, 0, -1 }, +}; + +/* layout(s) */ +static const float mfact = 0.55; /* factor of master area size [0.05..0.95] */ +static const int nmaster = 1; /* number of clients in master area */ +static const int resizehints = 1; /* 1 means respect size hints in tiled resizals */ +static const int lockfullscreen = 1; /* 1 will force focus on the fullscreen window */ + +static const Layout layouts[] = { + /* symbol arrange function */ + { "[]=", tile }, /* first entry is default */ + { "><>", NULL }, /* no layout function means floating behavior */ + { "[M]", monocle }, +}; + +/* key definitions */ +#define MODKEY Mod1Mask +//#define MODKEY2 Mod4Mask /* windows key */ +#define TAGKEYS(KEY,TAG) \ + { MODKEY, KEY, view, {.ui = 1 << TAG} }, \ + { MODKEY|ControlMask, KEY, toggleview, {.ui = 1 << TAG} }, \ + { MODKEY|ShiftMask, KEY, tag, {.ui = 1 << TAG} }, \ + { MODKEY|ControlMask|ShiftMask, KEY, toggletag, {.ui = 1 << TAG} }, + +/* helper for spawning shell commands in the pre dwm-5.0 fashion */ +#define SHCMD(cmd) { .v = (const char*[]){ "/bin/sh", "-c", cmd, NULL } } + +/* commands */ +static char dmenumon[2] = "0"; /* component of dmenucmd, manipulated in spawn() */ +static const char *dmenucmd[] = { "dmenu_run", "-m", dmenumon, "-fn", dmenufont, "-nb", col_gray1, "-nf", col_gray3, "-sb", col_cyan, "-sf", col_gray4, NULL }; +static const char *termcmd[] = { "st", NULL }; +static const char *scrot1[] = { "scrot", "'%+.png' -q 100 -z -e 'mv $f ~/screenshots/'", NULL }; // $(date).png, quality 100 (best), dont beep, move the file into ~/screenshots/ +static const char *scrot2[] = { "scrot", "'%+.png' -q 100 -s -z -e 'mv $f ~/screenshots/'", NULL }; // $(date).png, quality 100 (best), select area for screenschot, dont beep, move the file into ~/screenshots/ + +static Key keys[] = { + /* modifier key function argument */ + { 0, XK_Super_L, spawn, {.v = dmenucmd } }, //super opens dmenu + { 0, XK_Print, spawn, {.v = scrot1 } }, //print screen takes screenshot of whole screen with scrot + { MODKEY, XK_Print, spawn, {.v = scrot2 } }, //alt+print screen takes screenshot of area with scrot + { MODKEY|ShiftMask, XK_Return, spawn, {.v = termcmd } }, + { MODKEY, XK_b, togglebar, {0} }, + { MODKEY, XK_j, focusstack, {.i = +1 } }, + { MODKEY, XK_k, focusstack, {.i = -1 } }, + { MODKEY, XK_i, incnmaster, {.i = +1 } }, + { MODKEY, XK_d, incnmaster, {.i = -1 } }, + { MODKEY, XK_h, setmfact, {.f = -0.05} }, + { MODKEY, XK_l, setmfact, {.f = +0.05} }, + { MODKEY, XK_Return, zoom, {0} }, + { MODKEY, XK_Tab, view, {0} }, + { MODKEY|ShiftMask, XK_c, killclient, {0} }, + { MODKEY, XK_t, setlayout, {.v = &layouts[0]} }, + { MODKEY, XK_f, setlayout, {.v = &layouts[1]} }, + { MODKEY, XK_m, setlayout, {.v = &layouts[2]} }, + { MODKEY, XK_space, setlayout, {0} }, + { MODKEY|ShiftMask, XK_space, togglefloating, {0} }, +/* { MODKEY, XK_0, view, {.ui = ~0 } }, */ +/* { MODKEY|ShiftMask, XK_0, tag, {.ui = ~0 } }, */ + { MODKEY, XK_comma, focusmon, {.i = -1 } }, + { MODKEY, XK_period, focusmon, {.i = +1 } }, + { MODKEY|ShiftMask, XK_comma, tagmon, {.i = -1 } }, + { MODKEY|ShiftMask, XK_period, tagmon, {.i = +1 } }, + TAGKEYS( XK_0, 0) + TAGKEYS( XK_1, 1) + TAGKEYS( XK_2, 2) + TAGKEYS( XK_3, 3) + TAGKEYS( XK_4, 4) + TAGKEYS( XK_5, 5) + TAGKEYS( XK_6, 6) + TAGKEYS( XK_7, 7) + TAGKEYS( XK_8, 8) + TAGKEYS( XK_9, 9) + { MODKEY|ShiftMask, XK_q, quit, {0} }, +}; + +/* button definitions */ +/* 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]} }, + { ClkWinTitle, 0, Button2, zoom, {0} }, + { ClkStatusText, 0, Button2, spawn, {.v = termcmd } }, + { ClkClientWin, MODKEY, Button1, movemouse, {0} }, + { ClkClientWin, MODKEY, Button2, togglefloating, {0} }, + { ClkClientWin, MODKEY, Button3, resizemouse, {0} }, + { ClkTagBar, 0, Button1, view, {0} }, + { ClkTagBar, 0, Button3, toggleview, {0} }, + { ClkTagBar, MODKEY, Button1, tag, {0} }, + { ClkTagBar, MODKEY, Button3, toggletag, {0} }, +}; + diff --git a/dwm/config.mk b/dwm/config.mk new file mode 100644 index 0000000..33623d1 --- /dev/null +++ b/dwm/config.mk @@ -0,0 +1,38 @@ +# dwm version +VERSION = 6.2 + +# Customize below to fit your system + +# paths +PREFIX = /usr/bin +MANPREFIX = ${PREFIX}/share/man + +X11INC = /usr/X11R6/include +X11LIB = /usr/X11R6/lib + +# Xinerama, comment if you don't want it +XINERAMALIBS = -lXinerama +XINERAMAFLAGS = -DXINERAMA + +# freetype +FREETYPELIBS = -lfontconfig -lXft +FREETYPEINC = /usr/include/freetype2 +# OpenBSD (uncomment) +#FREETYPEINC = ${X11INC}/freetype2 + +# includes and libs +INCS = -I${X11INC} -I${FREETYPEINC} +LIBS = -L${X11LIB} -lX11 ${XINERAMALIBS} ${FREETYPELIBS} + +# flags +CPPFLAGS = -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_POSIX_C_SOURCE=200809L -DVERSION=\"${VERSION}\" ${XINERAMAFLAGS} +#CFLAGS = -g -std=c99 -pedantic -Wall -O0 ${INCS} ${CPPFLAGS} +CFLAGS = -std=c99 -pedantic -Wall -Wno-deprecated-declarations -Os ${INCS} ${CPPFLAGS} +LDFLAGS = ${LIBS} + +# Solaris +#CFLAGS = -fast ${INCS} -DVERSION=\"${VERSION}\" +#LDFLAGS = ${LIBS} + +# compiler and linker +CC = cc diff --git a/dwm/drw.c b/dwm/drw.c new file mode 100644 index 0000000..4cdbcbe --- /dev/null +++ b/dwm/drw.c @@ -0,0 +1,436 @@ +/* See LICENSE file for copyright and license details. */ +#include +#include +#include +#include +#include + +#include "drw.h" +#include "util.h" + +#define UTF_INVALID 0xFFFD +#define UTF_SIZ 4 + +static const unsigned char utfbyte[UTF_SIZ + 1] = {0x80, 0, 0xC0, 0xE0, 0xF0}; +static const unsigned char utfmask[UTF_SIZ + 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; +static const long utfmin[UTF_SIZ + 1] = { 0, 0, 0x80, 0x800, 0x10000}; +static const long utfmax[UTF_SIZ + 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; + +static long +utf8decodebyte(const char c, size_t *i) +{ + for (*i = 0; *i < (UTF_SIZ + 1); ++(*i)) + if (((unsigned char)c & utfmask[*i]) == utfbyte[*i]) + return (unsigned char)c & ~utfmask[*i]; + return 0; +} + +static size_t +utf8validate(long *u, size_t i) +{ + if (!BETWEEN(*u, utfmin[i], utfmax[i]) || BETWEEN(*u, 0xD800, 0xDFFF)) + *u = UTF_INVALID; + for (i = 1; *u > utfmax[i]; ++i) + ; + return i; +} + +static size_t +utf8decode(const char *c, long *u, size_t clen) +{ + size_t i, j, len, type; + long udecoded; + + *u = UTF_INVALID; + if (!clen) + return 0; + udecoded = utf8decodebyte(c[0], &len); + if (!BETWEEN(len, 1, UTF_SIZ)) + return 1; + for (i = 1, j = 1; i < clen && j < len; ++i, ++j) { + udecoded = (udecoded << 6) | utf8decodebyte(c[i], &type); + if (type) + return j; + } + if (j < len) + return 0; + *u = udecoded; + utf8validate(u, len); + + return len; +} + +Drw * +drw_create(Display *dpy, int screen, Window root, unsigned int w, unsigned int h) +{ + Drw *drw = ecalloc(1, sizeof(Drw)); + + drw->dpy = dpy; + drw->screen = screen; + drw->root = root; + drw->w = w; + drw->h = h; + drw->drawable = XCreatePixmap(dpy, root, w, h, DefaultDepth(dpy, screen)); + drw->gc = XCreateGC(dpy, root, 0, NULL); + XSetLineAttributes(dpy, drw->gc, 1, LineSolid, CapButt, JoinMiter); + + return drw; +} + +void +drw_resize(Drw *drw, unsigned int w, unsigned int h) +{ + if (!drw) + return; + + drw->w = w; + drw->h = h; + if (drw->drawable) + XFreePixmap(drw->dpy, drw->drawable); + drw->drawable = XCreatePixmap(drw->dpy, drw->root, w, h, DefaultDepth(drw->dpy, drw->screen)); +} + +void +drw_free(Drw *drw) +{ + XFreePixmap(drw->dpy, drw->drawable); + XFreeGC(drw->dpy, drw->gc); + drw_fontset_free(drw->fonts); + free(drw); +} + +/* This function is an implementation detail. Library users should use + * drw_fontset_create instead. + */ +static Fnt * +xfont_create(Drw *drw, const char *fontname, FcPattern *fontpattern) +{ + Fnt *font; + XftFont *xfont = NULL; + FcPattern *pattern = NULL; + + if (fontname) { + /* Using the pattern found at font->xfont->pattern does not yield the + * same substitution results as using the pattern returned by + * FcNameParse; using the latter results in the desired fallback + * behaviour whereas the former just results in missing-character + * rectangles being drawn, at least with some fonts. */ + if (!(xfont = XftFontOpenName(drw->dpy, drw->screen, fontname))) { + fprintf(stderr, "error, cannot load font from name: '%s'\n", fontname); + return NULL; + } + if (!(pattern = FcNameParse((FcChar8 *) fontname))) { + fprintf(stderr, "error, cannot parse font name to pattern: '%s'\n", fontname); + XftFontClose(drw->dpy, xfont); + return NULL; + } + } else if (fontpattern) { + if (!(xfont = XftFontOpenPattern(drw->dpy, fontpattern))) { + fprintf(stderr, "error, cannot load font from pattern.\n"); + return NULL; + } + } else { + die("no font specified."); + } + + /* Do not allow using color fonts. This is a workaround for a BadLength + * error from Xft with color glyphs. Modelled on the Xterm workaround. See + * https://bugzilla.redhat.com/show_bug.cgi?id=1498269 + * https://lists.suckless.org/dev/1701/30932.html + * https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=916349 + * and lots more all over the internet. + */ + FcBool iscol; + if(FcPatternGetBool(xfont->pattern, FC_COLOR, 0, &iscol) == FcResultMatch && iscol) { + XftFontClose(drw->dpy, xfont); + return NULL; + } + + font = ecalloc(1, sizeof(Fnt)); + font->xfont = xfont; + font->pattern = pattern; + font->h = xfont->ascent + xfont->descent; + font->dpy = drw->dpy; + + return font; +} + +static void +xfont_free(Fnt *font) +{ + if (!font) + return; + if (font->pattern) + FcPatternDestroy(font->pattern); + XftFontClose(font->dpy, font->xfont); + free(font); +} + +Fnt* +drw_fontset_create(Drw* drw, const char *fonts[], size_t fontcount) +{ + Fnt *cur, *ret = NULL; + size_t i; + + if (!drw || !fonts) + return NULL; + + for (i = 1; i <= fontcount; i++) { + if ((cur = xfont_create(drw, fonts[fontcount - i], NULL))) { + cur->next = ret; + ret = cur; + } + } + return (drw->fonts = ret); +} + +void +drw_fontset_free(Fnt *font) +{ + if (font) { + drw_fontset_free(font->next); + xfont_free(font); + } +} + +void +drw_clr_create(Drw *drw, Clr *dest, const char *clrname) +{ + if (!drw || !dest || !clrname) + return; + + if (!XftColorAllocName(drw->dpy, DefaultVisual(drw->dpy, drw->screen), + DefaultColormap(drw->dpy, drw->screen), + clrname, dest)) + die("error, cannot allocate color '%s'", clrname); +} + +/* Wrapper to create color schemes. The caller has to call free(3) on the + * returned color scheme when done using it. */ +Clr * +drw_scm_create(Drw *drw, const char *clrnames[], size_t clrcount) +{ + size_t i; + Clr *ret; + + /* need at least two colors for a scheme */ + if (!drw || !clrnames || clrcount < 2 || !(ret = ecalloc(clrcount, sizeof(XftColor)))) + return NULL; + + for (i = 0; i < clrcount; i++) + drw_clr_create(drw, &ret[i], clrnames[i]); + return ret; +} + +void +drw_setfontset(Drw *drw, Fnt *set) +{ + if (drw) + drw->fonts = set; +} + +void +drw_setscheme(Drw *drw, Clr *scm) +{ + if (drw) + drw->scheme = scm; +} + +void +drw_rect(Drw *drw, int x, int y, unsigned int w, unsigned int h, int filled, int invert) +{ + if (!drw || !drw->scheme) + return; + XSetForeground(drw->dpy, drw->gc, invert ? drw->scheme[ColBg].pixel : drw->scheme[ColFg].pixel); + if (filled) + XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h); + else + XDrawRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w - 1, h - 1); +} + +int +drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lpad, const char *text, int invert) +{ + char buf[1024]; + int ty; + unsigned int ew; + XftDraw *d = NULL; + Fnt *usedfont, *curfont, *nextfont; + size_t i, len; + int utf8strlen, utf8charlen, render = x || y || w || h; + long utf8codepoint = 0; + const char *utf8str; + FcCharSet *fccharset; + FcPattern *fcpattern; + FcPattern *match; + XftResult result; + int charexists = 0; + + if (!drw || (render && !drw->scheme) || !text || !drw->fonts) + return 0; + + if (!render) { + w = ~w; + } else { + XSetForeground(drw->dpy, drw->gc, drw->scheme[invert ? ColFg : ColBg].pixel); + XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h); + d = XftDrawCreate(drw->dpy, drw->drawable, + DefaultVisual(drw->dpy, drw->screen), + DefaultColormap(drw->dpy, drw->screen)); + x += lpad; + w -= lpad; + } + + usedfont = drw->fonts; + while (1) { + utf8strlen = 0; + utf8str = text; + nextfont = NULL; + while (*text) { + utf8charlen = utf8decode(text, &utf8codepoint, UTF_SIZ); + for (curfont = drw->fonts; curfont; curfont = curfont->next) { + charexists = charexists || XftCharExists(drw->dpy, curfont->xfont, utf8codepoint); + if (charexists) { + if (curfont == usedfont) { + utf8strlen += utf8charlen; + text += utf8charlen; + } else { + nextfont = curfont; + } + break; + } + } + + if (!charexists || nextfont) + break; + else + charexists = 0; + } + + if (utf8strlen) { + drw_font_getexts(usedfont, utf8str, utf8strlen, &ew, NULL); + /* shorten text if necessary */ + for (len = MIN(utf8strlen, sizeof(buf) - 1); len && ew > w; len--) + drw_font_getexts(usedfont, utf8str, len, &ew, NULL); + + if (len) { + memcpy(buf, utf8str, len); + buf[len] = '\0'; + if (len < utf8strlen) + for (i = len; i && i > len - 3; buf[--i] = '.') + ; /* NOP */ + + if (render) { + ty = y + (h - usedfont->h) / 2 + usedfont->xfont->ascent; + XftDrawStringUtf8(d, &drw->scheme[invert ? ColBg : ColFg], + usedfont->xfont, x, ty, (XftChar8 *)buf, len); + } + x += ew; + w -= ew; + } + } + + if (!*text) { + break; + } else if (nextfont) { + charexists = 0; + usedfont = nextfont; + } else { + /* Regardless of whether or not a fallback font is found, the + * character must be drawn. */ + charexists = 1; + + fccharset = FcCharSetCreate(); + FcCharSetAddChar(fccharset, utf8codepoint); + + if (!drw->fonts->pattern) { + /* Refer to the comment in xfont_create for more information. */ + die("the first font in the cache must be loaded from a font string."); + } + + fcpattern = FcPatternDuplicate(drw->fonts->pattern); + FcPatternAddCharSet(fcpattern, FC_CHARSET, fccharset); + FcPatternAddBool(fcpattern, FC_SCALABLE, FcTrue); + FcPatternAddBool(fcpattern, FC_COLOR, FcFalse); + + FcConfigSubstitute(NULL, fcpattern, FcMatchPattern); + FcDefaultSubstitute(fcpattern); + match = XftFontMatch(drw->dpy, drw->screen, fcpattern, &result); + + FcCharSetDestroy(fccharset); + FcPatternDestroy(fcpattern); + + if (match) { + usedfont = xfont_create(drw, NULL, match); + if (usedfont && XftCharExists(drw->dpy, usedfont->xfont, utf8codepoint)) { + for (curfont = drw->fonts; curfont->next; curfont = curfont->next) + ; /* NOP */ + curfont->next = usedfont; + } else { + xfont_free(usedfont); + usedfont = drw->fonts; + } + } + } + } + if (d) + XftDrawDestroy(d); + + return x + (render ? w : 0); +} + +void +drw_map(Drw *drw, Window win, int x, int y, unsigned int w, unsigned int h) +{ + if (!drw) + return; + + XCopyArea(drw->dpy, drw->drawable, win, drw->gc, x, y, w, h, x, y); + XSync(drw->dpy, False); +} + +unsigned int +drw_fontset_getwidth(Drw *drw, const char *text) +{ + if (!drw || !drw->fonts || !text) + return 0; + return drw_text(drw, 0, 0, 0, 0, 0, text, 0); +} + +void +drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h) +{ + XGlyphInfo ext; + + if (!font || !text) + return; + + XftTextExtentsUtf8(font->dpy, font->xfont, (XftChar8 *)text, len, &ext); + if (w) + *w = ext.xOff; + if (h) + *h = font->h; +} + +Cur * +drw_cur_create(Drw *drw, int shape) +{ + Cur *cur; + + if (!drw || !(cur = ecalloc(1, sizeof(Cur)))) + return NULL; + + cur->cursor = XCreateFontCursor(drw->dpy, shape); + + return cur; +} + +void +drw_cur_free(Drw *drw, Cur *cursor) +{ + if (!cursor) + return; + + XFreeCursor(drw->dpy, cursor->cursor); + free(cursor); +} diff --git a/dwm/drw.h b/dwm/drw.h new file mode 100644 index 0000000..4bcd5ad --- /dev/null +++ b/dwm/drw.h @@ -0,0 +1,57 @@ +/* See LICENSE file for copyright and license details. */ + +typedef struct { + Cursor cursor; +} Cur; + +typedef struct Fnt { + Display *dpy; + unsigned int h; + XftFont *xfont; + FcPattern *pattern; + struct Fnt *next; +} Fnt; + +enum { ColFg, ColBg, ColBorder }; /* Clr scheme index */ +typedef XftColor Clr; + +typedef struct { + unsigned int w, h; + Display *dpy; + int screen; + Window root; + Drawable drawable; + GC gc; + Clr *scheme; + Fnt *fonts; +} Drw; + +/* Drawable abstraction */ +Drw *drw_create(Display *dpy, int screen, Window win, unsigned int w, unsigned int h); +void drw_resize(Drw *drw, unsigned int w, unsigned int h); +void drw_free(Drw *drw); + +/* Fnt abstraction */ +Fnt *drw_fontset_create(Drw* drw, const char *fonts[], size_t fontcount); +void drw_fontset_free(Fnt* set); +unsigned int drw_fontset_getwidth(Drw *drw, const char *text); +void drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h); + +/* Colorscheme abstraction */ +void drw_clr_create(Drw *drw, Clr *dest, const char *clrname); +Clr *drw_scm_create(Drw *drw, const char *clrnames[], size_t clrcount); + +/* Cursor abstraction */ +Cur *drw_cur_create(Drw *drw, int shape); +void drw_cur_free(Drw *drw, Cur *cursor); + +/* Drawing context manipulation */ +void drw_setfontset(Drw *drw, Fnt *set); +void drw_setscheme(Drw *drw, Clr *scm); + +/* Drawing functions */ +void drw_rect(Drw *drw, int x, int y, unsigned int w, unsigned int h, int filled, int invert); +int drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lpad, const char *text, int invert); + +/* Map functions */ +void drw_map(Drw *drw, Window win, int x, int y, unsigned int w, unsigned int h); diff --git a/dwm/drw.o b/dwm/drw.o new file mode 100644 index 0000000000000000000000000000000000000000..2343487d57374e1bada6a6d56fed0a0a3bbf1ad5 GIT binary patch literal 10440 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 Q{-mF?nas0x>+AmeA6vCHX#fBK literal 0 HcmV?d00001 diff --git a/dwm/dwm b/dwm/dwm new file mode 100755 index 0000000000000000000000000000000000000000..7e814d0a7d836fd9990323cff83ea7ee51a49e64 GIT binary patch literal 66736 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 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#ifdef XINERAMA +#include +#endif /* XINERAMA */ +#include + +#include "drw.h" +#include "util.h" + +/* 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 ISVISIBLE(C) ((C->tags & C->mon->tagset[C->mon->seltags])) +#define LENGTH(X) (sizeof X / sizeof X[0]) +#define MOUSEMASK (BUTTONMASK|PointerMotionMask) +#define WIDTH(X) ((X)->w + 2 * (X)->bw) +#define HEIGHT(X) ((X)->h + 2 * (X)->bw) +#define TAGMASK ((1 << LENGTH(tags)) - 1) +#define TEXTW(X) (drw_fontset_getwidth(drw, (X)) + lrpad) + +/* enums */ +enum { CurNormal, CurResize, CurMove, CurLast }; /* cursor */ +enum { SchemeNorm, SchemeSel }; /* color schemes */ +enum { NetSupported, NetWMName, NetWMState, NetWMCheck, + NetWMFullscreen, NetActiveWindow, NetWMWindowType, + NetWMWindowTypeDialog, NetClientList, NetLast }; /* EWMH atoms */ +enum { WMProtocols, WMDelete, WMState, WMTakeFocus, WMLast }; /* default atoms */ +enum { ClkTagBar, ClkLtSymbol, ClkStatusText, ClkWinTitle, + ClkClientWin, ClkRootWin, ClkLast }; /* clicks */ + +typedef union { + int i; + unsigned int ui; + float f; + const void *v; +} Arg; + +typedef struct { + unsigned int click; + unsigned int mask; + unsigned int button; + void (*func)(const Arg *arg); + const Arg arg; +} Button; + +typedef struct Monitor Monitor; +typedef struct Client Client; +struct Client { + char name[256]; + float mina, maxa; + int x, y, w, h; + int oldx, oldy, oldw, oldh; + int basew, baseh, incw, inch, maxw, maxh, minw, minh; + int bw, oldbw; + unsigned int tags; + int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen; + Client *next; + Client *snext; + Monitor *mon; + Window win; +}; + +typedef struct { + unsigned int mod; + KeySym keysym; + void (*func)(const Arg *); + const Arg arg; +} Key; + +typedef struct { + const char *symbol; + void (*arrange)(Monitor *); +} Layout; + +struct Monitor { + char ltsymbol[16]; + float mfact; + int nmaster; + int num; + int by; /* bar geometry */ + int mx, my, mw, mh; /* screen size */ + int wx, wy, ww, wh; /* window area */ + unsigned int seltags; + unsigned int sellt; + unsigned int tagset[2]; + int showbar; + int topbar; + Client *clients; + Client *sel; + Client *stack; + Monitor *next; + Window barwin; + const Layout *lt[2]; +}; + +typedef struct { + const char *class; + const char *instance; + const char *title; + unsigned int tags; + int isfloating; + int monitor; +} Rule; + +/* function declarations */ +static void applyrules(Client *c); +static int applysizehints(Client *c, int *x, int *y, int *w, int *h, int interact); +static void arrange(Monitor *m); +static void arrangemon(Monitor *m); +static void attach(Client *c); +static void attachstack(Client *c); +static void buttonpress(XEvent *e); +static void checkotherwm(void); +static void cleanup(void); +static void cleanupmon(Monitor *mon); +static void clientmessage(XEvent *e); +static void configure(Client *c); +static void configurenotify(XEvent *e); +static void configurerequest(XEvent *e); +static Monitor *createmon(void); +static void destroynotify(XEvent *e); +static void detach(Client *c); +static void detachstack(Client *c); +static Monitor *dirtomon(int dir); +static void drawbar(Monitor *m); +static void drawbars(void); +static void enternotify(XEvent *e); +static void expose(XEvent *e); +static void focus(Client *c); +static void focusin(XEvent *e); +static void focusmon(const Arg *arg); +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 int gettextprop(Window w, Atom atom, char *text, unsigned int size); +static void grabbuttons(Client *c, int focused); +static void grabkeys(void); +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 mappingnotify(XEvent *e); +static void maprequest(XEvent *e); +static void monocle(Monitor *m); +static void motionnotify(XEvent *e); +static void movemouse(const Arg *arg); +static Client *nexttiled(Client *c); +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 resize(Client *c, int x, int y, int w, int h, int interact); +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 scan(void); +static int sendevent(Client *c, Atom proto); +static void sendmon(Client *c, Monitor *m); +static void setclientstate(Client *c, long state); +static void setfocus(Client *c); +static void setfullscreen(Client *c, int fullscreen); +static void setlayout(const Arg *arg); +static void setmfact(const Arg *arg); +static void setup(void); +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 tag(const Arg *arg); +static void tagmon(const Arg *arg); +static void tile(Monitor *); +static void togglebar(const Arg *arg); +static void togglefloating(const Arg *arg); +static void toggletag(const Arg *arg); +static void toggleview(const Arg *arg); +static void unfocus(Client *c, int setfocus); +static void unmanage(Client *c, int destroyed); +static void unmapnotify(XEvent *e); +static void updatebarpos(Monitor *m); +static void updatebars(void); +static void updateclientlist(void); +static int updategeom(void); +static void updatenumlockmask(void); +static void updatesizehints(Client *c); +static void updatestatus(void); +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 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 const char broken[] = "broken"; +static char stext[256]; +static int screen; +static int sw, sh; /* X display screen geometry width, height */ +static int bh, blw = 0; /* bar geometry */ +static int lrpad; /* sum of left and right padding for text */ +static int (*xerrorxlib)(Display *, XErrorEvent *); +static unsigned int numlockmask = 0; +static void (*handler[LASTEvent]) (XEvent *) = { + [ButtonPress] = buttonpress, + [ClientMessage] = clientmessage, + [ConfigureRequest] = configurerequest, + [ConfigureNotify] = configurenotify, + [DestroyNotify] = destroynotify, + [EnterNotify] = enternotify, + [Expose] = expose, + [FocusIn] = focusin, + [KeyPress] = keypress, + [MappingNotify] = mappingnotify, + [MapRequest] = maprequest, + [MotionNotify] = motionnotify, + [PropertyNotify] = propertynotify, + [UnmapNotify] = unmapnotify +}; +static Atom wmatom[WMLast], netatom[NetLast]; +static int running = 1; +static Cur *cursor[CurLast]; +static Clr **scheme; +static Display *dpy; +static Drw *drw; +static Monitor *mons, *selmon; +static Window root, wmcheckwin; + +/* configuration, allows nested code to access above variables */ +#include "config.h" + +/* compile-time check if all tags fit into an unsigned int bit array. */ +struct NumTags { char limitexceeded[LENGTH(tags) > 31 ? -1 : 1]; }; + +/* function implementations */ +void +applyrules(Client *c) +{ + const char *class, *instance; + unsigned int i; + const Rule *r; + Monitor *m; + XClassHint ch = { NULL, NULL }; + + /* rule matching */ + c->isfloating = 0; + c->tags = 0; + XGetClassHint(dpy, c->win, &ch); + class = ch.res_class ? ch.res_class : broken; + instance = ch.res_name ? ch.res_name : broken; + + for (i = 0; i < LENGTH(rules); i++) { + r = &rules[i]; + if ((!r->title || strstr(c->name, r->title)) + && (!r->class || strstr(class, r->class)) + && (!r->instance || strstr(instance, r->instance))) + { + c->isfloating = r->isfloating; + c->tags |= r->tags; + for (m = mons; m && m->num != r->monitor; m = m->next); + if (m) + c->mon = m; + } + } + if (ch.res_class) + XFree(ch.res_class); + if (ch.res_name) + XFree(ch.res_name); + c->tags = c->tags & TAGMASK ? c->tags & TAGMASK : c->mon->tagset[c->mon->seltags]; +} + +int +applysizehints(Client *c, int *x, int *y, int *w, int *h, int interact) +{ + int baseismin; + Monitor *m = c->mon; + + /* set minimum possible */ + *w = MAX(1, *w); + *h = MAX(1, *h); + if (interact) { + if (*x > sw) + *x = sw - WIDTH(c); + if (*y > sh) + *y = sh - HEIGHT(c); + if (*x + *w + 2 * c->bw < 0) + *x = 0; + if (*y + *h + 2 * c->bw < 0) + *y = 0; + } else { + if (*x >= m->wx + m->ww) + *x = m->wx + m->ww - WIDTH(c); + if (*y >= m->wy + m->wh) + *y = m->wy + m->wh - HEIGHT(c); + if (*x + *w + 2 * c->bw <= m->wx) + *x = m->wx; + if (*y + *h + 2 * c->bw <= m->wy) + *y = m->wy; + } + if (*h < bh) + *h = bh; + if (*w < bh) + *w = bh; + if (resizehints || c->isfloating || !c->mon->lt[c->mon->sellt]->arrange) { + /* see last two sentences in ICCCM 4.1.2.3 */ + baseismin = c->basew == c->minw && c->baseh == c->minh; + if (!baseismin) { /* temporarily remove base dimensions */ + *w -= c->basew; + *h -= c->baseh; + } + /* adjust for aspect limits */ + if (c->mina > 0 && c->maxa > 0) { + if (c->maxa < (float)*w / *h) + *w = *h * c->maxa + 0.5; + else if (c->mina < (float)*h / *w) + *h = *w * c->mina + 0.5; + } + if (baseismin) { /* increment calculation requires this */ + *w -= c->basew; + *h -= c->baseh; + } + /* adjust for increment value */ + if (c->incw) + *w -= *w % c->incw; + if (c->inch) + *h -= *h % c->inch; + /* restore base dimensions */ + *w = MAX(*w + c->basew, c->minw); + *h = MAX(*h + c->baseh, c->minh); + if (c->maxw) + *w = MIN(*w, c->maxw); + if (c->maxh) + *h = MIN(*h, c->maxh); + } + return *x != c->x || *y != c->y || *w != c->w || *h != c->h; +} + +void +arrange(Monitor *m) +{ + if (m) + showhide(m->stack); + else for (m = mons; m; m = m->next) + showhide(m->stack); + if (m) { + arrangemon(m); + restack(m); + } else for (m = mons; m; m = m->next) + arrangemon(m); +} + +void +arrangemon(Monitor *m) +{ + strncpy(m->ltsymbol, m->lt[m->sellt]->symbol, sizeof m->ltsymbol); + if (m->lt[m->sellt]->arrange) + m->lt[m->sellt]->arrange(m); +} + +void +attach(Client *c) +{ + c->next = c->mon->clients; + c->mon->clients = c; +} + +void +attachstack(Client *c) +{ + c->snext = c->mon->stack; + c->mon->stack = c; +} + +void +buttonpress(XEvent *e) +{ + unsigned int i, x, click; + Arg arg = {0}; + Client *c; + Monitor *m; + XButtonPressedEvent *ev = &e->xbutton; + + click = ClkRootWin; + /* focus monitor if necessary */ + if ((m = wintomon(ev->window)) && m != selmon) { + unfocus(selmon->sel, 1); + selmon = m; + focus(NULL); + } + if (ev->window == selmon->barwin) { + i = x = 0; + do + x += TEXTW(tags[i]); + while (ev->x >= x && ++i < LENGTH(tags)); + if (i < LENGTH(tags)) { + click = ClkTagBar; + arg.ui = 1 << i; + } else if (ev->x < x + blw) + click = ClkLtSymbol; + else if (ev->x > selmon->ww - (int)TEXTW(stext)) + click = ClkStatusText; + else + click = ClkWinTitle; + } else if ((c = wintoclient(ev->window))) { + focus(c); + restack(selmon); + XAllowEvents(dpy, ReplayPointer, CurrentTime); + click = ClkClientWin; + } + for (i = 0; i < LENGTH(buttons); i++) + if (click == buttons[i].click && buttons[i].func && buttons[i].button == ev->button + && CLEANMASK(buttons[i].mask) == CLEANMASK(ev->state)) + buttons[i].func(click == ClkTagBar && buttons[i].arg.i == 0 ? &arg : &buttons[i].arg); +} + +void +checkotherwm(void) +{ + xerrorxlib = XSetErrorHandler(xerrorstart); + /* this causes an error if some other window manager is running */ + XSelectInput(dpy, DefaultRootWindow(dpy), SubstructureRedirectMask); + XSync(dpy, False); + XSetErrorHandler(xerror); + XSync(dpy, False); +} + +void +cleanup(void) +{ + Arg a = {.ui = ~0}; + Layout foo = { "", NULL }; + Monitor *m; + size_t i; + + view(&a); + selmon->lt[selmon->sellt] = &foo; + for (m = mons; m; m = m->next) + while (m->stack) + unmanage(m->stack, 0); + XUngrabKey(dpy, AnyKey, AnyModifier, root); + while (mons) + cleanupmon(mons); + for (i = 0; i < CurLast; i++) + drw_cur_free(drw, cursor[i]); + for (i = 0; i < LENGTH(colors); i++) + free(scheme[i]); + XDestroyWindow(dpy, wmcheckwin); + drw_free(drw); + XSync(dpy, False); + XSetInputFocus(dpy, PointerRoot, RevertToPointerRoot, CurrentTime); + XDeleteProperty(dpy, root, netatom[NetActiveWindow]); +} + +void +cleanupmon(Monitor *mon) +{ + Monitor *m; + + if (mon == mons) + mons = mons->next; + else { + for (m = mons; m && m->next != mon; m = m->next); + m->next = mon->next; + } + XUnmapWindow(dpy, mon->barwin); + XDestroyWindow(dpy, mon->barwin); + free(mon); +} + +void +clientmessage(XEvent *e) +{ + XClientMessageEvent *cme = &e->xclient; + Client *c = wintoclient(cme->window); + + if (!c) + return; + if (cme->message_type == netatom[NetWMState]) { + if (cme->data.l[1] == netatom[NetWMFullscreen] + || cme->data.l[2] == netatom[NetWMFullscreen]) + setfullscreen(c, (cme->data.l[0] == 1 /* _NET_WM_STATE_ADD */ + || (cme->data.l[0] == 2 /* _NET_WM_STATE_TOGGLE */ && !c->isfullscreen))); + } else if (cme->message_type == netatom[NetActiveWindow]) { + if (c != selmon->sel && !c->isurgent) + seturgent(c, 1); + } +} + +void +configure(Client *c) +{ + XConfigureEvent ce; + + ce.type = ConfigureNotify; + ce.display = dpy; + ce.event = c->win; + ce.window = c->win; + ce.x = c->x; + ce.y = c->y; + ce.width = c->w; + ce.height = c->h; + ce.border_width = c->bw; + ce.above = None; + ce.override_redirect = False; + XSendEvent(dpy, c->win, False, StructureNotifyMask, (XEvent *)&ce); +} + +void +configurenotify(XEvent *e) +{ + Monitor *m; + Client *c; + XConfigureEvent *ev = &e->xconfigure; + int dirty; + + /* TODO: updategeom handling sucks, needs to be simplified */ + if (ev->window == root) { + dirty = (sw != ev->width || sh != ev->height); + sw = ev->width; + sh = ev->height; + if (updategeom() || dirty) { + drw_resize(drw, sw, bh); + updatebars(); + for (m = mons; m; m = m->next) { + 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); + } + focus(NULL); + arrange(NULL); + } + } +} + +void +configurerequest(XEvent *e) +{ + Client *c; + Monitor *m; + XConfigureRequestEvent *ev = &e->xconfigurerequest; + XWindowChanges wc; + + if ((c = wintoclient(ev->window))) { + if (ev->value_mask & CWBorderWidth) + c->bw = ev->border_width; + else if (c->isfloating || !selmon->lt[selmon->sellt]->arrange) { + m = c->mon; + if (ev->value_mask & CWX) { + c->oldx = c->x; + c->x = m->mx + ev->x; + } + if (ev->value_mask & CWY) { + c->oldy = c->y; + c->y = m->my + ev->y; + } + if (ev->value_mask & CWWidth) { + c->oldw = c->w; + c->w = ev->width; + } + if (ev->value_mask & CWHeight) { + c->oldh = c->h; + c->h = ev->height; + } + if ((c->x + c->w) > m->mx + m->mw && c->isfloating) + c->x = m->mx + (m->mw / 2 - WIDTH(c) / 2); /* center in x direction */ + if ((c->y + c->h) > m->my + m->mh && c->isfloating) + c->y = m->my + (m->mh / 2 - HEIGHT(c) / 2); /* center in y direction */ + if ((ev->value_mask & (CWX|CWY)) && !(ev->value_mask & (CWWidth|CWHeight))) + configure(c); + if (ISVISIBLE(c)) + XMoveResizeWindow(dpy, c->win, c->x, c->y, c->w, c->h); + } else + configure(c); + } else { + wc.x = ev->x; + wc.y = ev->y; + wc.width = ev->width; + wc.height = ev->height; + wc.border_width = ev->border_width; + wc.sibling = ev->above; + wc.stack_mode = ev->detail; + XConfigureWindow(dpy, ev->window, ev->value_mask, &wc); + } + XSync(dpy, False); +} + +Monitor * +createmon(void) +{ + Monitor *m; + + m = ecalloc(1, sizeof(Monitor)); + m->tagset[0] = m->tagset[1] = 1; + m->mfact = mfact; + m->nmaster = nmaster; + m->showbar = showbar; + m->topbar = topbar; + m->lt[0] = &layouts[0]; + m->lt[1] = &layouts[1 % LENGTH(layouts)]; + strncpy(m->ltsymbol, layouts[0].symbol, sizeof m->ltsymbol); + return m; +} + +void +destroynotify(XEvent *e) +{ + Client *c; + XDestroyWindowEvent *ev = &e->xdestroywindow; + + if ((c = wintoclient(ev->window))) + unmanage(c, 1); +} + +void +detach(Client *c) +{ + Client **tc; + + for (tc = &c->mon->clients; *tc && *tc != c; tc = &(*tc)->next); + *tc = c->next; +} + +void +detachstack(Client *c) +{ + Client **tc, *t; + + for (tc = &c->mon->stack; *tc && *tc != c; tc = &(*tc)->snext); + *tc = c->snext; + + if (c == c->mon->sel) { + for (t = c->mon->stack; t && !ISVISIBLE(t); t = t->snext); + c->mon->sel = t; + } +} + +Monitor * +dirtomon(int dir) +{ + Monitor *m = NULL; + + if (dir > 0) { + if (!(m = selmon->next)) + m = mons; + } else if (selmon == mons) + for (m = mons; m->next; m = m->next); + else + for (m = mons; m->next != selmon; m = m->next); + return m; +} + +void +drawbar(Monitor *m) +{ + int x, w, tw = 0; + int boxs = drw->fonts->h / 9; + int boxw = drw->fonts->h / 6 + 2; + unsigned int i, occ = 0, urg = 0; + Client *c; + + /* 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); + } + + for (c = m->clients; c; c = c->next) { + occ |= c->tags; + if (c->isurgent) + urg |= c->tags; + } + x = 0; + for (i = 0; i < LENGTH(tags); i++) { + w = TEXTW(tags[i]); + drw_setscheme(drw, scheme[m->tagset[m->seltags] & 1 << i ? SchemeSel : SchemeNorm]); + drw_text(drw, x, 0, w, bh, lrpad / 2, tags[i], urg & 1 << i); + if (occ & 1 << i) + drw_rect(drw, x + boxs, boxs, boxw, boxw, + m == selmon && selmon->sel && selmon->sel->tags & 1 << i, + urg & 1 << i); + x += w; + } + w = blw = TEXTW(m->ltsymbol); + drw_setscheme(drw, scheme[SchemeNorm]); + x = drw_text(drw, x, 0, w, bh, lrpad / 2, m->ltsymbol, 0); + + if ((w = m->ww - tw - x) > bh) { + if (m->sel) { + drw_setscheme(drw, scheme[m == selmon ? SchemeSel : SchemeNorm]); + drw_text(drw, x, 0, w, bh, lrpad / 2, m->sel->name, 0); + if (m->sel->isfloating) + drw_rect(drw, x + boxs, boxs, boxw, boxw, m->sel->isfixed, 0); + } else { + drw_setscheme(drw, scheme[SchemeNorm]); + drw_rect(drw, x, 0, w, bh, 1, 1); + } + } + drw_map(drw, m->barwin, 0, 0, m->ww, bh); +} + +void +drawbars(void) +{ + Monitor *m; + + for (m = mons; m; m = m->next) + drawbar(m); +} + +void +enternotify(XEvent *e) +{ + Client *c; + Monitor *m; + XCrossingEvent *ev = &e->xcrossing; + + if ((ev->mode != NotifyNormal || ev->detail == NotifyInferior) && ev->window != root) + return; + c = wintoclient(ev->window); + m = c ? c->mon : wintomon(ev->window); + if (m != selmon) { + unfocus(selmon->sel, 1); + selmon = m; + } else if (!c || c == selmon->sel) + return; + focus(c); +} + +void +expose(XEvent *e) +{ + Monitor *m; + XExposeEvent *ev = &e->xexpose; + + if (ev->count == 0 && (m = wintomon(ev->window))) + drawbar(m); +} + +void +focus(Client *c) +{ + if (!c || !ISVISIBLE(c)) + for (c = selmon->stack; c && !ISVISIBLE(c); c = c->snext); + if (selmon->sel && selmon->sel != c) + unfocus(selmon->sel, 0); + if (c) { + if (c->mon != selmon) + selmon = c->mon; + if (c->isurgent) + seturgent(c, 0); + detachstack(c); + attachstack(c); + grabbuttons(c, 1); + XSetWindowBorder(dpy, c->win, scheme[SchemeSel][ColBorder].pixel); + setfocus(c); + } else { + XSetInputFocus(dpy, root, RevertToPointerRoot, CurrentTime); + XDeleteProperty(dpy, root, netatom[NetActiveWindow]); + } + selmon->sel = c; + drawbars(); +} + +/* there are some broken focus acquiring clients needing extra handling */ +void +focusin(XEvent *e) +{ + XFocusChangeEvent *ev = &e->xfocus; + + if (selmon->sel && ev->window != selmon->sel->win) + setfocus(selmon->sel); +} + +void +focusmon(const Arg *arg) +{ + Monitor *m; + + if (!mons->next) + return; + if ((m = dirtomon(arg->i)) == selmon) + return; + unfocus(selmon->sel, 0); + selmon = m; + focus(NULL); +} + +void +focusstack(const Arg *arg) +{ + Client *c = NULL, *i; + + if (!selmon->sel || (selmon->sel->isfullscreen && lockfullscreen)) + return; + if (arg->i > 0) { + for (c = selmon->sel->next; c && !ISVISIBLE(c); c = c->next); + if (!c) + for (c = selmon->clients; c && !ISVISIBLE(c); c = c->next); + } else { + for (i = selmon->clients; i != selmon->sel; i = i->next) + if (ISVISIBLE(i)) + c = i; + if (!c) + for (; i; i = i->next) + if (ISVISIBLE(i)) + c = i; + } + if (c) { + focus(c); + restack(selmon); + } +} + +Atom +getatomprop(Client *c, Atom prop) +{ + int di; + unsigned long dl; + unsigned char *p = NULL; + Atom da, atom = None; + + if (XGetWindowProperty(dpy, c->win, prop, 0L, sizeof atom, False, XA_ATOM, + &da, &di, &dl, &dl, &p) == Success && p) { + atom = *(Atom *)p; + XFree(p); + } + return atom; +} + +int +getrootptr(int *x, int *y) +{ + int di; + unsigned int dui; + Window dummy; + + return XQueryPointer(dpy, root, &dummy, &dummy, x, y, &di, &di, &dui); +} + +long +getstate(Window w) +{ + int format; + long result = -1; + unsigned char *p = NULL; + unsigned long n, extra; + Atom real; + + if (XGetWindowProperty(dpy, w, wmatom[WMState], 0L, 2L, False, wmatom[WMState], + &real, &format, &n, &extra, (unsigned char **)&p) != Success) + return -1; + if (n != 0) + result = *p; + XFree(p); + return result; +} + +int +gettextprop(Window w, Atom atom, char *text, unsigned int size) +{ + char **list = NULL; + int n; + XTextProperty name; + + if (!text || size == 0) + return 0; + text[0] = '\0'; + if (!XGetTextProperty(dpy, w, &name, atom) || !name.nitems) + return 0; + if (name.encoding == XA_STRING) + strncpy(text, (char *)name.value, size - 1); + else { + if (XmbTextPropertyToTextList(dpy, &name, &list, &n) >= Success && n > 0 && *list) { + strncpy(text, *list, size - 1); + XFreeStringList(list); + } + } + text[size - 1] = '\0'; + XFree(name.value); + return 1; +} + +void +grabbuttons(Client *c, int focused) +{ + updatenumlockmask(); + { + unsigned int i, j; + unsigned int modifiers[] = { 0, LockMask, numlockmask, numlockmask|LockMask }; + XUngrabButton(dpy, AnyButton, AnyModifier, c->win); + if (!focused) + XGrabButton(dpy, AnyButton, AnyModifier, c->win, False, + BUTTONMASK, GrabModeSync, GrabModeSync, None, None); + for (i = 0; i < LENGTH(buttons); i++) + if (buttons[i].click == ClkClientWin) + for (j = 0; j < LENGTH(modifiers); j++) + XGrabButton(dpy, buttons[i].button, + buttons[i].mask | modifiers[j], + c->win, False, BUTTONMASK, + GrabModeAsync, GrabModeSync, None, None); + } +} + +void +grabkeys(void) +{ + updatenumlockmask(); + { + unsigned int i, j; + unsigned int modifiers[] = { 0, LockMask, numlockmask, numlockmask|LockMask }; + KeyCode code; + + XUngrabKey(dpy, AnyKey, AnyModifier, root); + for (i = 0; i < LENGTH(keys); i++) + if ((code = XKeysymToKeycode(dpy, keys[i].keysym))) + for (j = 0; j < LENGTH(modifiers); j++) + XGrabKey(dpy, code, keys[i].mod | modifiers[j], root, + True, GrabModeAsync, GrabModeAsync); + } +} + +void +incnmaster(const Arg *arg) +{ + selmon->nmaster = MAX(selmon->nmaster + arg->i, 0); + arrange(selmon); +} + +#ifdef XINERAMA +static int +isuniquegeom(XineramaScreenInfo *unique, size_t n, XineramaScreenInfo *info) +{ + while (n--) + if (unique[n].x_org == info->x_org && unique[n].y_org == info->y_org + && unique[n].width == info->width && unique[n].height == info->height) + return 0; + return 1; +} +#endif /* XINERAMA */ + +void +keypress(XEvent *e) +{ + unsigned int i; + KeySym keysym; + XKeyEvent *ev; + + ev = &e->xkey; + keysym = XKeycodeToKeysym(dpy, (KeyCode)ev->keycode, 0); + for (i = 0; i < LENGTH(keys); i++) + if (keysym == keys[i].keysym + && CLEANMASK(keys[i].mod) == CLEANMASK(ev->state) + && keys[i].func) + keys[i].func(&(keys[i].arg)); +} + +void +killclient(const Arg *arg) +{ + if (!selmon->sel) + return; + if (!sendevent(selmon->sel, wmatom[WMDelete])) { + XGrabServer(dpy); + XSetErrorHandler(xerrordummy); + XSetCloseDownMode(dpy, DestroyAll); + XKillClient(dpy, selmon->sel->win); + XSync(dpy, False); + XSetErrorHandler(xerror); + XUngrabServer(dpy); + } +} + +void +manage(Window w, XWindowAttributes *wa) +{ + Client *c, *t = NULL; + Window trans = None; + XWindowChanges wc; + + c = ecalloc(1, sizeof(Client)); + c->win = w; + /* geometry */ + c->x = c->oldx = wa->x; + c->y = c->oldy = wa->y; + c->w = c->oldw = wa->width; + c->h = c->oldh = wa->height; + c->oldbw = wa->border_width; + + updatetitle(c); + if (XGetTransientForHint(dpy, w, &trans) && (t = wintoclient(trans))) { + c->mon = t->mon; + c->tags = t->tags; + } else { + c->mon = selmon; + applyrules(c); + } + + if (c->x + WIDTH(c) > c->mon->mx + c->mon->mw) + c->x = c->mon->mx + c->mon->mw - WIDTH(c); + if (c->y + HEIGHT(c) > c->mon->my + c->mon->mh) + c->y = c->mon->my + c->mon->mh - HEIGHT(c); + c->x = MAX(c->x, c->mon->mx); + /* only fix client y-offset, if the client center might cover the bar */ + c->y = MAX(c->y, ((c->mon->by == c->mon->my) && (c->x + (c->w / 2) >= c->mon->wx) + && (c->x + (c->w / 2) < c->mon->wx + c->mon->ww)) ? bh : c->mon->my); + c->bw = borderpx; + + wc.border_width = c->bw; + XConfigureWindow(dpy, w, CWBorderWidth, &wc); + XSetWindowBorder(dpy, w, scheme[SchemeNorm][ColBorder].pixel); + configure(c); /* propagates border_width, if size doesn't change */ + updatewindowtype(c); + updatesizehints(c); + updatewmhints(c); + XSelectInput(dpy, w, EnterWindowMask|FocusChangeMask|PropertyChangeMask|StructureNotifyMask); + grabbuttons(c, 0); + if (!c->isfloating) + c->isfloating = c->oldstate = trans != None || c->isfixed; + if (c->isfloating) + XRaiseWindow(dpy, c->win); + attach(c); + attachstack(c); + XChangeProperty(dpy, root, netatom[NetClientList], XA_WINDOW, 32, PropModeAppend, + (unsigned char *) &(c->win), 1); + XMoveResizeWindow(dpy, c->win, c->x + 2 * sw, c->y, c->w, c->h); /* some windows require this */ + setclientstate(c, NormalState); + if (c->mon == selmon) + unfocus(selmon->sel, 0); + c->mon->sel = c; + arrange(c->mon); + XMapWindow(dpy, c->win); + focus(NULL); +} + +void +mappingnotify(XEvent *e) +{ + XMappingEvent *ev = &e->xmapping; + + XRefreshKeyboardMapping(ev); + if (ev->request == MappingKeyboard) + grabkeys(); +} + +void +maprequest(XEvent *e) +{ + static XWindowAttributes wa; + XMapRequestEvent *ev = &e->xmaprequest; + + if (!XGetWindowAttributes(dpy, ev->window, &wa)) + return; + if (wa.override_redirect) + return; + if (!wintoclient(ev->window)) + manage(ev->window, &wa); +} + +void +monocle(Monitor *m) +{ + unsigned int n = 0; + Client *c; + + for (c = m->clients; c; c = c->next) + if (ISVISIBLE(c)) + n++; + if (n > 0) /* override layout symbol */ + snprintf(m->ltsymbol, sizeof m->ltsymbol, "[%d]", n); + for (c = nexttiled(m->clients); c; c = nexttiled(c->next)) + resize(c, m->wx, m->wy, m->ww - 2 * c->bw, m->wh - 2 * c->bw, 0); +} + +void +motionnotify(XEvent *e) +{ + static Monitor *mon = NULL; + Monitor *m; + XMotionEvent *ev = &e->xmotion; + + if (ev->window != root) + return; + if ((m = recttomon(ev->x_root, ev->y_root, 1, 1)) != mon && mon) { + unfocus(selmon->sel, 1); + selmon = m; + focus(NULL); + } + mon = m; +} + +void +movemouse(const Arg *arg) +{ + int x, y, ocx, ocy, nx, ny; + Client *c; + Monitor *m; + XEvent ev; + Time lasttime = 0; + + if (!(c = selmon->sel)) + return; + if (c->isfullscreen) /* no support moving fullscreen windows by mouse */ + return; + restack(selmon); + ocx = c->x; + ocy = c->y; + if (XGrabPointer(dpy, root, False, MOUSEMASK, GrabModeAsync, GrabModeAsync, + None, cursor[CurMove]->cursor, CurrentTime) != GrabSuccess) + return; + if (!getrootptr(&x, &y)) + return; + do { + XMaskEvent(dpy, MOUSEMASK|ExposureMask|SubstructureRedirectMask, &ev); + switch(ev.type) { + case ConfigureRequest: + case Expose: + case MapRequest: + handler[ev.type](&ev); + break; + case MotionNotify: + if ((ev.xmotion.time - lasttime) <= (1000 / 60)) + continue; + lasttime = ev.xmotion.time; + + nx = ocx + (ev.xmotion.x - x); + ny = ocy + (ev.xmotion.y - y); + if (abs(selmon->wx - nx) < snap) + nx = selmon->wx; + else if (abs((selmon->wx + selmon->ww) - (nx + WIDTH(c))) < snap) + nx = selmon->wx + selmon->ww - WIDTH(c); + if (abs(selmon->wy - ny) < snap) + ny = selmon->wy; + else if (abs((selmon->wy + selmon->wh) - (ny + HEIGHT(c))) < snap) + ny = selmon->wy + selmon->wh - HEIGHT(c); + if (!c->isfloating && selmon->lt[selmon->sellt]->arrange + && (abs(nx - c->x) > snap || abs(ny - c->y) > snap)) + togglefloating(NULL); + if (!selmon->lt[selmon->sellt]->arrange || c->isfloating) + resize(c, nx, ny, c->w, c->h, 1); + break; + } + } while (ev.type != ButtonRelease); + XUngrabPointer(dpy, CurrentTime); + if ((m = recttomon(c->x, c->y, c->w, c->h)) != selmon) { + sendmon(c, m); + selmon = m; + focus(NULL); + } +} + +Client * +nexttiled(Client *c) +{ + for (; c && (c->isfloating || !ISVISIBLE(c)); c = c->next); + return c; +} + +void +pop(Client *c) +{ + detach(c); + attach(c); + focus(c); + arrange(c->mon); +} + +void +propertynotify(XEvent *e) +{ + Client *c; + Window trans; + XPropertyEvent *ev = &e->xproperty; + + if ((ev->window == root) && (ev->atom == XA_WM_NAME)) + updatestatus(); + else if (ev->state == PropertyDelete) + return; /* ignore */ + else if ((c = wintoclient(ev->window))) { + switch(ev->atom) { + default: break; + case XA_WM_TRANSIENT_FOR: + if (!c->isfloating && (XGetTransientForHint(dpy, c->win, &trans)) && + (c->isfloating = (wintoclient(trans)) != NULL)) + arrange(c->mon); + break; + case XA_WM_NORMAL_HINTS: + updatesizehints(c); + break; + case XA_WM_HINTS: + updatewmhints(c); + drawbars(); + break; + } + if (ev->atom == XA_WM_NAME || ev->atom == netatom[NetWMName]) { + updatetitle(c); + if (c == c->mon->sel) + drawbar(c->mon); + } + if (ev->atom == netatom[NetWMWindowType]) + updatewindowtype(c); + } +} + +void +quit(const Arg *arg) +{ + running = 0; +} + +Monitor * +recttomon(int x, int y, int w, int h) +{ + Monitor *m, *r = selmon; + int a, area = 0; + + for (m = mons; m; m = m->next) + if ((a = INTERSECT(x, y, w, h, m)) > area) { + area = a; + r = m; + } + return r; +} + +void +resize(Client *c, int x, int y, int w, int h, int interact) +{ + if (applysizehints(c, &x, &y, &w, &h, interact)) + resizeclient(c, x, y, w, h); +} + +void +resizeclient(Client *c, int x, int y, int w, int h) +{ + XWindowChanges wc; + + c->oldx = c->x; c->x = wc.x = x; + c->oldy = c->y; c->y = wc.y = y; + c->oldw = c->w; c->w = wc.width = w; + c->oldh = c->h; c->h = wc.height = h; + wc.border_width = c->bw; + XConfigureWindow(dpy, c->win, CWX|CWY|CWWidth|CWHeight|CWBorderWidth, &wc); + configure(c); + XSync(dpy, False); +} + +void +resizemouse(const Arg *arg) +{ + int ocx, ocy, nw, nh; + Client *c; + Monitor *m; + XEvent ev; + Time lasttime = 0; + + if (!(c = selmon->sel)) + return; + if (c->isfullscreen) /* no support resizing fullscreen windows by mouse */ + return; + restack(selmon); + ocx = c->x; + ocy = c->y; + if (XGrabPointer(dpy, root, False, MOUSEMASK, GrabModeAsync, GrabModeAsync, + None, cursor[CurResize]->cursor, CurrentTime) != GrabSuccess) + return; + XWarpPointer(dpy, None, c->win, 0, 0, 0, 0, c->w + c->bw - 1, c->h + c->bw - 1); + do { + XMaskEvent(dpy, MOUSEMASK|ExposureMask|SubstructureRedirectMask, &ev); + switch(ev.type) { + case ConfigureRequest: + case Expose: + case MapRequest: + handler[ev.type](&ev); + break; + case MotionNotify: + if ((ev.xmotion.time - lasttime) <= (1000 / 60)) + continue; + lasttime = ev.xmotion.time; + + nw = MAX(ev.xmotion.x - ocx - 2 * c->bw + 1, 1); + nh = MAX(ev.xmotion.y - ocy - 2 * c->bw + 1, 1); + if (c->mon->wx + nw >= selmon->wx && c->mon->wx + nw <= selmon->wx + selmon->ww + && c->mon->wy + nh >= selmon->wy && c->mon->wy + nh <= selmon->wy + selmon->wh) + { + if (!c->isfloating && selmon->lt[selmon->sellt]->arrange + && (abs(nw - c->w) > snap || abs(nh - c->h) > snap)) + togglefloating(NULL); + } + if (!selmon->lt[selmon->sellt]->arrange || c->isfloating) + resize(c, c->x, c->y, nw, nh, 1); + break; + } + } while (ev.type != ButtonRelease); + XWarpPointer(dpy, None, c->win, 0, 0, 0, 0, c->w + c->bw - 1, c->h + c->bw - 1); + XUngrabPointer(dpy, CurrentTime); + while (XCheckMaskEvent(dpy, EnterWindowMask, &ev)); + if ((m = recttomon(c->x, c->y, c->w, c->h)) != selmon) { + sendmon(c, m); + selmon = m; + focus(NULL); + } +} + +void +restack(Monitor *m) +{ + Client *c; + XEvent ev; + XWindowChanges wc; + + drawbar(m); + if (!m->sel) + return; + if (m->sel->isfloating || !m->lt[m->sellt]->arrange) + XRaiseWindow(dpy, m->sel->win); + if (m->lt[m->sellt]->arrange) { + wc.stack_mode = Below; + wc.sibling = m->barwin; + for (c = m->stack; c; c = c->snext) + if (!c->isfloating && ISVISIBLE(c)) { + XConfigureWindow(dpy, c->win, CWSibling|CWStackMode, &wc); + wc.sibling = c->win; + } + } + XSync(dpy, False); + while (XCheckMaskEvent(dpy, EnterWindowMask, &ev)); +} + +void +run(void) +{ + XEvent ev; + /* main event loop */ + XSync(dpy, False); + while (running && !XNextEvent(dpy, &ev)) + if (handler[ev.type]) + handler[ev.type](&ev); /* call handler */ +} + +void +scan(void) +{ + unsigned int i, num; + Window d1, d2, *wins = NULL; + XWindowAttributes wa; + + if (XQueryTree(dpy, root, &d1, &d2, &wins, &num)) { + for (i = 0; i < num; i++) { + if (!XGetWindowAttributes(dpy, wins[i], &wa) + || wa.override_redirect || XGetTransientForHint(dpy, wins[i], &d1)) + continue; + if (wa.map_state == IsViewable || getstate(wins[i]) == IconicState) + manage(wins[i], &wa); + } + for (i = 0; i < num; i++) { /* now the transients */ + if (!XGetWindowAttributes(dpy, wins[i], &wa)) + continue; + if (XGetTransientForHint(dpy, wins[i], &d1) + && (wa.map_state == IsViewable || getstate(wins[i]) == IconicState)) + manage(wins[i], &wa); + } + if (wins) + XFree(wins); + } +} + +void +sendmon(Client *c, Monitor *m) +{ + if (c->mon == m) + return; + unfocus(c, 1); + detach(c); + detachstack(c); + c->mon = m; + c->tags = m->tagset[m->seltags]; /* assign tags of target monitor */ + attach(c); + attachstack(c); + focus(NULL); + arrange(NULL); +} + +void +setclientstate(Client *c, long state) +{ + long data[] = { state, None }; + + XChangeProperty(dpy, c->win, wmatom[WMState], wmatom[WMState], 32, + PropModeReplace, (unsigned char *)data, 2); +} + +int +sendevent(Client *c, Atom proto) +{ + int n; + Atom *protocols; + int exists = 0; + XEvent ev; + + if (XGetWMProtocols(dpy, c->win, &protocols, &n)) { + while (!exists && n--) + exists = protocols[n] == proto; + XFree(protocols); + } + if (exists) { + ev.type = ClientMessage; + ev.xclient.window = c->win; + ev.xclient.message_type = wmatom[WMProtocols]; + ev.xclient.format = 32; + ev.xclient.data.l[0] = proto; + ev.xclient.data.l[1] = CurrentTime; + XSendEvent(dpy, c->win, False, NoEventMask, &ev); + } + return exists; +} + +void +setfocus(Client *c) +{ + if (!c->neverfocus) { + XSetInputFocus(dpy, c->win, RevertToPointerRoot, CurrentTime); + XChangeProperty(dpy, root, netatom[NetActiveWindow], + XA_WINDOW, 32, PropModeReplace, + (unsigned char *) &(c->win), 1); + } + sendevent(c, wmatom[WMTakeFocus]); +} + +void +setfullscreen(Client *c, int fullscreen) +{ + if (fullscreen && !c->isfullscreen) { + XChangeProperty(dpy, c->win, netatom[NetWMState], XA_ATOM, 32, + PropModeReplace, (unsigned char*)&netatom[NetWMFullscreen], 1); + c->isfullscreen = 1; + c->oldstate = c->isfloating; + c->oldbw = c->bw; + c->bw = 0; + c->isfloating = 1; + resizeclient(c, c->mon->mx, c->mon->my, c->mon->mw, c->mon->mh); + XRaiseWindow(dpy, c->win); + } else if (!fullscreen && c->isfullscreen){ + XChangeProperty(dpy, c->win, netatom[NetWMState], XA_ATOM, 32, + PropModeReplace, (unsigned char*)0, 0); + c->isfullscreen = 0; + c->isfloating = c->oldstate; + c->bw = c->oldbw; + c->x = c->oldx; + c->y = c->oldy; + c->w = c->oldw; + c->h = c->oldh; + resizeclient(c, c->x, c->y, c->w, c->h); + arrange(c->mon); + } +} + +void +setlayout(const Arg *arg) +{ + if (!arg || !arg->v || arg->v != selmon->lt[selmon->sellt]) + selmon->sellt ^= 1; + if (arg && arg->v) + selmon->lt[selmon->sellt] = (Layout *)arg->v; + strncpy(selmon->ltsymbol, selmon->lt[selmon->sellt]->symbol, sizeof selmon->ltsymbol); + if (selmon->sel) + arrange(selmon); + else + drawbar(selmon); +} + +/* arg > 1.0 will set mfact absolutely */ +void +setmfact(const Arg *arg) +{ + float f; + + if (!arg || !selmon->lt[selmon->sellt]->arrange) + return; + f = arg->f < 1.0 ? arg->f + selmon->mfact : arg->f - 1.0; + if (f < 0.05 || f > 0.95) + return; + selmon->mfact = f; + arrange(selmon); +} + +void +setup(void) +{ + int i; + XSetWindowAttributes wa; + Atom utf8string; + + /* clean up any zombies immediately */ + sigchld(0); + + /* init screen */ + screen = DefaultScreen(dpy); + sw = DisplayWidth(dpy, screen); + sh = DisplayHeight(dpy, screen); + root = RootWindow(dpy, screen); + drw = drw_create(dpy, screen, root, sw, sh); + if (!drw_fontset_create(drw, fonts, LENGTH(fonts))) + die("no fonts could be loaded."); + lrpad = drw->fonts->h; + bh = drw->fonts->h + 2; + updategeom(); + /* init atoms */ + utf8string = XInternAtom(dpy, "UTF8_STRING", False); + wmatom[WMProtocols] = XInternAtom(dpy, "WM_PROTOCOLS", False); + wmatom[WMDelete] = XInternAtom(dpy, "WM_DELETE_WINDOW", False); + 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[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 */ + 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); + /* init appearance */ + scheme = ecalloc(LENGTH(colors), sizeof(Clr *)); + for (i = 0; i < LENGTH(colors); i++) + scheme[i] = drw_scm_create(drw, colors[i], 3); + /* init bars */ + updatebars(); + updatestatus(); + /* supporting window for NetWMCheck */ + wmcheckwin = XCreateSimpleWindow(dpy, root, 0, 0, 1, 1, 0, 0, 0); + XChangeProperty(dpy, wmcheckwin, netatom[NetWMCheck], XA_WINDOW, 32, + PropModeReplace, (unsigned char *) &wmcheckwin, 1); + XChangeProperty(dpy, wmcheckwin, netatom[NetWMName], utf8string, 8, + PropModeReplace, (unsigned char *) "dwm", 3); + XChangeProperty(dpy, root, netatom[NetWMCheck], XA_WINDOW, 32, + PropModeReplace, (unsigned char *) &wmcheckwin, 1); + /* EWMH support per view */ + XChangeProperty(dpy, root, netatom[NetSupported], XA_ATOM, 32, + PropModeReplace, (unsigned char *) netatom, NetLast); + XDeleteProperty(dpy, root, netatom[NetClientList]); + /* select events */ + wa.cursor = cursor[CurNormal]->cursor; + wa.event_mask = SubstructureRedirectMask|SubstructureNotifyMask + |ButtonPressMask|PointerMotionMask|EnterWindowMask + |LeaveWindowMask|StructureNotifyMask|PropertyChangeMask; + XChangeWindowAttributes(dpy, root, CWEventMask|CWCursor, &wa); + XSelectInput(dpy, root, wa.event_mask); + grabkeys(); + focus(NULL); +} + + +void +seturgent(Client *c, int urg) +{ + XWMHints *wmh; + + c->isurgent = urg; + if (!(wmh = XGetWMHints(dpy, c->win))) + return; + wmh->flags = urg ? (wmh->flags | XUrgencyHint) : (wmh->flags & ~XUrgencyHint); + XSetWMHints(dpy, c->win, wmh); + XFree(wmh); +} + +void +showhide(Client *c) +{ + if (!c) + return; + if (ISVISIBLE(c)) { + /* show clients top down */ + XMoveWindow(dpy, c->win, c->x, c->y); + if ((!c->mon->lt[c->mon->sellt]->arrange || c->isfloating) && !c->isfullscreen) + resize(c, c->x, c->y, c->w, c->h, 0); + showhide(c->snext); + } else { + /* hide clients bottom up */ + showhide(c->snext); + XMoveWindow(dpy, c->win, WIDTH(c) * -2, c->y); + } +} + +void +sigchld(int unused) +{ + if (signal(SIGCHLD, sigchld) == SIG_ERR) + die("can't install SIGCHLD handler:"); + while (0 < waitpid(-1, NULL, WNOHANG)); +} + +void +spawn(const Arg *arg) +{ + if (arg->v == dmenucmd) + dmenumon[0] = '0' + selmon->num; + if (fork() == 0) { + if (dpy) + close(ConnectionNumber(dpy)); + setsid(); + execvp(((char **)arg->v)[0], (char **)arg->v); + fprintf(stderr, "dwm: execvp %s", ((char **)arg->v)[0]); + perror(" failed"); + exit(EXIT_SUCCESS); + } +} + +void +tag(const Arg *arg) +{ + if (selmon->sel && arg->ui & TAGMASK) { + selmon->sel->tags = arg->ui & TAGMASK; + focus(NULL); + arrange(selmon); + } +} + +void +tagmon(const Arg *arg) +{ + if (!selmon->sel || !mons->next) + return; + sendmon(selmon->sel, dirtomon(arg->i)); +} + +void +tile(Monitor *m) +{ + unsigned int i, n, h, mw, my, ty; + Client *c; + + for (n = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), n++); + if (n == 0) + return; + + if (n > m->nmaster) + mw = m->nmaster ? m->ww * m->mfact : 0; + else + mw = m->ww; + for (i = my = ty = 0, c = nexttiled(m->clients); c; c = nexttiled(c->next), i++) + if (i < m->nmaster) { + h = (m->wh - my) / (MIN(n, m->nmaster) - i); + resize(c, m->wx, m->wy + my, mw - (2*c->bw), h - (2*c->bw), 0); + if (my + HEIGHT(c) < m->wh) + my += HEIGHT(c); + } else { + h = (m->wh - ty) / (n - i); + resize(c, m->wx + mw, m->wy + ty, m->ww - mw - (2*c->bw), h - (2*c->bw), 0); + if (ty + HEIGHT(c) < m->wh) + ty += HEIGHT(c); + } +} + +void +togglebar(const Arg *arg) +{ + selmon->showbar = !selmon->showbar; + updatebarpos(selmon); + XMoveResizeWindow(dpy, selmon->barwin, selmon->wx, selmon->by, selmon->ww, bh); + arrange(selmon); +} + +void +togglefloating(const Arg *arg) +{ + if (!selmon->sel) + return; + if (selmon->sel->isfullscreen) /* no support for fullscreen windows */ + return; + selmon->sel->isfloating = !selmon->sel->isfloating || selmon->sel->isfixed; + if (selmon->sel->isfloating) + resize(selmon->sel, selmon->sel->x, selmon->sel->y, + selmon->sel->w, selmon->sel->h, 0); + arrange(selmon); +} + +void +toggletag(const Arg *arg) +{ + unsigned int newtags; + + if (!selmon->sel) + return; + newtags = selmon->sel->tags ^ (arg->ui & TAGMASK); + if (newtags) { + selmon->sel->tags = newtags; + focus(NULL); + arrange(selmon); + } +} + +void +toggleview(const Arg *arg) +{ + unsigned int newtagset = selmon->tagset[selmon->seltags] ^ (arg->ui & TAGMASK); + + if (newtagset) { + selmon->tagset[selmon->seltags] = newtagset; + focus(NULL); + arrange(selmon); + } +} + +void +unfocus(Client *c, int setfocus) +{ + if (!c) + return; + grabbuttons(c, 0); + XSetWindowBorder(dpy, c->win, scheme[SchemeNorm][ColBorder].pixel); + if (setfocus) { + XSetInputFocus(dpy, root, RevertToPointerRoot, CurrentTime); + XDeleteProperty(dpy, root, netatom[NetActiveWindow]); + } +} + +void +unmanage(Client *c, int destroyed) +{ + Monitor *m = c->mon; + XWindowChanges wc; + + detach(c); + detachstack(c); + if (!destroyed) { + wc.border_width = c->oldbw; + XGrabServer(dpy); /* avoid race conditions */ + XSetErrorHandler(xerrordummy); + XConfigureWindow(dpy, c->win, CWBorderWidth, &wc); /* restore border */ + XUngrabButton(dpy, AnyButton, AnyModifier, c->win); + setclientstate(c, WithdrawnState); + XSync(dpy, False); + XSetErrorHandler(xerror); + XUngrabServer(dpy); + } + free(c); + focus(NULL); + updateclientlist(); + arrange(m); +} + +void +unmapnotify(XEvent *e) +{ + Client *c; + XUnmapEvent *ev = &e->xunmap; + + if ((c = wintoclient(ev->window))) { + if (ev->send_event) + setclientstate(c, WithdrawnState); + else + unmanage(c, 0); + } +} + +void +updatebars(void) +{ + Monitor *m; + XSetWindowAttributes wa = { + .override_redirect = True, + .background_pixmap = ParentRelative, + .event_mask = ButtonPressMask|ExposureMask + }; + XClassHint ch = {"dwm", "dwm"}; + 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), + CopyFromParent, DefaultVisual(dpy, screen), + CWOverrideRedirect|CWBackPixmap|CWEventMask, &wa); + XDefineCursor(dpy, m->barwin, cursor[CurNormal]->cursor); + XMapRaised(dpy, m->barwin); + XSetClassHint(dpy, m->barwin, &ch); + } +} + +void +updatebarpos(Monitor *m) +{ + m->wy = m->my; + m->wh = m->mh; + if (m->showbar) { + m->wh -= bh; + m->by = m->topbar ? m->wy : m->wy + m->wh; + m->wy = m->topbar ? m->wy + bh : m->wy; + } else + m->by = -bh; +} + +void +updateclientlist() +{ + Client *c; + Monitor *m; + + XDeleteProperty(dpy, root, netatom[NetClientList]); + for (m = mons; m; m = m->next) + for (c = m->clients; c; c = c->next) + XChangeProperty(dpy, root, netatom[NetClientList], + XA_WINDOW, 32, PropModeAppend, + (unsigned char *) &(c->win), 1); +} + +int +updategeom(void) +{ + int dirty = 0; + +#ifdef XINERAMA + if (XineramaIsActive(dpy)) { + int i, j, n, nn; + Client *c; + Monitor *m; + XineramaScreenInfo *info = XineramaQueryScreens(dpy, &nn); + XineramaScreenInfo *unique = NULL; + + for (n = 0, m = mons; m; m = m->next, n++); + /* only consider unique geometries as separate screens */ + unique = ecalloc(nn, sizeof(XineramaScreenInfo)); + for (i = 0, j = 0; i < nn; i++) + if (isuniquegeom(unique, j, &info[i])) + memcpy(&unique[j++], &info[i], sizeof(XineramaScreenInfo)); + XFree(info); + nn = j; + if (n <= nn) { /* new monitors available */ + for (i = 0; i < (nn - n); i++) { + for (m = mons; m && m->next; m = m->next); + if (m) + m->next = createmon(); + else + mons = createmon(); + } + for (i = 0, m = mons; i < nn && m; m = m->next, i++) + if (i >= n + || unique[i].x_org != m->mx || unique[i].y_org != m->my + || unique[i].width != m->mw || unique[i].height != m->mh) + { + dirty = 1; + m->num = i; + m->mx = m->wx = unique[i].x_org; + m->my = m->wy = unique[i].y_org; + m->mw = m->ww = unique[i].width; + m->mh = m->wh = unique[i].height; + updatebarpos(m); + } + } else { /* less monitors available nn < n */ + for (i = nn; i < n; i++) { + for (m = mons; m && m->next; m = m->next); + while ((c = m->clients)) { + dirty = 1; + m->clients = c->next; + detachstack(c); + c->mon = mons; + attach(c); + attachstack(c); + } + if (m == selmon) + selmon = mons; + cleanupmon(m); + } + } + free(unique); + } else +#endif /* XINERAMA */ + { /* default monitor setup */ + if (!mons) + mons = createmon(); + if (mons->mw != sw || mons->mh != sh) { + dirty = 1; + mons->mw = mons->ww = sw; + mons->mh = mons->wh = sh; + updatebarpos(mons); + } + } + if (dirty) { + selmon = mons; + selmon = wintomon(root); + } + return dirty; +} + +void +updatenumlockmask(void) +{ + unsigned int i, j; + XModifierKeymap *modmap; + + numlockmask = 0; + modmap = XGetModifierMapping(dpy); + for (i = 0; i < 8; i++) + for (j = 0; j < modmap->max_keypermod; j++) + if (modmap->modifiermap[i * modmap->max_keypermod + j] + == XKeysymToKeycode(dpy, XK_Num_Lock)) + numlockmask = (1 << i); + XFreeModifiermap(modmap); +} + +void +updatesizehints(Client *c) +{ + long msize; + XSizeHints size; + + if (!XGetWMNormalHints(dpy, c->win, &size, &msize)) + /* size is uninitialized, ensure that size.flags aren't used */ + size.flags = PSize; + if (size.flags & PBaseSize) { + c->basew = size.base_width; + c->baseh = size.base_height; + } else if (size.flags & PMinSize) { + c->basew = size.min_width; + c->baseh = size.min_height; + } else + c->basew = c->baseh = 0; + if (size.flags & PResizeInc) { + c->incw = size.width_inc; + c->inch = size.height_inc; + } else + c->incw = c->inch = 0; + if (size.flags & PMaxSize) { + c->maxw = size.max_width; + c->maxh = size.max_height; + } else + c->maxw = c->maxh = 0; + if (size.flags & PMinSize) { + c->minw = size.min_width; + c->minh = size.min_height; + } else if (size.flags & PBaseSize) { + c->minw = size.base_width; + c->minh = size.base_height; + } else + c->minw = c->minh = 0; + if (size.flags & PAspect) { + c->mina = (float)size.min_aspect.y / size.min_aspect.x; + c->maxa = (float)size.max_aspect.x / size.max_aspect.y; + } else + c->maxa = c->mina = 0.0; + c->isfixed = (c->maxw && c->maxh && c->maxw == c->minw && c->maxh == c->minh); +} + +void +updatestatus(void) +{ + if (!gettextprop(root, XA_WM_NAME, stext, sizeof(stext))) + strcpy(stext, "dwm-"VERSION); + drawbar(selmon); +} + +void +updatetitle(Client *c) +{ + if (!gettextprop(c->win, netatom[NetWMName], c->name, sizeof c->name)) + gettextprop(c->win, XA_WM_NAME, c->name, sizeof c->name); + if (c->name[0] == '\0') /* hack to mark broken clients */ + strcpy(c->name, broken); +} + +void +updatewindowtype(Client *c) +{ + Atom state = getatomprop(c, netatom[NetWMState]); + Atom wtype = getatomprop(c, netatom[NetWMWindowType]); + + if (state == netatom[NetWMFullscreen]) + setfullscreen(c, 1); + if (wtype == netatom[NetWMWindowTypeDialog]) + c->isfloating = 1; +} + +void +updatewmhints(Client *c) +{ + XWMHints *wmh; + + if ((wmh = XGetWMHints(dpy, c->win))) { + if (c == selmon->sel && wmh->flags & XUrgencyHint) { + wmh->flags &= ~XUrgencyHint; + XSetWMHints(dpy, c->win, wmh); + } else + c->isurgent = (wmh->flags & XUrgencyHint) ? 1 : 0; + if (wmh->flags & InputHint) + c->neverfocus = !wmh->input; + else + c->neverfocus = 0; + XFree(wmh); + } +} + +void +view(const Arg *arg) +{ + if ((arg->ui & TAGMASK) == selmon->tagset[selmon->seltags]) + return; + selmon->seltags ^= 1; /* toggle sel tagset */ + if (arg->ui & TAGMASK) + selmon->tagset[selmon->seltags] = arg->ui & TAGMASK; + focus(NULL); + arrange(selmon); +} + +Client * +wintoclient(Window w) +{ + Client *c; + Monitor *m; + + for (m = mons; m; m = m->next) + for (c = m->clients; c; c = c->next) + if (c->win == w) + return c; + return NULL; +} + +Monitor * +wintomon(Window w) +{ + int x, y; + Client *c; + Monitor *m; + + if (w == root && getrootptr(&x, &y)) + return recttomon(x, y, 1, 1); + for (m = mons; m; m = m->next) + if (w == m->barwin) + return m; + if ((c = wintoclient(w))) + return c->mon; + return selmon; +} + +/* 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. */ +int +xerror(Display *dpy, XErrorEvent *ee) +{ + if (ee->error_code == BadWindow + || (ee->request_code == X_SetInputFocus && ee->error_code == BadMatch) + || (ee->request_code == X_PolyText8 && ee->error_code == BadDrawable) + || (ee->request_code == X_PolyFillRectangle && ee->error_code == BadDrawable) + || (ee->request_code == X_PolySegment && ee->error_code == BadDrawable) + || (ee->request_code == X_ConfigureWindow && ee->error_code == BadMatch) + || (ee->request_code == X_GrabButton && ee->error_code == BadAccess) + || (ee->request_code == X_GrabKey && ee->error_code == BadAccess) + || (ee->request_code == X_CopyArea && ee->error_code == BadDrawable)) + return 0; + fprintf(stderr, "dwm: fatal error: request code=%d, error code=%d\n", + ee->request_code, ee->error_code); + return xerrorxlib(dpy, ee); /* may call exit */ +} + +int +xerrordummy(Display *dpy, XErrorEvent *ee) +{ + return 0; +} + +/* Startup Error handler to check if another window manager + * is already running. */ +int +xerrorstart(Display *dpy, XErrorEvent *ee) +{ + die("dwm: another window manager is already running"); + return -1; +} + +void +zoom(const Arg *arg) +{ + Client *c = selmon->sel; + + if (!selmon->lt[selmon->sellt]->arrange + || (selmon->sel && selmon->sel->isfloating)) + return; + if (c == nexttiled(selmon->clients)) + if (!c || !(c = nexttiled(c->next))) + return; + pop(c); +} + +int +main(int argc, char *argv[]) +{ + if (argc == 2 && !strcmp("-v", argv[1])) + die("dwm-"VERSION); + else if (argc != 1) + die("usage: dwm [-v]"); + if (!setlocale(LC_CTYPE, "") || !XSupportsLocale()) + fputs("warning: no locale support\n", stderr); + if (!(dpy = XOpenDisplay(NULL))) + die("dwm: cannot open display"); + checkotherwm(); + setup(); +#ifdef __OpenBSD__ + if (pledge("stdio rpath proc exec", NULL) == -1) + die("pledge"); +#endif /* __OpenBSD__ */ + scan(); + run(); + cleanup(); + XCloseDisplay(dpy); + return EXIT_SUCCESS; +} diff --git a/dwm/dwm.o b/dwm/dwm.o new file mode 100644 index 0000000000000000000000000000000000000000..a2d667b6be7f04efb7275613ec1fd8e19be0a975 GIT binary patch literal 56728 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 literal 0 HcmV?d00001 diff --git a/dwm/dwm.png b/dwm/dwm.png new file mode 100644 index 0000000000000000000000000000000000000000..b1f9ba7e5f4cc7350ee2392ebcea5fcbe00fb49b GIT binary patch literal 373 zcmeAS@N?(olHy`uVBq!ia0vp^2Y@($g9*gC@m3f}u_bxCyDx`7I;J! zGca%iWx0hJ8D`Cq01C2~c>21sUt<^MF=V?Ztt9{yk}YwKC~?lu%}vcKVQ?-=O)N=G zQ7F$W$xsN%NL6t6^bL5QqM8R(c+=CxF{I+w+q;fj4F)_6j>`Z3pZ>_($QEQ&92OXP z%lpEKGwG8$G-U1H{@Y%;mx-mNK|p|siBVAj$Z~Mt-~h6K0!}~{PyozQ07(f5fTdVi zm=-zT`NweeJ#%S&{fequZGmkDDC*%x$$Sa*fAP=$`nJkhx1Y~k<8b2;Hq)FOdV=P$ q&oWzoxz_&nv&n0)xBzV8k*jsxheTIy&cCY600f?{elF{r5}E*x)opSB literal 0 HcmV?d00001 diff --git a/dwm/transient.c b/dwm/transient.c new file mode 100644 index 0000000..040adb5 --- /dev/null +++ b/dwm/transient.c @@ -0,0 +1,42 @@ +/* cc transient.c -o transient -lX11 */ + +#include +#include +#include +#include + +int main(void) { + Display *d; + Window r, f, t = None; + XSizeHints h; + XEvent e; + + d = XOpenDisplay(NULL); + if (!d) + exit(1); + r = DefaultRootWindow(d); + + f = XCreateSimpleWindow(d, r, 100, 100, 400, 400, 0, 0, 0); + h.min_width = h.max_width = h.min_height = h.max_height = 400; + h.flags = PMinSize | PMaxSize; + XSetWMNormalHints(d, f, &h); + XStoreName(d, f, "floating"); + XMapWindow(d, f); + + XSelectInput(d, f, ExposureMask); + while (1) { + XNextEvent(d, &e); + + if (t == None) { + sleep(5); + t = XCreateSimpleWindow(d, r, 50, 50, 100, 100, 0, 0, 0); + XSetTransientForHint(d, t, f); + XStoreName(d, t, "transient"); + XMapWindow(d, t); + XSelectInput(d, t, ExposureMask); + } + } + + XCloseDisplay(d); + exit(0); +} diff --git a/dwm/util.c b/dwm/util.c new file mode 100644 index 0000000..fe044fc --- /dev/null +++ b/dwm/util.c @@ -0,0 +1,35 @@ +/* See LICENSE file for copyright and license details. */ +#include +#include +#include +#include + +#include "util.h" + +void * +ecalloc(size_t nmemb, size_t size) +{ + void *p; + + if (!(p = calloc(nmemb, size))) + die("calloc:"); + return p; +} + +void +die(const char *fmt, ...) { + va_list ap; + + va_start(ap, fmt); + vfprintf(stderr, fmt, ap); + va_end(ap); + + if (fmt[0] && fmt[strlen(fmt)-1] == ':') { + fputc(' ', stderr); + perror(NULL); + } else { + fputc('\n', stderr); + } + + exit(1); +} diff --git a/dwm/util.h b/dwm/util.h new file mode 100644 index 0000000..f633b51 --- /dev/null +++ b/dwm/util.h @@ -0,0 +1,8 @@ +/* See LICENSE file for copyright and license details. */ + +#define MAX(A, B) ((A) > (B) ? (A) : (B)) +#define MIN(A, B) ((A) < (B) ? (A) : (B)) +#define BETWEEN(X, A, B) ((A) <= (X) && (X) <= (B)) + +void die(const char *fmt, ...); +void *ecalloc(size_t nmemb, size_t size); diff --git a/dwm/util.o b/dwm/util.o new file mode 100644 index 0000000000000000000000000000000000000000..53a67f47339e520b38ba4cdde751ff98c1a6258b GIT binary patch literal 2272 zcmbW1Pe>GT6vyAXm7A$+*}>4m9)5%iI_wCQY$dC%I@yq#M%JNiTt{5mf6R_y5oBbC zMHW1C2|5UM=+GgeW1!F>Z*>pKOHkOsP8|}q_hx=S{V~m6e9P>6^LgL*`@Q-7<~I+A z^x?Wd0E7jg5BBc_1z0=0A19nJ0Zq^dlUn(&_UP9ZBCTw*UadUE5?WbjnqE$_aUG-S z@`Zv`%Z`&-3>Q&nn^?4;`;aQ`X<7Fz=CjrE~boU9Fb2haT^-{0KwxNfw0hv#~E z?i9~mZw_(d@2LMn@Vn}l4f^ghKoEa^7Q~VKEC|xeAqj#|FWv2jLHNICm(&{P$bK=8 zn$Vw+`ghg}*>SvDxJGugu6|sp!Ozv;eKmNp2ESZ`k0Yo4Mc>YTBAck8hXp&6Rnnm7 zgE5%Rm{783P0NDRh(0#ZucyX_hbM=oQd9i{`cMk)&K0dp-kw91l{ND)S6r~uP{e|Q z1?FPL2Gf0QA&2vk)!}svuidI^&j_N?j?-ET{(>NySO3<V!uEs-OJg=40_vg z{k(`QGixZexoCqjYuE-TGo=zJR)KCvt|}@h=|V1N=24O_*rqaaiHqv)cN7d$h zYR)oprfB29K7y6>4e*|z1TwNZN=oLrg?7x*S|ucn6B0xV@d8>)3dPB$^G0?WFXG-W z%7Aww5~F!>7cv_2p!F&rvj@FwVt&!D$cg!dPhf)Lm=JJ3$_1SsF6xB8gn1FmQp*qe zXm4>Qzmqi1BCn_Kh-}{Hxq(T){R{C=t#~flKdKe=V*h9w@7#!Blh41!J$*aG7?c