import { Background, Controls, Panel, ReactFlow, ReactFlowProvider, MarkerType, } from '@xyflow/react'; import '@xyflow/react/dist/style.css'; import {useEffect, useState} from "react"; import {useShallow} from 'zustand/react/shallow'; import useProgramStore from "../../utils/programStore.ts"; import {DndToolbar} from './visualProgrammingUI/components/DragDropSidebar.tsx'; import useFlowStore from './visualProgrammingUI/VisProgStores.tsx'; import type {FlowState} from './visualProgrammingUI/VisProgTypes.tsx'; import styles from './VisProg.module.css' import { NodeReduces, NodeTypes } from './visualProgrammingUI/NodeRegistry.ts'; import SaveLoadPanel from './visualProgrammingUI/components/SaveLoadPanel.tsx'; import MonitoringPage from '../MonitoringPage/MonitoringPage.tsx'; // --| config starting params for flow |-- /** * defines how the default edge looks inside the editor */ const DEFAULT_EDGE_OPTIONS = { type: 'default', markerEnd: { type: MarkerType.ArrowClosed, color: '#505050', }, }; /** * defines what functions in the FlowState store map to which names, * @param state */ const selector = (state: FlowState) => ({ nodes: state.nodes, edges: state.edges, onNodesChange: state.onNodesChange, onEdgesDelete: state.onEdgesDelete, onEdgesChange: state.onEdgesChange, onConnect: state.onConnect, onReconnectStart: state.onReconnectStart, onReconnectEnd: state.onReconnectEnd, onReconnect: state.onReconnect, undo: state.undo, redo: state.redo, beginBatchAction: state.beginBatchAction, endBatchAction: state.endBatchAction }); // --| define ReactFlow editor |-- /** * Defines the ReactFlow visual programming editor component * any implementations of editor logic should be encapsulated where possible * so the Component definition stays as readable as possible * @constructor */ const VisProgUI = () => { const { nodes, edges, onNodesChange, onEdgesDelete, onEdgesChange, onConnect, onReconnect, onReconnectStart, onReconnectEnd, undo, redo, beginBatchAction, endBatchAction } = useFlowStore(useShallow(selector)); // instructs the editor to use the corresponding functions from the FlowStore // adds ctrl+z and ctrl+y support to respectively undo and redo actions useEffect(() => { const handler = (e: KeyboardEvent) => { if (e.ctrlKey && e.key === 'z') undo(); if (e.ctrlKey && e.key === 'y') redo(); }; window.addEventListener('keydown', handler); return () => window.removeEventListener('keydown', handler); }); return (
{/* contains the drag and drop panel for nodes */}
); }; /** * Places the VisProgUI component inside a ReactFlowProvider * * Wrapping the editor component inside a ReactFlowProvider * allows us to access and interact with the components inside the editor, outside the editor definition, * thus facilitating the addition of node specific functions inside their node definitions */ function VisualProgrammingUI() { return ( ); } // currently outputs the prepared program to the console function runProgramm() { const phases = graphReducer(); const program = {phases} console.log(JSON.stringify(program, null, 2)); fetch( "http://localhost:8000/program", { method: "POST", headers: {"Content-Type": "application/json"}, body: JSON.stringify(program), } ).then((res) => { if (!res.ok) throw new Error("Failed communicating with the backend.") console.log("Successfully sent the program to the backend."); // store reduced program in global program store for further use in the UI // when the program was sent to the backend successfully: useProgramStore.getState().setProgramState(structuredClone(program)); }).catch(() => console.log("Failed to send program to the backend.")); } /** * Reduces the graph into its phases' information and recursively calls their reducing function */ function graphReducer() { const { nodes } = useFlowStore.getState(); return nodes .filter((n) => n.type == 'phase') .map((n) => { const reducer = NodeReduces['phase']; return reducer(n, nodes) }); } /** * houses the entire page, so also UI elements * that are not a part of the Visual Programming UI * @constructor */ function VisProgPage() { const [showSimpleProgram, setShowSimpleProgram] = useState(false); const setProgramState = useProgramStore((state) => state.setProgramState); const runProgram = () => { const phases = graphReducer(); // reduce graph setProgramState({ phases }); // <-- save to store setShowSimpleProgram(true); // show SimpleProgram runProgramm(); // send to backend if needed }; if (showSimpleProgram) { return (
); } return ( <> ) } export default VisProgPage