grim-rs 0.1.9

Rust implementation of grim screenshot utility for Wayland
Documentation
# Library Usage Examples (`grim-rs`)

These examples show public API usage for `grim-rs`.

Prerequisites:

- Run inside a Wayland session.
- Use output names returned by `grim.get_outputs()` (examples use placeholders like `DP-1`).

## Backend selection

`Grim::new()` auto-detects the best available capture protocol (`ext-image-copy-capture-v1` preferred, `wlr-screencopy` fallback). You can force a specific backend:

- `Grim::new_ext()` — force `ext-image-copy-capture-v1` (Sway ≥2025, Hyprland, COSMIC)
- `Grim::new_wlr()` — force `wlr-screencopy` (older Sway, River, Wayfire)
- `Backend` enum — `Auto`, `ExtImageCopyCapture`, `WlrScreencopy`

```rust,no_run
use grim_rs::{Backend, Grim};

fn main() -> grim_rs::Result<()> {
    // Auto-detect — works everywhere
    let mut grim = Grim::new()?;
    // Force new protocol (Sway ≥2025, Hyprland, COSMIC)
    let mut grim = Grim::new_ext()?;
    // Force legacy protocol (older compositors)
    let mut grim = Grim::new_wlr()?;
    let result = grim.capture_all()?;
    grim.save_png(result.data(), result.width(), result.height(), "screenshot.png")?;
    Ok(())
}
```

## Basic capture operations

```rust,no_run
use grim_rs::{Region, Grim};

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let result = grim.capture_all()?;
    grim.save_png(result.data(), result.width(), result.height(), "screenshot.png")?;
    let region = Region::new(100, 100, 800, 600);
    let result = grim.capture_region(region)?;
    grim.save_png(result.data(), result.width(), result.height(), "region.png")?;
    let result = grim.capture_output("DP-1")?;
    grim.save_png(result.data(), result.width(), result.height(), "output.png")?;
    Ok(())
}
```

## Getting outputs information

```rust,no_run
use grim_rs::Grim;

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let outputs = grim.get_outputs()?;
    for output in outputs {
        println!("Output: {}", output.name());
        println!(
            "  Position: ({}, {})",
            output.geometry().x(),
            output.geometry().y()
        );
        println!(
            "  Size: {}x{}",
            output.geometry().width(),
            output.geometry().height()
        );
        println!("  Scale: {}", output.scale());
        if let Some(desc) = output.description() {
            println!("  Description: {}", desc);
        }
    }
    Ok(())
}
```

## Capture with scaling

```rust,no_run
use grim_rs::{Region, Grim};

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let result = grim.capture_all_with_scale(0.5)?;
    grim.save_png(result.data(), result.width(), result.height(), "thumbnail.png")?;
    let region = Region::new(0, 0, 1920, 1080);
    let result = grim.capture_region_with_scale(region, 0.8)?;
    grim.save_png(result.data(), result.width(), result.height(), "scaled.png")?;
    let result = grim.capture_output_with_scale("DP-1", 0.5)?;
    grim.save_png(result.data(), result.width(), result.height(), "output_scaled.png")?;
    Ok(())
}
```

## Multiple outputs

```rust,no_run
use grim_rs::{Region, CaptureParameters, Grim};

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let parameters = vec![
        CaptureParameters::new("DP-1").overlay_cursor(true),
        CaptureParameters::new("HDMI-A-1").region(Region::new(0, 0, 1920, 1080)),
    ];
    let results = grim.capture_outputs_with_scale(parameters, 0.5)?;
    for (output_name, result) in results.into_outputs() {
        let filename = format!("{}.png", output_name);
        grim.save_png(result.data(), result.width(), result.height(), &filename)?;
    }
    Ok(())
}
```

## Save to different formats

```rust,no_run
use grim_rs::Grim;

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let result = grim.capture_all()?;
    grim.save_png(result.data(), result.width(), result.height(), "screenshot.png")?;
    grim.save_png_with_compression(result.data(), result.width(), result.height(), "compressed.png", 9)?;
    grim.save_jpeg(result.data(), result.width(), result.height(), "screenshot.jpg")?;
    grim.save_jpeg_with_quality(result.data(), result.width(), result.height(), "quality.jpg", 95)?;
    Ok(())
}
```

## Convert to bytes

```rust,no_run
use grim_rs::Grim;

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let result = grim.capture_all()?;
    let png_bytes = grim.to_png(result.data(), result.width(), result.height())?;
    println!("PNG size: {} bytes", png_bytes.len());
    let _png_bytes = grim.to_png_with_compression(result.data(), result.width(), result.height(), 9)?;
    let jpeg_bytes = grim.to_jpeg(result.data(), result.width(), result.height())?;
    println!("JPEG size: {} bytes", jpeg_bytes.len());
    let _jpeg_bytes = grim.to_jpeg_with_quality(result.data(), result.width(), result.height(), 85)?;
    Ok(())
}
```

## Write encoded data to stdout

```rust,no_run
use grim_rs::Grim;

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let result = grim.capture_all()?;
    grim.write_png_to_stdout(result.data(), result.width(), result.height())?;
    grim.write_png_to_stdout_with_compression(result.data(), result.width(), result.height(), 6)?;
    grim.write_jpeg_to_stdout(result.data(), result.width(), result.height())?;
    grim.write_jpeg_to_stdout_with_quality(result.data(), result.width(), result.height(), 90)?;
    Ok(())
}
```

## Read region from stdin

```rust,no_run
use grim_rs::Grim;

fn main() -> grim_rs::Result<()> {
    let mut grim = Grim::new()?;
    let region = Grim::read_region_from_stdin()?;
    let result = grim.capture_region(region)?;
    grim.save_png(result.data(), result.width(), result.height(), "region.png")?;
    Ok(())
}
```

## Pixel format conversion

```rust,no_run
use grim_rs::pixel_format::{self, PixelFormat};

fn process_raw_frame(mut data: Vec<u8>) -> Vec<u8> {
    // Convert from ARGB8888 (DRM_FORMAT_ARGB8888) to RGBA
    pixel_format::convert_to_rgba(&mut data, PixelFormat::Argb8888);
    data
}

fn detect_format(fourcc: u32) -> Option<PixelFormat> {
    pixel_format::fourcc_to_format(fourcc)
}
```