Martin Klöckner's Webpage


dwm: The dynamic window manager

23-Oct-2021 (last update 02-nov-2024)

A window manager is a software that can manage the layout and appearance of every window spawned in your desktop, most people confuse them with desktop enviroments, which aren’t the same since a desktop enviroment is more like an ecosystem, they come with a more ‘complete’ set of tools, like a basic web browser, a terminal emulator or a graphical calculator, an example of desktop enviroment would be gnome, xfce or kde plasma; instead a window manager is only the program that manages the windows spawned, although there are window managers that come with a little more, like docks or taskbars (openbox for example).

In my case I use dwm, which is a window manager written in C developed by suckless software. The most relevant thing of this window manager is that out of the box it comes with the most basic functionallity, and if you want to extend it you need to patch it (I explain patching later).

By default dwm comes with 9 workspaces, in which you can open as many windows as you please; to spawn a new window for example a web browser you need to assign it a keybinding or use an application launcher like dmenu

Screenshot

Installing dwm

Requisites

As you can see above you can only install dwm on GNU/Linux or BSD based distros, unfortunally dwm is not available for Windows users and I’m not sure if there is an alternative.

Steps

In order to install dwm download the source code from suckless.org/dwm, you can clone the repo from git.suckless.org/dwm or download it as a tar file.

After you obtain the source code navigate to the root folder of the source code and execute the following command

$ sudo make install

after that you can log out of you user account, if you use a display manager, select dwm as window manager on it and log back in, if you don’t use a display manager, you need to edit your .xinitrc file located at your home folder, so that when you start Xorg dwm gets launched, you do this by adding exec dwm to the end of the .xinitrc file, its important that you add it to the end of the file.

exec dwm

Then you can start the X server by executing startx on a tty and dwm should start without any problems.

This is what vanilla dwm looks like:

Vanilla dwm

Customizing dwm

Setting a wallpaper

Before getting into dwm configuration, let’s add a wallpaper to make things look a little bit nicer. You can set a wallpaper from different ways, the simpler for me is installing feh and the executing the following command

$ feh --bg-fill <image>

you can also avoid setting –bg-fill and chossing other feh options to set the wallpaper (read man feh).

Then add .fehbg & to your .xinitrc so the wallpaper gets loaded when Xorg starts, it is important that you add it before dwm gets executed, otherwise it will never be ran

$HOME/.fehbg &

exec dwm

A graphical way of setting a wallpaper is with nitrogen, just install nitrogen open it with dmenu (MOD+p by default), include the folder where you have your wallpaper, select the image you want and the apply. To make changes persistent on every startup add this to you .xinirc file

nitrogen --restore

exec dwm

dwm configuration

Basically you configure dwm by editing its source code, there is a C header file, named config.h in the root folder of dwm, if you open it with a text editor you can see a lot of variables, for example the line static const int topbar = 0 defines a variable named topbar which you can set to 1 or 0, to select if the status bar should spawn in the top or bottom of the screen respectively. You can also change the MOD key (by default its left alt), I like to remap mine to left Super (windows key).

After every change you make to the source code you need to recompile dwm.

Show information on the status bar

dwm by default won’t show any information on the status bar, this is done by using the xsetroot utility, which sets the value of WM_NAME enviroment variable, the content of this variable is automatically displayed by dwm on the right side of the statusbar. Lets assume you want to set the clock and date on the status bar, well you could accomplish this by executing the following command

$ xsetroot -iname $(date)

and all the output of the command date would be stored on the WM_NAME variable causing dwm to print the date on the statusbar, this makes dwm status bar highly scriptable, in fact there are a ton of status bar implementations, the one that I use is called dwmblocks and its also written in C and the configuration its pretty much the same as dwm, in order to get information you need to have scripts that prints the desired data to stdout, then you can include them on the config.h of dwmblocks. It is important that the scripts are on your user’s PATH, otherwise it won’t work.

Getting emoji support on dwm

dwm by default doesn’t come with emoji support, if you want to render an emoji in the status bar, it’s going to either show it as a box or in the worst case crash.

In order to get emoji support first make sure you have installed libxft, then you need to get a font with emoji support, I’m using JoyPixels® font you can also use Google’s noto font, or any other font that comes with emoji support, then open dwm config.def.h and append to the fonts array the name of the font you want to use as fallback for the first font, since the emojis are not being printed because the font used doesn’t include emojis, my config looks like this:

static const char *fonts[] = { "Source Code Pro:style=Regular:size=9",
                                "JoyPixels:style=Regular:size=8",
                                 "DejaVu Sans" };

I’ve also added DejaVu Sans to the fonts array because, sometimes, the emojis where being displayed with a little box or square next to them, this was a quick solution.

After you setup the font you need to remove or comment a chunk of code from drw.c, between lines 136-150 there is a function named iscol, you need to remove it or comment it, since this causes dwm to crash.

/* 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;
}

Finally you can recompile dwm and, if everything went fine, you will get emoji support.

Patching dwm

Since dwm is a simple program than doesn’t include much features, if you want to extend it, for example by adding a systray to the status bar, you need to patch dwm. To do this first you need to download the patch from suckless.org/patches, then make sure you got it ‘patch’ installed, although I think it comes with most linux distributions by default nowdays.

To apply a patch navigate to dwm’s root folder and execute this command

$ patch -p1 < <file.diff>

being file.diff the patch file downloaded previoulsy of course.

If you never patched dwm before then probably no errors would be reported, but if you already have applied a ton of patches, (or sometimes just a couple) then probably you would get a HUNK ## FAILED, in this case you need to get your hands dirty, and manually patch all the files that failed, you do this by opening the files with .rej extension (short for rejected), and the corresponding file to be patched, for example dwm.c and dwm.c.rej, and then you add all the chunks of code from the rejected file into the corresponding place in the non rejected file, you know where you should put the chunks of code because in the rej file you can see at the start of every chunk there is a ‘@@’ followed by a number of line; also lines starting with plus means add, and minus means delete, if I’m not clear you should google how to use diff and patch to modify a file.

More screenshots

Debugging 6502 emulator with dwm, vim and tmux

Debugging 6502 emulator  

Floating window manager?

Floating show

Useful links


Back to top