jellyflow-runtime 0.1.0

Headless store, rules, schema, profile, and change pipeline for Jellyflow.
Documentation
use jellyflow_core::core::{CanvasPoint, CanvasRect};

use super::geometry::{compute_fit_view_target_top_left, compute_target_for_canvas_rect};
use super::projection::project_nodes_to_top_left;
use super::{FitViewComputeOptions, FitViewNodeInfo};

const FIT_VIEW_SOLVE_ITERATIONS: usize = 4;

/// Computes the viewport pan/zoom that frames the given nodes in view.
///
/// The returned pan/zoom matches the UI contract: `pan` is in canvas space (added in the render
/// transform) and `zoom` is a scalar.
pub fn compute_fit_view_target(
    nodes: &[FitViewNodeInfo],
    options: FitViewComputeOptions,
) -> Option<(CanvasPoint, f32)> {
    let options = options.normalized()?;
    if nodes.is_empty() {
        return None;
    }

    let mut zoom_guess = options.max_zoom;
    let mut best: Option<(CanvasPoint, f32)> = None;

    for _ in 0..FIT_VIEW_SOLVE_ITERATIONS {
        if !zoom_guess.is_finite() || zoom_guess <= 0.0 {
            zoom_guess = 1.0;
        }
        zoom_guess = zoom_guess.clamp(options.min_zoom, options.max_zoom);

        let scratch = project_nodes_to_top_left(nodes, options.node_origin, zoom_guess);

        let Some((pan, zoom_next)) = compute_fit_view_target_top_left(&scratch, options) else {
            return best;
        };
        best = Some((pan, zoom_next));
        zoom_guess = zoom_next;
    }

    best
}

/// Computes the viewport pan/zoom that frames the given canvas-space rect in view.
pub fn compute_fit_view_target_for_canvas_rect(
    target_canvas: CanvasRect,
    options: FitViewComputeOptions,
) -> Option<(CanvasPoint, f32)> {
    compute_target_for_canvas_rect(target_canvas, options.normalized()?)
}