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>
99 lines
4.0 KiB
JavaScript
99 lines
4.0 KiB
JavaScript
import React, { useCallback } from 'react';
|
|
import { useStore, useNodeId, getNodesBounds, internalsSymbol, Position } from '@reactflow/core';
|
|
import cc from 'classcat';
|
|
import { shallow } from 'zustand/shallow';
|
|
import { createPortal } from 'react-dom';
|
|
|
|
const selector = (state) => state.domNode?.querySelector('.react-flow__renderer');
|
|
function NodeToolbarPortal({ children }) {
|
|
const wrapperRef = useStore(selector);
|
|
if (!wrapperRef) {
|
|
return null;
|
|
}
|
|
return createPortal(children, wrapperRef);
|
|
}
|
|
|
|
const nodeEqualityFn = (a, b) => a?.positionAbsolute?.x === b?.positionAbsolute?.x &&
|
|
a?.positionAbsolute?.y === b?.positionAbsolute?.y &&
|
|
a?.width === b?.width &&
|
|
a?.height === b?.height &&
|
|
a?.selected === b?.selected &&
|
|
a?.[internalsSymbol]?.z === b?.[internalsSymbol]?.z;
|
|
const nodesEqualityFn = (a, b) => {
|
|
return a.length === b.length && a.every((node, i) => nodeEqualityFn(node, b[i]));
|
|
};
|
|
const storeSelector = (state) => ({
|
|
transform: state.transform,
|
|
nodeOrigin: state.nodeOrigin,
|
|
selectedNodesCount: state.getNodes().filter((node) => node.selected).length,
|
|
});
|
|
function getTransform(nodeRect, transform, position, offset, align) {
|
|
let alignmentOffset = 0.5;
|
|
if (align === 'start') {
|
|
alignmentOffset = 0;
|
|
}
|
|
else if (align === 'end') {
|
|
alignmentOffset = 1;
|
|
}
|
|
// position === Position.Top
|
|
// we set the x any y position of the toolbar based on the nodes position
|
|
let pos = [
|
|
(nodeRect.x + nodeRect.width * alignmentOffset) * transform[2] + transform[0],
|
|
nodeRect.y * transform[2] + transform[1] - offset,
|
|
];
|
|
// and than shift it based on the alignment. The shift values are in %.
|
|
let shift = [-100 * alignmentOffset, -100];
|
|
switch (position) {
|
|
case Position.Right:
|
|
pos = [
|
|
(nodeRect.x + nodeRect.width) * transform[2] + transform[0] + offset,
|
|
(nodeRect.y + nodeRect.height * alignmentOffset) * transform[2] + transform[1],
|
|
];
|
|
shift = [0, -100 * alignmentOffset];
|
|
break;
|
|
case Position.Bottom:
|
|
pos[1] = (nodeRect.y + nodeRect.height) * transform[2] + transform[1] + offset;
|
|
shift[1] = 0;
|
|
break;
|
|
case Position.Left:
|
|
pos = [
|
|
nodeRect.x * transform[2] + transform[0] - offset,
|
|
(nodeRect.y + nodeRect.height * alignmentOffset) * transform[2] + transform[1],
|
|
];
|
|
shift = [-100, -100 * alignmentOffset];
|
|
break;
|
|
}
|
|
return `translate(${pos[0]}px, ${pos[1]}px) translate(${shift[0]}%, ${shift[1]}%)`;
|
|
}
|
|
function NodeToolbar({ nodeId, children, className, style, isVisible, position = Position.Top, offset = 10, align = 'center', ...rest }) {
|
|
const contextNodeId = useNodeId();
|
|
const nodesSelector = useCallback((state) => {
|
|
const nodeIds = Array.isArray(nodeId) ? nodeId : [nodeId || contextNodeId || ''];
|
|
return nodeIds.reduce((acc, id) => {
|
|
const node = state.nodeInternals.get(id);
|
|
if (node) {
|
|
acc.push(node);
|
|
}
|
|
return acc;
|
|
}, []);
|
|
}, [nodeId, contextNodeId]);
|
|
const nodes = useStore(nodesSelector, nodesEqualityFn);
|
|
const { transform, nodeOrigin, selectedNodesCount } = useStore(storeSelector, shallow);
|
|
const isActive = typeof isVisible === 'boolean' ? isVisible : nodes.length === 1 && nodes[0].selected && selectedNodesCount === 1;
|
|
if (!isActive || !nodes.length) {
|
|
return null;
|
|
}
|
|
const nodeRect = getNodesBounds(nodes, nodeOrigin);
|
|
const zIndex = Math.max(...nodes.map((node) => (node[internalsSymbol]?.z || 1) + 1));
|
|
const wrapperStyle = {
|
|
position: 'absolute',
|
|
transform: getTransform(nodeRect, transform, position, offset, align),
|
|
zIndex,
|
|
...style,
|
|
};
|
|
return (React.createElement(NodeToolbarPortal, null,
|
|
React.createElement("div", { style: wrapperStyle, className: cc(['react-flow__node-toolbar', className]), ...rest }, children)));
|
|
}
|
|
|
|
export { NodeToolbar };
|