lynxkite / web /src /LynxKiteFlow.svelte
darabos's picture
Use Pydantic for op metadata.
942065e
raw
history blame
5.95 kB
<script lang="ts">
import { setContext } from 'svelte';
import { writable, derived } from 'svelte/store';
import {
SvelteFlow,
Controls,
Background,
MiniMap,
MarkerType,
useSvelteFlow,
type XYPosition,
type Node,
type Edge,
type Connection,
type NodeTypes,
} from '@xyflow/svelte';
import { useQuery, useMutation, useQueryClient } from '@sveltestack/svelte-query';
import NodeWithParams from './NodeWithParams.svelte';
import NodeWithParamsVertical from './NodeWithParamsVertical.svelte';
import NodeWithVisualization from './NodeWithVisualization.svelte';
import NodeWithTableView from './NodeWithTableView.svelte';
import NodeWithSubFlow from './NodeWithSubFlow.svelte';
import NodeWithArea from './NodeWithArea.svelte';
import NodeSearch from './NodeSearch.svelte';
import '@xyflow/svelte/dist/style.css';
export let path = '';
const { screenToFlowPosition } = useSvelteFlow();
const queryClient = useQueryClient();
const backendWorkspace = useQuery(['workspace', path], async () => {
const res = await fetch(`/api/load?path=${path}`);
return res.json();
}, {staleTime: 10000, retry: false});
const mutation = useMutation(async(update) => {
const res = await fetch('/api/save', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify(update),
});
return await res.json();
}, {
onSuccess: data => queryClient.setQueryData(['workspace', path], data),
})
const nodeTypes: NodeTypes = {
basic: NodeWithParams,
vertical: NodeWithParamsVertical,
visualization: NodeWithVisualization,
table_view: NodeWithTableView,
sub_flow: NodeWithSubFlow,
area: NodeWithArea,
};
const nodes = writable<Node[]>([]);
const edges = writable<Edge[]>([]);
let doNotSave = true;
$: if ($backendWorkspace.isSuccess) {
doNotSave = true; // Change is coming from the backend.
nodes.set(JSON.parse(JSON.stringify($backendWorkspace.data?.nodes || [])));
edges.set(JSON.parse(JSON.stringify($backendWorkspace.data?.edges || [])));
doNotSave = false;
}
function closeNodeSearch() {
nodeSearchSettings = undefined;
}
function toggleNodeSearch({ detail: { event } }) {
if (nodeSearchSettings) {
closeNodeSearch();
return;
}
event.preventDefault();
nodeSearchSettings = {
pos: { x: event.clientX, y: event.clientY },
boxes: $boxes,
};
}
function addNode(e) {
const meta = {...e.detail};
nodes.update((n) => {
const node = {
type: meta.type,
data: {
title: meta.name,
params: Object.fromEntries(
Object.values(meta.params).map((p) => [p.name, p.default])),
inputs: meta.inputs,
outputs: meta.outputs,
},
};
node.position = screenToFlowPosition({x: nodeSearchSettings.pos.x, y: nodeSearchSettings.pos.y});
const title = node.data.title;
let i = 1;
node.id = `${title} ${i}`;
while (n.find((x) => x.id === node.id)) {
i += 1;
node.id = `${title} ${i}`;
}
node.parentId = nodeSearchSettings.parentId;
if (node.parentId) {
node.extent = 'parent';
const parent = n.find((x) => x.id === node.parentId);
node.position = { x: node.position.x - parent.position.x, y: node.position.y - parent.position.y };
}
return [...n, node]
});
closeNodeSearch();
}
const boxes = writable([]);
async function getBoxes() {
const res = await fetch('/api/catalog');
const j = await res.json();
boxes.set(j);
}
getBoxes();
let nodeSearchSettings: {
pos: XYPosition,
boxes: any[],
parentId: string,
};
const graph = derived([nodes, edges], ([nodes, edges]) => ({ nodes, edges }));
// Like JSON.stringify, but with keys sorted.
function orderedJSON(obj: any) {
const allKeys = new Set();
JSON.stringify(obj, (key, value) => (allKeys.add(key), value));
return JSON.stringify(obj, Array.from(allKeys).sort());
}
graph.subscribe(async (g) => {
if (doNotSave) return;
const dragging = g.nodes.find((n) => n.dragging);
if (dragging) return;
g = JSON.parse(JSON.stringify(g));
for (const node of g.nodes) {
delete node.computed;
delete node.selected;
}
const ws = orderedJSON(g);
const bd = orderedJSON($backendWorkspace.data);
if (ws === bd) return;
$mutation.mutate({ path, ws: g });
});
function onconnect(connection: Connection) {
edges.update((edges) => {
// Only one source can connect to a given target.
return edges.filter((e) =>
e.source === connection.source
|| e.target !== connection.target
|| e.targetHandle !== connection.targetHandle);
});
}
function getMeta(title) {
return $boxes.find((m) => m.data.title === title);
}
setContext('LynxKiteFlow', { getMeta });
function nodeClick(e) {
const node = e.detail.node;
const meta = getMeta(node.data.title);
if (!meta) return;
const sub_nodes = meta.sub_nodes;
if (!sub_nodes) return;
const event = e.detail.event;
if (event.target.classList.contains('title')) return;
nodeSearchSettings = {
pos: { x: event.clientX, y: event.clientY },
boxes: sub_nodes,
parentId: node.id,
};
}
</script>
<div style:height="100%">
<SvelteFlow {nodes} {edges} {nodeTypes} fitView
on:paneclick={toggleNodeSearch}
on:nodeclick={nodeClick}
proOptions={{ hideAttribution: true }}
maxZoom={3}
minZoom={0.3}
onconnect={onconnect}
defaultEdgeOptions={{ markerEnd: { type: MarkerType.Arrow } }}
>
<Controls />
<MiniMap />
{#if nodeSearchSettings}
<NodeSearch pos={nodeSearchSettings.pos} boxes={nodeSearchSettings.boxes} on:cancel={closeNodeSearch} on:add={addNode} />
{/if}
</SvelteFlow>
</div>