Files
mapmaker/public/src/components/map/Toolbar.tsx
2025-12-13 01:53:24 +05:00

186 lines
5.7 KiB
TypeScript

import { useDrawingStore } from '../../stores/drawingStore';
import type { DrawingTool } from '../../types/mapItem';
import { CABLE_COLORS, CABLE_LABELS } from '../../types/mapItem';
interface ToolbarProps {
mapId: string;
readOnly?: boolean;
}
interface ToolButton {
id: DrawingTool;
label: string;
icon: string;
color?: string;
description: string;
}
const SELECT_TOOL: ToolButton = {
id: 'select',
label: 'Select',
icon: '👆',
description: 'Select and edit items',
};
const CONNECTION_TOOLS: ToolButton[] = [
{
id: 'wireless_mesh',
label: 'Wireless',
icon: 'wireless',
color: '#10B981',
description: 'Wireless Mesh Link',
},
{
id: 'fiber',
label: 'Fiber',
icon: '━',
color: CABLE_COLORS.fiber,
description: CABLE_LABELS.fiber,
},
{
id: 'cat6',
label: 'Cat6',
icon: '━',
color: CABLE_COLORS.cat6,
description: CABLE_LABELS.cat6,
},
{
id: 'cat6_poe',
label: 'Cat6 PoE',
icon: '━',
color: CABLE_COLORS.cat6_poe,
description: CABLE_LABELS.cat6_poe,
},
];
const DEVICE_TOOLS: ToolButton[] = [
{
id: 'switch',
label: 'Switch',
icon: 'switch',
description: 'Network Switch',
},
{
id: 'indoor_ap',
label: 'Indoor AP',
icon: 'indoor_ap',
description: 'Indoor Access Point',
},
{
id: 'outdoor_ap',
label: 'Outdoor AP',
icon: 'outdoor_ap',
description: 'Outdoor Access Point',
},
];
export function Toolbar({ mapId, readOnly = false }: ToolbarProps) {
const { activeTool, setActiveTool } = useDrawingStore();
const renderIcon = (tool: ToolButton, isDisabled: boolean) => {
if (tool.icon === 'wireless') {
return (
<svg width="20" height="20" viewBox="0 0 20 20" style={{ color: isDisabled ? undefined : tool.color }}>
<line x1="2" y1="10" x2="18" y2="10" stroke="currentColor" strokeWidth="2" strokeDasharray="3,2" />
</svg>
);
}
if (tool.icon === 'switch') {
return (
<svg width="20" height="20" viewBox="0 0 24 24" fill="currentColor" style={{ color: '#8B5CF6' }}>
<rect x="2" y="4" width="20" height="16" rx="2" fill="none" stroke="currentColor" strokeWidth="2"/>
<circle cx="6" cy="10" r="1.5"/>
<circle cx="10" cy="10" r="1.5"/>
<circle cx="14" cy="10" r="1.5"/>
<circle cx="18" cy="10" r="1.5"/>
<circle cx="6" cy="14" r="1.5"/>
<circle cx="10" cy="14" r="1.5"/>
<circle cx="14" cy="14" r="1.5"/>
<circle cx="18" cy="14" r="1.5"/>
</svg>
);
}
if (tool.icon === 'indoor_ap') {
return (
<svg width="20" height="20" viewBox="0 0 24 24" fill="currentColor" style={{ color: '#10B981' }}>
<path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z" opacity="0.3"/>
<circle cx="12" cy="12" r="3"/>
<path d="M12 6c-3.31 0-6 2.69-6 6s2.69 6 6 6 6-2.69 6-6-2.69-6-6-6zm0 10c-2.21 0-4-1.79-4-4s1.79-4 4-4 4 1.79 4 4-1.79 4-4 4z"/>
</svg>
);
}
if (tool.icon === 'outdoor_ap') {
return (
<svg width="20" height="20" viewBox="0 0 24 24" fill="currentColor" style={{ color: '#F59E0B' }}>
<path d="M1 9l2 2c4.97-4.97 13.03-4.97 18 0l2-2C16.93 2.93 7.08 2.93 1 9z"/>
<path d="M9 17l3 3 3-3c-1.65-1.66-4.34-1.66-6 0z"/>
<path d="M5 13l2 2c2.76-2.76 7.24-2.76 10 0l2-2C15.14 9.14 8.87 9.14 5 13z"/>
</svg>
);
}
return (
<span
className="text-lg"
style={tool.color && !isDisabled ? { color: tool.color } : undefined}
>
{tool.icon}
</span>
);
};
const renderToolButton = (tool: ToolButton) => {
const isDisabled = readOnly && tool.id !== 'select';
return (
<button
key={tool.id}
onClick={() => !isDisabled && setActiveTool(tool.id)}
disabled={isDisabled}
className={`w-full px-3 py-2 rounded text-left flex items-center gap-2 transition-colors ${
isDisabled
? 'opacity-50 cursor-not-allowed text-gray-400 dark:text-gray-600'
: activeTool === tool.id
? 'bg-blue-100 dark:bg-blue-900/30 text-blue-700 dark:text-blue-400 font-medium shadow-sm'
: 'hover:bg-gray-100 dark:hover:bg-gray-700 text-gray-700 dark:text-gray-300'
}`}
title={isDisabled ? 'Not available in read-only mode' : tool.description}
>
{renderIcon(tool, isDisabled)}
<span className="text-sm">{tool.label}</span>
</button>
);
};
return (
<div className="bg-white dark:bg-gray-800 border-t border-gray-200 dark:border-gray-700 p-3 space-y-1 transition-colors" style={{ minWidth: '150px' }}>
{/* Read-only indicator */}
{readOnly && (
<div className="w-full px-3 py-2 rounded bg-yellow-100 dark:bg-yellow-900/30 text-yellow-800 dark:text-yellow-400 text-xs font-medium mb-2 text-center border border-yellow-200 dark:border-yellow-800">
Read-Only Mode
</div>
)}
{/* Tools section header */}
<h2 className="text-lg font-semibold mb-3 text-gray-900 dark:text-white">Tools</h2>
{/* Select tool */}
{renderToolButton(SELECT_TOOL)}
<div className="border-t border-gray-200 dark:border-gray-700 my-2"></div>
{/* Connections section */}
<h3 className="text-sm font-semibold text-gray-700 dark:text-gray-300 px-1 mb-2">Connections</h3>
{CONNECTION_TOOLS.map(renderToolButton)}
<div className="border-t border-gray-200 dark:border-gray-700 my-2"></div>
{/* Devices section */}
<h3 className="text-sm font-semibold text-gray-700 dark:text-gray-300 px-1 mb-2">Devices</h3>
{DEVICE_TOOLS.map(renderToolButton)}
</div>
);
}