A small CLI tool to freeze the screen of a wlroots compositor, this can be useful to, for example, take a screenshot. Supports multiple monitors & fractional scaling.
Run wayfreeze
, click or press escape to exit.
Usage: wayfreeze [OPTIONS]
Options:
--hide-cursor
Hide cursor when freezing the screen
--before-freeze-cmd <BEFORE_FREEZE_CMD>
Command to run before freezing the screen
--before-freeze-timeout <BEFORE_FREEZE_TIMEOUT>
Amount of milliseconds to wait between before-freeze-cmd and freezing the screen
--after-freeze-cmd <AFTER_FREEZE_CMD>
Command to run after freezing the screen
--after-freeze-timeout <AFTER_FREEZE_TIMEOUT>
Amount of milliseconds to wait between freezing the screen and running after-freeze-cmd
-h, --help
Print help
-V, --version
Print version
Example usage with Grim & Slurp:
# for e.g. Hyprland or Sway (as of Sway 1.10-rc4):
wayfreeze & PID=$!; sleep .1; grim -g "$(slurp)" - | wl-copy; kill $PID
# or:
wayfreeze --after-freeze-cmd 'grim -g "$(slurp)" - | wl-copy; killall wayfreeze'
# for compositors that order layer surfaces the other way around:
wayfreeze --before-freeze-cmd 'grim -g "$(slurp)" - | wl-copy; killall wayfreeze' --before-freeze-timeout 10
Note: the Wayland specification states the following: "Multiple surfaces can share a single layer, and ordering within a single layer is undefined." This means that compositors can put new layer surfaces over or under existing layer surfaces (given they're on the same layer), and both of those options are compliant to the spec. Compositors like e.g. Hyprland and Sway (since Sway 1.10-rc4) put new layer surfaces over older ones, while e.g. Sway (before 1.10-rc4) put new layer surfaces underneath already existing ones. If you're unsure how your compositor handles this, just try both commands while playing a video or something. One will work, the other one won't.
Wayfreeze can be installed either by using nixpkgs-unstable or flake.
Add this to your configuration and rebuild your system:
environment.systemPackages = [ pkgs.wayfreeze ];
Add this repository as a flake to your inputs:
wayfreeze.url = "github:jappie3/wayfreeze";
Define the package and then rebuild your system:
environment.systemPackages = [ inputs.wayfreeze.packages.${pkgs.system}.wayfreeze ];
There is an AUR package available at https://aur.archlinux.org/packages/wayfreeze-git.
Note: I do not maintain this package myself.
First, make sure you have Rust & Cargo installed. Then clone this repo & run cargo build --release
, the compiled binary should be under ./target/release/wayfreeze
.
The following protocols should be supported by your compositor:
wlr-layer-shell-unstable-v1
-> used for creating & rendering a layer surfacewlr-screencopy-unstable-v1
-> used for copying the current output to a client bufferwp-fractional-scale-v1
-> to support fractional scalingwp-viewporter
-> for scaling the surface
In no particular order, here are some resources that were helpful when creating this tool & learning about the Wayland protocol:
- https://wayland.app/protocols/wayland
- https://github.com/hiasen/wayland-rust-client-experiment/
- https://github.com/Smithay/wayland-window/blob/master/examples/simple_window.rs
- https://github.com/rafaelrc7/wayland-pipewire-idle-inhibit/
- https://levans.fr/rust_wayland_1.html
- https://bugaevc.gitbooks.io/writing-wayland-clients/content/black-square/allocating-a-buffer.html
- https://danyspin97.org/talks/writing-a-wayland-wallpaper-daemon-in-rust
- https://docs.rs/wayland-client/latest/wayland_client
- https://wayland-book.com