Spaces:
Running
Running
<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> | |