 85bf1341f3
			
		
	
	85bf1341f3
	
	
	
		
			
			Frontend Enhancements: - Complete React TypeScript frontend with modern UI components - Distributed workflows management interface with real-time updates - Socket.IO integration for live agent status monitoring - Agent management dashboard with cluster visualization - Project management interface with metrics and task tracking - Responsive design with proper error handling and loading states Backend Infrastructure: - Distributed coordinator for multi-agent workflow orchestration - Cluster management API with comprehensive agent operations - Enhanced database models for agents and projects - Project service for filesystem-based project discovery - Performance monitoring and metrics collection - Comprehensive API documentation and error handling Documentation: - Complete distributed development guide (README_DISTRIBUTED.md) - Comprehensive development report with architecture insights - System configuration templates and deployment guides The platform now provides a complete web interface for managing the distributed AI cluster with real-time monitoring, workflow orchestration, and agent coordination capabilities. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
		
			
				
	
	
		
			144 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import React, { memo, useRef, useEffect } from 'react';
 | |
| import cc from 'classcat';
 | |
| import { shallow } from 'zustand/shallow';
 | |
| import { zoom, zoomIdentity } from 'd3-zoom';
 | |
| import { select, pointer } from 'd3-selection';
 | |
| import { useStore, getNodePositionWithOrigin, useStoreApi, Panel, getBoundsOfRects, getNodesBounds } from '@reactflow/core';
 | |
| 
 | |
| const MiniMapNode = ({ id, x, y, width, height, style, color, strokeColor, strokeWidth, className, borderRadius, shapeRendering, onClick, selected, }) => {
 | |
|     const { background, backgroundColor } = style || {};
 | |
|     const fill = (color || background || backgroundColor);
 | |
|     return (React.createElement("rect", { className: cc(['react-flow__minimap-node', { selected }, className]), x: x, y: y, rx: borderRadius, ry: borderRadius, width: width, height: height, fill: fill, stroke: strokeColor, strokeWidth: strokeWidth, shapeRendering: shapeRendering, onClick: onClick ? (event) => onClick(event, id) : undefined }));
 | |
| };
 | |
| MiniMapNode.displayName = 'MiniMapNode';
 | |
| var MiniMapNode$1 = memo(MiniMapNode);
 | |
| 
 | |
| /* eslint-disable @typescript-eslint/ban-ts-comment */
 | |
| const selector$1 = (s) => s.nodeOrigin;
 | |
| const selectorNodes = (s) => s.getNodes().filter((node) => !node.hidden && node.width && node.height);
 | |
| const getAttrFunction = (func) => (func instanceof Function ? func : () => func);
 | |
| function MiniMapNodes({ nodeStrokeColor = 'transparent', nodeColor = '#e2e2e2', nodeClassName = '', nodeBorderRadius = 5, nodeStrokeWidth = 2, 
 | |
| // We need to rename the prop to be `CapitalCase` so that JSX will render it as
 | |
| // a component properly.
 | |
| nodeComponent: NodeComponent = MiniMapNode$1, onClick, }) {
 | |
|     const nodes = useStore(selectorNodes, shallow);
 | |
|     const nodeOrigin = useStore(selector$1);
 | |
|     const nodeColorFunc = getAttrFunction(nodeColor);
 | |
|     const nodeStrokeColorFunc = getAttrFunction(nodeStrokeColor);
 | |
|     const nodeClassNameFunc = getAttrFunction(nodeClassName);
 | |
|     const shapeRendering = typeof window === 'undefined' || !!window.chrome ? 'crispEdges' : 'geometricPrecision';
 | |
|     return (React.createElement(React.Fragment, null, nodes.map((node) => {
 | |
|         const { x, y } = getNodePositionWithOrigin(node, nodeOrigin).positionAbsolute;
 | |
|         return (React.createElement(NodeComponent, { key: node.id, x: x, y: y, width: node.width, height: node.height, style: node.style, selected: node.selected, className: nodeClassNameFunc(node), color: nodeColorFunc(node), borderRadius: nodeBorderRadius, strokeColor: nodeStrokeColorFunc(node), strokeWidth: nodeStrokeWidth, shapeRendering: shapeRendering, onClick: onClick, id: node.id }));
 | |
|     })));
 | |
| }
 | |
