Driftwm
A trackpad-first infinite canvas Wayland compositor.
**driftwm** is A trackpad-first infinite canvas Wayland compositor. The project is written primarily in Rust, distributed under the Other license, first published in 2026. It has gained significant community traction with 1,378 stars and 48 forks on GitHub. Key topics include: compositor, gestures, infinite-canvas, linux, rust.
https://github.com/user-attachments/assets/df24e442-6ad0-4520-9491-cb666da06d05
Traditional window managers arrange windows to fit your screen. Stacking compositors do so by piling windows on top of each other; tiling compositors do so by squeezing them to fit and utilizing workspaces.
driftwm is an infinite-canvas compositor: windows live at their native size on an infinite 2D canvas, and your display is a camera viewing it. When two windows come close, they snap together, forming implicit groups that can be moved, resized, and viewed together. No tiling, no workspaces, window overlaps happen only on purpose.
Designed with laptops in mind: navigation and window management are trackpad-first; the infinite canvas makes the most of a small screen.
Built on smithay. Inspired by vxwm; borrows implementation details from niri.
[!WARNING]
This is experimental software, primarily built with AI.
Features
Pan & zoom
https://github.com/user-attachments/assets/a5f14739-7762-4515-abb1-0de6990de4a3
Infinite 2D canvas with viewport panning, zoom, and scroll momentum. A quick
flick carries the viewport smoothly until friction stops it.
| Input | Action | Context |
|---|---|---|
| 3-finger swipe | Pan viewport | anywhere |
| Trackpad scroll | Pan viewport | on-canvas |
Mod + LMB drag | Pan viewport | anywhere |
Mod+Ctrl + arrow | Pan viewport | — |
| 2-finger pinch | Zoom | on-canvas |
| 3-finger pinch | Zoom | anywhere |
Mod + scroll | Zoom at cursor | anywhere |
Mod+= / Mod+- | Zoom in / out | — |
Mod+0 / Mod+Z | Reset zoom to 1.0 | — |
Window navigation
https://github.com/user-attachments/assets/5b7d89cd-b065-4309-ae74-30bfe68a8abb
Jump to the nearest window in any direction via cone search. MRU cycling
(Alt-Tab) with hold-to-commit. Zoom-to-fit shows all windows at once.
Configurable anchors act as navigation targets for directional jumps even
with no window there — useful for areas with pinned widgets.
| Input | Action |
|---|---|
| 4-finger swipe | Jump to nearest window (natural direction) |
Mod+Ctrl + LMB drag | Jump to nearest window (natural direction) |
Mod + arrow | Jump to nearest window in direction |
Alt-Tab / Alt-Shift-Tab | Cycle windows (MRU) |
4-finger pinch in / Mod+W | Zoom-to-fit (overview) |
4-finger pinch out / Mod+A | Home toggle (origin and back) |
4-finger hold / Mod+C | Center focused window |
Mod+1-4 | Jump to bookmarked canvas position |
All 4-finger navigation gestures also work as Mod + 3-finger for smaller
trackpads.
Snapping
https://github.com/user-attachments/assets/8a468e69-8887-4d27-8457-cdd2753948ca
Move window with 3-finger doubletap-swipe or Alt + drag. Resize with Alt + 3-finger swipe. Snapping kicks in as edges approach each other. Drag past the viewport edge and the canvas auto-pans.
Snapped windows form a cluster. Two benefits: neighbors stay visible at your view's edge for spatial context, and Shift + any move/resize/fit action acts on the whole cluster. Shuffle a layout in one drag, resize a row of panes proportionally, or scope an overview to just the cluster (Mod+Shift+W). No explicit grouping to manage.
[!TIP]
While dragging a window, keyboard shortcuts still work. UseMod+1-4
to jump to a bookmark orMod+Ato go home — your held window comes with you.
Fit-window (Mod+M) is the maximize analogue — centers the viewport, resets
zoom to 1.0, and resizes the window to fill the screen. Toggle again to
restore. Fullscreen (Mod+F) is a viewport mode, not a window state — any canvas
action (launching an app, navigating) naturally exits it.
| Input | Action |
|---|---|
| 3-finger doubletap-swipe | Move window |
Alt + LMB drag | Move window |
Alt+Shift + LMB drag | Move snapped windows |
Alt + 3-finger swipe | Resize window |
Alt+Shift + 3-finger swipe | Resize snapped window |
Alt + RMB drag | Resize window |
Alt + MMB click / Mod+M | Fit window (maximize/restore) |
Alt+Shift + MMB click / Mod+Shift+M | Fit snapped window |
Mod + 4-finger pinch in / Mod+Shift+W | Zoom-to-fit snapped windows |
Alt + 2-finger pinch in/out | Fit window |
Alt + 3-finger pinch in/out | Toggle fullscreen |
Mod + MMB click / Mod+F | Toggle fullscreen |
Mod+Shift + arrow | Nudge window 20px |
Infinite background
https://github.com/user-attachments/assets/6e9eb7f7-0c73-4fdd-b7aa-230b8ff0a172
https://github.com/user-attachments/assets/fb1cd5a1-242c-45d7-b302-952a15aaa24d
The background is part of the canvas — it scrolls and zooms with the viewport,
not stuck to the screen. This gives spatial awareness when panning.
Four modes:
shader— procedural GLSL, animated or static, optionally sampling an image viatexture. Default is a dot grid. See docs/shaders.md to write your own. Bundled shaders live inextras/wallpapers/{static,animated,textured}/.tile— PNG/JPG (single texture, tiled infinitely), or a tiled pyramidal TIFF for gigapixel wallpapers. Setmirror_tile = trueto mirror-fold a non-seamless image so it tiles without seams (kaleidoscope look).wallpaper— single image stretched to fill viewport (does not scroll/zoom) — a classic desktop wallpaper.none— no built-in background, so an externalwlr-layer-shellwallpaper daemon (swaybg,swww,mpvpaperfor live video) becomes the wallpaper instead.
[!NOTE]
GPU cost scales with what a shader reads: one that reads no viewport uniforms renders once (as cheap aswallpaper); readingu_camera/u_zoomredraws on pan/zoom; readingu_timeredraws every frame. Tiles redraw on pan/zoom;wallpaperrenders once.
toml[background] type = "shader" path = "~/.config/driftwm/bg.glsl" # texture = "~/Pictures/img.jpg" # if it's a texture-based shader # Or: type = "tile", path = "~/Pictures/tile.png" # Or: type = "tile", path = "~/Pictures/world.tif" # pyramidal TIFF # Or: type = "wallpaper", path = "~/Pictures/wallpaper.jpg" # Or: type = "none" # external wallpaper daemon (swaybg/mpvpaper/…)
Window rules
https://github.com/user-attachments/assets/af603001-9f08-4d42-b50a-0342d06e954b
Match windows by app_id and/or title (glob patterns) and control position,
size, decorations, blur, opacity, key pass-through, and placement — fields
combine freely.
Two special placement modes: widget = true fixes a window to the canvas
(immovable, below normal windows, out of Alt-Tab — clocks, trays, and
layer-shell surfaces like waybar); pinned_to_screen = true fixes it to the
screen instead, so it ignores pan/zoom and floats above normal windows
(Picture-in-Picture, call toolbars) — toggleable live with Mod+T.
toml# Frosted-glass terminal [[window_rules]] app_id = "Alacritty" opacity = 0.85 blur = true # Desktop widget — pinned to the canvas, borderless [[window_rules]] app_id = "my-clock" position = [50, 50] widget = true decoration = "none"
[!TIP]
To find a window'sapp_idortitle, rundriftwm msg state— it lists
every open window with its app ID, title, position, and size.
See docs/window-rules.md for more details.
Multi-monitor
https://github.com/user-attachments/assets/3f6cc3a8-a4ed-4d78-80fc-d5a92478c48f
Multiple monitors are independent viewports on the same canvas. An outline on each monitor shows where the
other monitors' viewports are. Cursor crosses between monitors freely; dragged
windows teleport to the target viewport's canvas position.
| Input | Action |
|---|---|
Mod+Alt + arrow | Send window to adjacent output |
Panels, docks & taskbars
https://github.com/user-attachments/assets/83c2ad30-fbfa-4cf2-aa47-905826889dcb
Layer shell surfaces (waybar, fuzzel, mako) work as expected. Foreign toplevel
management means your dock/taskbar shows all windows — click one and the
viewport pans to it and centers it. See extras/ for a fuzzel
window-search script that lets you search and jump to any open window.
Everything else
- New window placement: in viewport center (default), under cursor, or snapped adjacent to the focused window's cluster
- Click-to-focus (default) or focus-follows-mouse (sloppy focus)
- Session lock (swaylock), idle notify (swayidle/hypridle)
- Screen capture: screencasting (OBS, Firefox, Discord) and screenshots, incl. built-in canvas/DPI capture
- 40+ Wayland protocols
- IPC control: script the compositor over a Unix socket with
driftwm msg
Install
Arch Linux (AUR)
bashyay -S driftwm
or for latest main:
bashyay -S driftwm-git
NixOS / Nix
A flake.nix is included. To build:
bashnix build
For development (provides native deps, uses your system Rust):
bashnix develop cargo build cargo run
To enable driftwm on NixOS, you can import and use the provided NixOS module in your configuration.
Using Flakes:
nix# flake.nix { inputs = { nixpkgs.url = "github:NixOS/nixpkgs/nixos-unstable"; driftwm.url = "github:malbiruk/driftwm"; }; outputs = { self, nixpkgs, driftwm, ... }: { nixosConfigurations.myHost = nixpkgs.lib.nixosSystem { modules = [ driftwm.nixosModules.default ./configuration.nix ]; }; }; }
Then, enable it in your configuration:
nix# configuration.nix { programs.driftwm.enable = true; }
Alternatively, without flakes (by importing the flake's output directly):
nixlet driftwm-flake = builtins.getFlake "github:malbiruk/driftwm"; in { imports = [ driftwm-flake.nixosModules.default ]; programs.driftwm.enable = true; }
NixOS Module Options
The NixOS module provides the following options under programs.driftwm:
enable: Whether to enabledriftwm(defaults tofalse).package: The package containing thedriftwmcompositor binary.
By default, the module enables XWayland support via xwayland-satellite by defaulting programs.xwayland.enable to true. If you want to disable or explicitly enable it, configure:
nixprograms.xwayland.enable = true; # or false to disable XWayland and xwayland-satellite
Build from source
Requires Rust 1.88+ (edition 2024).
Install build dependencies:
Fedora:
bashsudo dnf install libseat-devel libdisplay-info-devel libinput-devel mesa-libgbm-devel libxkbcommon-devel
Ubuntu/Debian:
bashsudo apt install libseat-dev libdisplay-info-dev libinput-dev libudev-dev libgbm-dev libxkbcommon-dev libwayland-dev
Arch Linux:
bashsudo pacman -S libdisplay-info libinput seatd mesa libxkbcommon
[!NOTE]
Ubuntu 24.04 ships Rust 1.75 which is too old. Install via
rustup instead ofapt install rustc.
Then build and install:
bashgit clone https://github.com/malbiruk/driftwm.git cd driftwm make build sudo make install
To uninstall, run sudo make uninstall from the repository.
Optional runtime dependencies
driftwm runs standalone — none of these are required — but each enables or
improves a feature:
xwayland-satellite(≥ 0.7) — X11 app support (see below).xdg-desktop-portal+xdg-desktop-portal-wlr(≥ 0.8.0) orxdg-desktop-portal-cosmic— screencasting. wlr needs a dmenu-style picker in$PATH(wmenu/wofi/rofi/bemenu/mew/fuzzel) to choose what to share.grim+slurp— screenshots (+ cropping to region). driftwm also has a built-in canvas/DPI capture: see IPC › Screenshots.adwaita-fonts— renders SSD title bars inAdwaita Sansto match GTK apps; without it a generic sans-serif is substituted. Font, size, weight, and alignment are configurable under[decorations].- A cursor theme — most desktops set one up already; on a bare install driftwm falls back to a basic built-in arrow.
X11 apps run through xwayland-satellite,
which driftwm spawns at startup, exporting DISPLAY=:N so X11 clients connect
transparently — no extra config beyond having the binary in $PATH.
- Arch:
sudo pacman -S xwayland-satellite - Fedora:
sudo dnf install xwayland-satellite - NixOS:
pkgs.xwayland-satellite - Debian/Ubuntu: not yet packaged —
cargo install --locked xwayland-satellite
If satellite isn't found at startup, driftwm logs a warning and continues without
X11 support. You can override the binary path or disable the integration in
config.reference.toml under [xwayland].
Running
driftwm auto-detects whether it's running nested (inside an existing Wayland
session) or on real hardware (from a TTY). Just run driftwm. For display
manager integration, select "driftwm" from the session menu.
[!TIP]
When launched by a display manager, driftwm runs as a systemd user service — view logs withjournalctl --user -u driftwm.service(add-fto follow). Run directly and logs go to stderr.
Quick start
mod is Super by default. Terminal and launcher are auto-detected (foot/alacritty/kitty, fuzzel/wofi/bemenu); override in config.
| Shortcut | Action |
|---|---|
mod+return | Open terminal |
mod+d | Open launcher |
mod+q | Close window |
mod+l | Lock screen |
mod+ctrl+shift+q | Quit |
Feature-specific bindings (navigation, zoom, snap) are in their respective sections above.
Configuration
Config file: ~/.config/driftwm/config.toml (respects XDG_CONFIG_HOME).
bashmkdir -p ~/.config/driftwm cp /etc/driftwm/config.reference.toml ~/.config/driftwm/config.toml
Missing file uses built-in defaults. Partial configs merge with defaults —
only specify what you want to change. Use "none" to unbind a default binding.
Validate without starting: driftwm --check-config.
toml# Launch programs at startup autostart = ["waybar", "swaync", "swayosd-server"]
Every option is documented in docs/config.md (generated
from config.reference.toml): input settings,
scroll/momentum tuning, snap behavior, decorations, effects, per-output config,
gesture bindings, mouse bindings, and window rules.
Example setup
driftwm is just a compositor — everything else is standard Wayland tooling.
Here are some tools that work well with it:
- waybar — Status bar / taskbar
- crystal-dock — macOS-style dock
- fuzzel / wofi — App launcher
- mako / swaync — Notifications
- swaylock — Lock screen
- swayidle / hypridle — Idle timeout (lock, suspend)
- swayosd — Volume/brightness OSD
- grim + slurp — Screenshots (or the built-in canvas/DPI capture: IPC › Screenshots)
- wlr-randr / wdisplays — Output configuration
- COSMIC Settings — Wi-Fi, Bluetooth, sound (or nm-applet + blueman + pavucontrol)
Compositor-agnostic full Wayland shells like noctalia, wayle, and dank-material-shell should work too (driftwm supports wlr-layer-shell protocol) but without compositor-specific features.
The extras/ directory contains a complete setup — driftwm config,
GLSL shader wallpapers, Python widgets (clock, calendar, system stats, power
menu), waybar with taskbar/tray, fuzzel window-search script, and window rules
tying it all together. Use it as a starting point or steal pieces.
Community
- driftwm-settings — GTK4 GUI config editor
- driftwm-noctalia — noctalia shell fork adapted for driftwm
- Just Enough Shell — minimal QuickShell desktop shell, driftwm-focused
- Gallery — community shaders & rices, share your own
Contributing
See CONTRIBUTING.md.
TL;DR: open an issue before writing non-trivial code, keep PRs small and focused.
Merch
If you want to support the project (or just want a shirt), this is the way.
<p align="left"><img src="assets/tshirt.png" width="400"></p>XL
100 GEL · 37 USD · 2800 RUB
Ships worldwide from Tbilisi.
Order via Telegram, Instagram, or email 2601074@gmail.com.
Revenue goes to me as driftwm's primary maintainer. If you've contributed substantively and want a shirt, drop me a line.
License
GPL-3.0-or-later
Contributors
Showing top 12 contributors by commit count.
