jellyflow-runtime 0.2.0

Headless store, rules, schema, profile, and change pipeline for Jellyflow.
Documentation
use crate::io::{NodeGraphInteractionState, NodeGraphViewState};
use crate::node_origin::resolve_node_origin;
use crate::runtime::policy::resolve_node_interaction_policy;
use jellyflow_core::core::{CanvasPoint, CanvasRect, CanvasSize, Graph, GroupId, Node, NodeId};

use super::constraints::{normalized_size, resolved_extent_rect};

#[derive(Debug, Clone, Copy)]
pub(super) struct DragCandidate {
    pub(super) node: NodeId,
    pub(super) from: CanvasPoint,
    pub(super) size: CanvasSize,
    pub(super) origin: (f32, f32),
    pub(super) parent: Option<GroupId>,
    pub(super) expand_parent: bool,
    pub(super) extent: Option<CanvasRect>,
    pub(super) node_extent_override: bool,
}

pub(super) fn drag_candidates(
    graph: &Graph,
    view_state: &NodeGraphViewState,
    interaction: &NodeGraphInteractionState,
    primary: NodeId,
) -> Vec<DragCandidate> {
    let mut nodes = view_state.selected_nodes.clone();
    nodes.push(primary);
    nodes.sort();
    nodes.dedup();

    nodes
        .into_iter()
        .filter_map(|node| {
            let graph_node = graph.nodes.get(&node)?;
            if !node_is_draggable(graph_node, view_state, interaction) {
                return None;
            }
            let policy = resolve_node_interaction_policy(graph_node, interaction);
            let node_drag = interaction.node_drag_interaction();
            let fallback_origin = node_drag.node_origin.normalized();
            Some(DragCandidate {
                node,
                from: graph_node.pos,
                size: normalized_size(graph_node.size),
                origin: resolve_node_origin(
                    graph_node.origin,
                    (fallback_origin.x, fallback_origin.y),
                ),
                parent: graph_node.parent,
                expand_parent: policy.expand_parent,
                extent: resolved_extent_rect(
                    graph,
                    graph_node,
                    policy.extent,
                    policy.expand_parent,
                ),
                node_extent_override: graph_node.extent.is_some(),
            })
        })
        .collect()
}

fn node_is_draggable(
    node: &Node,
    view_state: &NodeGraphViewState,
    interaction: &NodeGraphInteractionState,
) -> bool {
    if node.hidden || !node.pos.is_finite() {
        return false;
    }
    if node
        .parent
        .is_some_and(|parent| view_state.selected_groups.contains(&parent))
    {
        return false;
    }
    resolve_node_interaction_policy(node, interaction).draggable
}