| var MiniMapNodes$1 = memo(MiniMapNodes);
 | |
| 
 | |
| /* eslint-disable @typescript-eslint/ban-ts-comment */
 | |
| const defaultWidth = 200;
 | |
| const defaultHeight = 150;
 | |
| const selector = (s) => {
 | |
|     const nodes = s.getNodes();
 | |
|     const viewBB = {
 | |
|         x: -s.transform[0] / s.transform[2],
 | |
|         y: -s.transform[1] / s.transform[2],
 | |
|         width: s.width / s.transform[2],
 | |
|         height: s.height / s.transform[2],
 | |
|     };
 | |
|     return {
 | |
|         viewBB,
 | |
|         boundingRect: nodes.length > 0 ? getBoundsOfRects(getNodesBounds(nodes, s.nodeOrigin), viewBB) : viewBB,
 | |
|         rfId: s.rfId,
 | |
|     };
 | |
| };
 | |
| const ARIA_LABEL_KEY = 'react-flow__minimap-desc';
 | |
| function MiniMap({ style, className, nodeStrokeColor = 'transparent', nodeColor = '#e2e2e2', nodeClassName = '', nodeBorderRadius = 5, nodeStrokeWidth = 2, 
 | |
| // We need to rename the prop to be `CapitalCase` so that JSX will render it as
 | |
| // a component properly.
 | |
| nodeComponent, maskColor = 'rgb(240, 240, 240, 0.6)', maskStrokeColor = 'none', maskStrokeWidth = 1, position = 'bottom-right', onClick, onNodeClick, pannable = false, zoomable = false, ariaLabel = 'React Flow mini map', inversePan = false, zoomStep = 10, offsetScale = 5, }) {
 | |
|     const store = useStoreApi();
 | |
|     const svg = useRef(null);
 | |
|     const { boundingRect, viewBB, rfId } = useStore(selector, shallow);
 | |
|     const elementWidth = style?.width ?? defaultWidth;
 | |
|     const elementHeight = style?.height ?? defaultHeight;
 | |
|     const scaledWidth = boundingRect.width / elementWidth;
 | |
|     const scaledHeight = boundingRect.height / elementHeight;
 | |
|     const viewScale = Math.max(scaledWidth, scaledHeight);
 | |
|     const viewWidth = viewScale * elementWidth;
 | |
|     const viewHeight = viewScale * elementHeight;
 | |
|     const offset = offsetScale * viewScale;
 | |
|     const x = boundingRect.x - (viewWidth - boundingRect.width) / 2 - offset;
 | |
|     const y = boundingRect.y - (viewHeight - boundingRect.height) / 2 - offset;
 | |
|     const width = viewWidth + offset * 2;
 | |
|     const height = viewHeight + offset * 2;
 | |
|     const labelledBy = `${ARIA_LABEL_KEY}-${rfId}`;
 | |
|     const viewScaleRef = useRef(0);
 | |
|     viewScaleRef.current = viewScale;
 | |
|     useEffect(() => {
 | |
|         if (svg.current) {
 | |
|             const selection = select(svg.current);
 | |
|             const zoomHandler = (event) => {
 | |
|                 const { transform, d3Selection, d3Zoom } = store.getState();
 | |
|                 if (event.sourceEvent.type !== 'wheel' || !d3Selection || !d3Zoom) {
 | |
|                     return;
 | |
|                 }
 | |
|                 const pinchDelta = -event.sourceEvent.deltaY *
 | |
|                     (event.sourceEvent.deltaMode === 1 ? 0.05 : event.sourceEvent.deltaMode ? 1 : 0.002) *
 | |
|                     zoomStep;
 | |
|                 const zoom = transform[2] * Math.pow(2, pinchDelta);
 | |
|                 d3Zoom.scaleTo(d3Selection, zoom);
 | |
|             };
 | |
|             const panHandler = (event) => {
 | |
|                 const { transform, d3Selection, d3Zoom, translateExtent, width, height } = store.getState();
 | |
|                 if (event.sourceEvent.type !== 'mousemove' || !d3Selection || !d3Zoom) {
 | |
|                     return;
 | |
|                 }
 | |
|                 // @TODO: how to calculate the correct next position? Math.max(1, transform[2]) is a workaround.
 | |
|                 const moveScale = viewScaleRef.current * Math.max(1, transform[2]) * (inversePan ? -1 : 1);
 | |
|                 const position = {
 | |
|                     x: transform[0] - event.sourceEvent.movementX * moveScale,
 | |
|                     y: transform[1] - event.sourceEvent.movementY * moveScale,
 | |
|                 };
 | |
|                 const extent = [
 | |
|                     [0, 0],
 | |
|                     [width, height],
 | |
|                 ];
 | |
|                 const nextTransform = zoomIdentity.translate(position.x, position.y).scale(transform[2]);
 | |
|                 const constrainedTransform = d3Zoom.constrain()(nextTransform, extent, translateExtent);
 | |
|                 d3Zoom.transform(d3Selection, constrainedTransform);
 | |
|             };
 | |
|             const zoomAndPanHandler = zoom()
 | |
|                 // @ts-ignore
 | |
|                 .on('zoom', pannable ? panHandler : null)
 | |
|                 // @ts-ignore
 | |
|                 .on('zoom.wheel', zoomable ? zoomHandler : null);
 | |
|             selection.call(zoomAndPanHandler);
 | |
|             return () => {
 | |
|                 selection.on('zoom', null);
 | |
|             };
 | |
|         }
 | |
|     }, [pannable, zoomable, inversePan, zoomStep]);
 | |
|     const onSvgClick = onClick
 | |
|         ? (event) => {
 | |
|             const rfCoord = pointer(event);
 | |
|             onClick(event, { x: rfCoord[0], y: rfCoord[1] });
 | |
|         }
 | |
|         : undefined;
 | |
|     const onSvgNodeClick = onNodeClick
 | |
|         ? (event, nodeId) => {
 | |
|             const node = store.getState().nodeInternals.get(nodeId);
 | |
|             onNodeClick(event, node);
 | |
|         }
 | |
|         : undefined;
 | |
|     return (React.createElement(Panel, { position: position, style: style, className: cc(['react-flow__minimap', className]), "data-testid": "rf__minimap" },
 | |
|         React.createElement("svg", { width: elementWidth, height: elementHeight, viewBox: `${x} ${y} ${width} ${height}`, role: "img", "aria-labelledby": labelledBy, ref: svg, onClick: onSvgClick },
 | |
|             ariaLabel && React.createElement("title", { id: labelledBy }, ariaLabel),
 | |
|             React.createElement(MiniMapNodes$1, { onClick: onSvgNodeClick, nodeColor: nodeColor, nodeStrokeColor: nodeStrokeColor, nodeBorderRadius: nodeBorderRadius, nodeClassName: nodeClassName, nodeStrokeWidth: nodeStrokeWidth, nodeComponent: nodeComponent }),
 | |
|             React.createElement("path", { className: "react-flow__minimap-mask", d: `M${x - offset},${y - offset}h${width + offset * 2}v${height + offset * 2}h${-width - offset * 2}z
 | |
|         M${viewBB.x},${viewBB.y}h${viewBB.width}v${viewBB.height}h${-viewBB.width}z`, fill: maskColor, fillRule: "evenodd", stroke: maskStrokeColor, strokeWidth: maskStrokeWidth, pointerEvents: "none" }))));
 | |
| }
 | |
| MiniMap.displayName = 'MiniMap';
 | |
| var MiniMap$1 = memo(MiniMap);
 | |
| 
 | |
| export { MiniMap$1 as MiniMap };
 |