import { memo } from 'react' import { Position, NodeResizer, type NodeProps } from '@xyflow/react' import { BaseNode, BaseNodeHeader, BaseNodeHeaderTitle, BaseNodeContent } from '../ui/base-node' import { BaseHandle } from '../ui/base-handle' import { NodeStatusIndicator, type NodeStatus } from '../ui/node-status-indicator' import { NodeTooltip, NodeTooltipTrigger, NodeTooltipContent } from '../ui/node-tooltip' import { getDeviceRenderConfig } from './deviceRegistry' import type { DeviceProperties } from '@/types' export interface DeviceNodeData { label: string deviceType: string category?: string properties: DeviceProperties [key: string]: unknown } function TooltipRow({ label, value }: { label: string; value: string | null | undefined }) { if (!value) return null return (
{label} {value}
) } const NODE_DEFAULT = 120 // default square side in px const NODE_MIN = 80 // minimum square side in px const NODE_MAX = 280 // maximum square side in px function DeviceNodeComponent({ data, selected, width, height }: NodeProps) { const nodeData = data as unknown as DeviceNodeData const { icon: Icon, color } = getDeviceRenderConfig(nodeData.deviceType, nodeData.category) const status = (nodeData.properties?.status || 'unknown') as NodeStatus const ip = nodeData.properties?.ip const props = nodeData.properties || {} // Use the shorter dimension so content never overflows a non-square node const size = Math.min(width ?? NODE_DEFAULT, height ?? NODE_DEFAULT) const scale = size / NODE_DEFAULT // Icon: 28px at default, clamped to [14, 72] const iconPx = Math.round(Math.max(14, Math.min(72, scale * 28))) // Label font: 11px at default, clamped to [9, 20] const labelPx = Math.max(9, Math.min(20, Math.round(scale * 11))) // IP font: 9px at default, clamped to [8, 16] const ipPx = Math.max(8, Math.min(16, Math.round(scale * 9))) const hasTooltipContent = props.hostname || props.ip || props.vendor || props.model || props.role || props.notes return ( <> {nodeData.label} {ip && ( {ip} )} {hasTooltipContent && (
{(props.vendor || props.model) && ( )} {props.notes && ( 100 ? props.notes.slice(0, 100) + '...' : props.notes} /> )}
)}
) } export const DeviceNode = memo(DeviceNodeComponent)