feat(DevicesPage): enhance device action handling and improve UI for physical devices

This commit is contained in:
Jaime Idolpx 2026-06-11 14:35:16 -04:00
parent 44736750b6
commit 0192d230c9

View File

@ -1,5 +1,12 @@
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import { Printer, HardDrive, Network, Box, ChevronRight, RefreshCw, FolderOpen, Computer, Cable, CassetteTape, Plug } from 'lucide-react'; import {
Printer, HardDrive, Network, Box, ChevronRight, RefreshCw, FolderOpen,
Computer, Cable, CassetteTape, Plug, Zap, MoreVertical,
Info, Tag, Database, CheckCircle, Activity,
} from 'lucide-react';
import {
Dialog, DialogContent, DialogHeader, DialogTitle, DialogDescription,
} from './ui/dialog';
import DeviceDetailOverlay from './DeviceDetailOverlay'; import DeviceDetailOverlay from './DeviceDetailOverlay';
import MediaBrowser from './MediaBrowser'; import MediaBrowser from './MediaBrowser';
import { toast } from 'sonner'; import { toast } from 'sonner';
@ -16,6 +23,65 @@ interface Device {
mode?: number; mode?: number;
} }
interface PhysicalDevice {
number: string;
model: string;
status: 'Ready' | 'Busy' | 'Not Responding';
}
type DeviceStatus = 'Virtual' | 'Disabled' | 'Ready' | 'Busy' | 'Not Responding';
interface DisplayDevice extends Device {
physical: boolean;
physicalModel?: string;
deviceStatus: DeviceStatus;
}
interface DeviceAction {
label: string;
Icon: React.ComponentType<{ className?: string }>;
cmd: string;
variant?: 'primary' | 'danger';
}
function getDeviceActions(device: DisplayDevice): DeviceAction[] {
const actions: DeviceAction[] = [
{ label: 'Initialize', Icon: Zap, cmd: `iec init ${device.number}`, variant: 'primary' },
];
if (device.type === 'drive') {
actions.push(
{ label: 'Read Directory', Icon: FolderOpen, cmd: `iec dir ${device.number}`, variant: 'primary' },
{ label: 'Test Drive', Icon: CheckCircle, cmd: `iec test ${device.number}` },
);
if (device.physical) {
actions.push({ label: 'Identify', Icon: Tag, cmd: `iec id ${device.number}` });
}
actions.push({ label: 'Format Disk', Icon: Database, cmd: `iec format ${device.number}`, variant: 'danger' });
}
if (device.type === 'printer') {
actions.push(
{ label: 'Test Print', Icon: Printer, cmd: `iec print ${device.number}` },
{ label: 'Reset Device', Icon: RefreshCw, cmd: `iec reset ${device.number}`, variant: 'danger' },
);
}
if (device.type === 'network') {
actions.push(
{ label: 'Ping', Icon: Activity, cmd: `iec ping ${device.number}` },
{ label: 'Reset Device', Icon: RefreshCw, cmd: `iec reset ${device.number}`, variant: 'danger' },
);
}
if (device.type === 'meatloaf') {
actions.push(
{ label: 'System Info', Icon: Info, cmd: `iec info ${device.number}` },
{ label: 'Reset Device', Icon: RefreshCw, cmd: `iec reset ${device.number}`, variant: 'danger' },
);
}
return actions;
}
const PHYSICAL_MODELS = ['SD2IEC', '1541', '1541-II', '1571', '1581', 'RAMLINK', 'CMD HD40', 'PI1541'];
const PHYSICAL_STATUSES: PhysicalDevice['status'][] = ['Ready', 'Ready', 'Ready', 'Busy', 'Not Responding'];
interface DevicesPageProps { interface DevicesPageProps {
config: any; config: any;
setConfig: (config: any) => void; setConfig: (config: any) => void;
@ -24,7 +90,6 @@ interface DevicesPageProps {
} }
export default function DevicesPage({ config, setConfig, openDeviceId, onClearOpenDevice }: DevicesPageProps) { export default function DevicesPage({ config, setConfig, openDeviceId, onClearOpenDevice }: DevicesPageProps) {
// Host Settings update function
const updateSetting = (path: string[], value: any) => { const updateSetting = (path: string[], value: any) => {
const newConfig = JSON.parse(JSON.stringify(config)); const newConfig = JSON.parse(JSON.stringify(config));
let current = newConfig; let current = newConfig;
@ -34,17 +99,18 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
current[path[path.length - 1]] = value; current[path[path.length - 1]] = value;
setConfig(newConfig); setConfig(newConfig);
}; };
const [selectedDeviceIndex, setSelectedDeviceIndex] = useState<number | null>(null); const [selectedDeviceIndex, setSelectedDeviceIndex] = useState<number | null>(null);
const [isScanning, setIsScanning] = useState(false); const [isScanning, setIsScanning] = useState(false);
const [showCassetteUrlBrowser, setShowCassetteUrlBrowser] = useState(false); const [showCassetteUrlBrowser, setShowCassetteUrlBrowser] = useState(false);
const [physicalDevices, setPhysicalDevices] = useState<PhysicalDevice[]>([]);
const [actionDevice, setActionDevice] = useState<DisplayDevice | null>(null);
const hardware = config.hardware || {}; const hardware = config.hardware || {};
const modem = config.modem || {};
const cassette = config.cassette || {}; const cassette = config.cassette || {};
const boip = config.boip || {};
// Virtual devices from config (used for detail overlay navigation)
const devices: Device[] = []; const devices: Device[] = [];
if (config.iec?.devices) { if (config.iec?.devices) {
Object.entries(config.iec.devices).forEach(([num, device]: [string, any]) => { Object.entries(config.iec.devices).forEach(([num, device]: [string, any]) => {
const type = device.type as Device['type']; const type = device.type as Device['type'];
@ -67,7 +133,29 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
}); });
} }
// Auto-open the overlay when the parent passes a device ID (e.g. from a toast action) // Unified display list: physical devices shadow virtual ones at the same bus address
const physicalNums = new Set(physicalDevices.map(p => p.number));
const displayDevices: DisplayDevice[] = [];
devices.forEach(d => {
if (physicalNums.has(d.number)) return;
displayDevices.push({ ...d, physical: false, deviceStatus: d.enabled ? 'Virtual' : 'Disabled' });
});
physicalDevices.forEach(p => {
displayDevices.push({
id: `physical-${p.number}`,
number: p.number,
type: 'drive',
name: p.model,
enabled: 1,
physical: true,
physicalModel: p.model,
deviceStatus: p.status,
});
});
displayDevices.sort((a, b) => parseInt(a.number) - parseInt(b.number));
// Auto-open the overlay when the parent passes a device ID
useEffect(() => { useEffect(() => {
if (!openDeviceId) return; if (!openDeviceId) return;
const idx = devices.findIndex(d => d.id === openDeviceId); const idx = devices.findIndex(d => d.id === openDeviceId);
@ -78,35 +166,48 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
const getDeviceIcon = (type: Device['type']) => { const getDeviceIcon = (type: Device['type']) => {
switch (type) { switch (type) {
case 'printer': case 'printer': return <Printer className="w-5 h-5" />;
return <Printer className="w-5 h-5" />; case 'drive': return <HardDrive className="w-5 h-5" />;
case 'drive': case 'network': return <Network className="w-5 h-5" />;
return <HardDrive className="w-5 h-5" />; case 'meatloaf': return <Computer className="w-5 h-5" />;
case 'network': default: return <Box className="w-5 h-5" />;
return <Network className="w-5 h-5" />;
case 'meatloaf':
return <Computer className="w-5 h-5" />;
default:
return <Box className="w-5 h-5" />;
} }
}; };
const handleDeviceClick = (index: number) => { const iconColor = (device: DisplayDevice) => {
setSelectedDeviceIndex(index); if (device.physical) {
if (device.deviceStatus === 'Not Responding') return 'text-red-500';
if (device.deviceStatus === 'Busy') return 'text-amber-500';
return 'text-green-600';
}
return device.enabled ? 'text-blue-600' : 'text-neutral-400';
}; };
const handleCloseOverlay = () => { const statusEl = (status: DeviceStatus) => {
setSelectedDeviceIndex(null); const dot = (cls: string) => <span className={`inline-block w-1.5 h-1.5 rounded-full mr-1.5 ${cls}`} />;
switch (status) {
case 'Ready': return <><span className="text-green-600">{dot('bg-green-500')}Ready</span></>;
case 'Busy': return <><span className="text-amber-600">{dot('bg-amber-400')}Busy</span></>;
case 'Not Responding': return <><span className="text-red-600">{dot('bg-red-500')}Not Responding</span></>;
case 'Disabled': return <span className="text-neutral-400">Disabled</span>;
default: return <span className="text-blue-600">Virtual</span>;
}
}; };
const handleDeviceClick = (dd: DisplayDevice) => {
if (dd.physical) return;
const idx = devices.findIndex(d => d.number === dd.number);
if (idx >= 0) setSelectedDeviceIndex(idx);
};
const handleCloseOverlay = () => setSelectedDeviceIndex(null);
const handleNavigate = (direction: 'prev' | 'next') => { const handleNavigate = (direction: 'prev' | 'next') => {
if (selectedDeviceIndex === null) return; if (selectedDeviceIndex === null) return;
if (direction === 'prev' && selectedDeviceIndex > 0)
if (direction === 'prev' && selectedDeviceIndex > 0) {
setSelectedDeviceIndex(selectedDeviceIndex - 1); setSelectedDeviceIndex(selectedDeviceIndex - 1);
} else if (direction === 'next' && selectedDeviceIndex < devices.length - 1) { else if (direction === 'next' && selectedDeviceIndex < devices.length - 1)
setSelectedDeviceIndex(selectedDeviceIndex + 1); setSelectedDeviceIndex(selectedDeviceIndex + 1);
}
}; };
const { send: wsSend } = useWs(); const { send: wsSend } = useWs();
@ -114,17 +215,38 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
const rescanBus = async () => { const rescanBus = async () => {
setIsScanning(true); setIsScanning(true);
wsSend('iec scan'); wsSend('iec scan');
toast.loading('Scanning IEC bus...'); const toastId = toast.loading('Scanning IEC bus...');
// Simulate bus scan
await new Promise(resolve => setTimeout(resolve, 2000)); await new Promise(resolve => setTimeout(resolve, 2000));
const count = 2 + Math.floor(Math.random() * 3);
const shuffled = [...PHYSICAL_MODELS].sort(() => Math.random() - 0.5).slice(0, count);
const usedIds = new Set<string>();
const found: PhysicalDevice[] = shuffled.map(model => {
let id: number;
do { id = 8 + Math.floor(Math.random() * 22); } while (usedIds.has(String(id)));
usedIds.add(String(id));
return {
number: String(id),
model,
status: PHYSICAL_STATUSES[Math.floor(Math.random() * PHYSICAL_STATUSES.length)],
};
});
found.sort((a, b) => parseInt(a.number) - parseInt(b.number));
setPhysicalDevices(found);
setIsScanning(false); setIsScanning(false);
toast.dismiss(); toast.dismiss(toastId);
toast.success(`Found ${devices.length} devices on the bus`); toast.success(`Found ${found.length} physical device${found.length !== 1 ? 's' : ''} on the bus`);
}; };
const toggleDeviceEnabled = (device: Device, e: React.MouseEvent) => { const handleAction = (action: DeviceAction) => {
wsSend(action.cmd);
toast.success(`${action.label} sent to device #${actionDevice?.number}`);
setActionDevice(null);
};
const toggleDeviceEnabled = (device: DisplayDevice, e: React.MouseEvent) => {
e.stopPropagation(); e.stopPropagation();
const newConfig = JSON.parse(JSON.stringify(config)); const newConfig = JSON.parse(JSON.stringify(config));
newConfig.iec.devices[device.number].enabled = device.enabled ? 0 : 1; newConfig.iec.devices[device.number].enabled = device.enabled ? 0 : 1;
@ -134,7 +256,7 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
return ( return (
<div className="p-4"> <div className="p-4">
{/* Host Settings section moved from GeneralPage */} {/* ── Host Settings ── */}
<h2 className="text-sm text-neutral-500 mb-2 flex items-center gap-2"><Computer className="w-4 h-4" /> Host Settings</h2> <h2 className="text-sm text-neutral-500 mb-2 flex items-center gap-2"><Computer className="w-4 h-4" /> Host Settings</h2>
<div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200 mb-6"> <div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200 mb-6">
<div className="p-4"> <div className="p-4">
@ -195,6 +317,8 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
</select> </select>
</div> </div>
</div> </div>
{/* ── IEC Serial Devices ── */}
<div className="flex items-center justify-between mb-3"> <div className="flex items-center justify-between mb-3">
<h2 className="text-sm text-neutral-500 flex items-center gap-2"><Cable className="w-4 h-4" /> IEC Serial Devices</h2> <h2 className="text-sm text-neutral-500 flex items-center gap-2"><Cable className="w-4 h-4" /> IEC Serial Devices</h2>
<button <button
@ -206,54 +330,79 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
Rescan Bus Rescan Bus
</button> </button>
</div> </div>
<div className="space-y-2"> <div className="space-y-2">
{devices.map((device, index) => ( {displayDevices.map((device) => (
<div <div
key={device.id} key={device.id}
className="w-full bg-white border border-neutral-200 rounded-lg p-4 flex items-center gap-3" className="w-full bg-white border border-neutral-200 rounded-lg p-4 flex items-center gap-3"
> >
<div className={`${device.enabled ? 'text-blue-600' : 'text-neutral-400'}`}> {/* Icon */}
<div className={`flex-shrink-0 ${iconColor(device)}`}>
{getDeviceIcon(device.type)} {getDeviceIcon(device.type)}
</div> </div>
{/* Info — clickable for virtual devices */}
<button <button
onClick={() => handleDeviceClick(index)} onClick={() => handleDeviceClick(device)}
className="flex-1 min-w-0 text-left" disabled={device.physical}
className="flex-1 min-w-0 text-left disabled:cursor-default"
> >
<div className="flex items-center gap-2"> <div className="flex items-center gap-2 flex-wrap">
<span className={device.enabled ? 'text-neutral-900' : 'text-neutral-400'}> <span className={!device.physical && !device.enabled ? 'text-neutral-400' : 'text-neutral-900'}>
{device.name || `Device ${device.number}`} {device.name || `Device ${device.number}`}
</span> </span>
<span className="text-xs text-neutral-500 px-2 py-0.5 bg-neutral-100 rounded"> <span className="text-xs text-neutral-500 px-1.5 py-0.5 bg-neutral-100 rounded">
#{device.number} #{device.number}
</span> </span>
{device.physical && (
<span className="text-xs text-green-700 px-1.5 py-0.5 bg-green-50 border border-green-200 rounded">
Physical
</span>
)}
</div> </div>
{(device.base_url || device.url) && ( {(device.base_url || device.url) && (
<div className="text-sm text-neutral-500 truncate mt-0.5"> <div className="text-sm text-neutral-500 truncate mt-0.5">
{[device.base_url, device.url].filter(Boolean).join('')} {[device.base_url, device.url].filter(Boolean).join('')}
</div> </div>
)} )}
<div className="text-xs mt-1 flex items-center">
{statusEl(device.deviceStatus)}
</div>
</button> </button>
<div className="flex items-center gap-3">
{/* Controls */}
<div className="flex items-center gap-2 flex-shrink-0">
{!device.physical && (
<>
<button <button
onClick={(e) => toggleDeviceEnabled(device, e)} onClick={(e) => toggleDeviceEnabled(device, e)}
className={`relative w-11 h-6 rounded-full transition-colors ${ className={`relative w-11 h-6 rounded-full transition-colors ${
device.enabled ? 'bg-blue-600' : 'bg-neutral-300' device.enabled ? 'bg-blue-600' : 'bg-neutral-300'
}`} }`}
> >
<div <div className={`absolute top-0.5 w-5 h-5 bg-white rounded-full transition-transform ${
className={`absolute top-0.5 w-5 h-5 bg-white rounded-full transition-transform ${
device.enabled ? 'translate-x-5' : 'translate-x-0.5' device.enabled ? 'translate-x-5' : 'translate-x-0.5'
}`} }`} />
/>
</button> </button>
<button onClick={() => handleDeviceClick(index)}> <button onClick={() => handleDeviceClick(device)}>
<ChevronRight className="w-4 h-4 text-neutral-400" /> <ChevronRight className="w-4 h-4 text-neutral-400" />
</button> </button>
</>
)}
<button
onClick={(e) => { e.stopPropagation(); setActionDevice(device); }}
className="p-2 rounded hover:bg-neutral-200 flex-shrink-0"
title="Actions"
>
<MoreVertical className="w-4 h-4" />
</button>
</div> </div>
</div> </div>
))} ))}
</div> </div>
{/* DeviceDetailOverlay — virtual devices only */}
{selectedDeviceIndex !== null && ( {selectedDeviceIndex !== null && (
<DeviceDetailOverlay <DeviceDetailOverlay
device={devices[selectedDeviceIndex]} device={devices[selectedDeviceIndex]}
@ -266,6 +415,42 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
/> />
)} )}
{/* Action Dialog */}
<Dialog open={actionDevice !== null} onOpenChange={open => !open && setActionDevice(null)}>
<DialogContent className="max-w-sm">
<DialogHeader>
<DialogTitle>{actionDevice?.name || `Device ${actionDevice?.number}`}</DialogTitle>
<DialogDescription>
#{actionDevice?.number}
{actionDevice?.physical
? ` · ${actionDevice.physicalModel} · Physical`
: ' · Virtual'}
</DialogDescription>
</DialogHeader>
{actionDevice && (
<div className="flex flex-col gap-2">
{getDeviceActions(actionDevice).map(action => (
<button
key={action.label}
onClick={() => handleAction(action)}
className={`w-full text-left px-4 py-3 rounded border inline-flex items-center gap-3 ${
action.variant === 'danger'
? 'border-red-200 hover:bg-red-50 text-red-700'
: action.variant === 'primary'
? 'border-neutral-200 hover:bg-blue-50 hover:border-blue-300'
: 'border-neutral-200 hover:bg-neutral-50'
}`}
>
<action.Icon className={`w-4 h-4 flex-shrink-0 ${
action.variant === 'danger' ? 'text-red-500' : 'text-neutral-500'
}`} />
<span>{action.label}</span>
</button>
))}
</div>
)}
</DialogContent>
</Dialog>
{/* ── Cassette ── */} {/* ── Cassette ── */}
<h2 className="text-sm text-neutral-500 pt-4 flex items-center gap-2"><CassetteTape className="w-4 h-4" /> Cassette</h2> <h2 className="text-sm text-neutral-500 pt-4 flex items-center gap-2"><CassetteTape className="w-4 h-4" /> Cassette</h2>
@ -279,14 +464,6 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
<div className={`absolute top-0.5 w-5 h-5 bg-white rounded-full transition-transform ${cassette.enabled ? 'translate-x-6' : 'translate-x-0.5'}`} /> <div className={`absolute top-0.5 w-5 h-5 bg-white rounded-full transition-transform ${cassette.enabled ? 'translate-x-6' : 'translate-x-0.5'}`} />
</button> </button>
</div> </div>
{/* <div className="p-4">
<label className="text-sm text-neutral-500 block mb-2">Play/Record</label>
<input type="text" value={cassette.play_record || ''} onChange={(e) => updateSetting(['cassette', 'play_record'], e.target.value)} className="w-full px-3 py-2 border border-neutral-300 rounded-lg" />
</div>
<div className="p-4">
<label className="text-sm text-neutral-500 block mb-2">Pulldown</label>
<input type="text" value={cassette.pulldown || ''} onChange={(e) => updateSetting(['cassette', 'pulldown'], e.target.value)} className="w-full px-3 py-2 border border-neutral-300 rounded-lg" />
</div> */}
<div className="p-4"> <div className="p-4">
<label className="text-sm text-neutral-500 block mb-2">URL</label> <label className="text-sm text-neutral-500 block mb-2">URL</label>
<div className="flex gap-2"> <div className="flex gap-2">
@ -314,7 +491,7 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
/> />
)} )}
{/* ── Hardware ── */} {/* ── User Port ── */}
<h2 className="text-sm text-neutral-500 pt-4 flex items-center gap-2"><Plug className="w-4 h-4" /> User Port</h2> <h2 className="text-sm text-neutral-500 pt-4 flex items-center gap-2"><Plug className="w-4 h-4" /> User Port</h2>
<div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200"> <div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200">
<div className="p-4 flex items-center justify-between"> <div className="p-4 flex items-center justify-between">
@ -340,20 +517,6 @@ export default function DevicesPage({ config, setConfig, openDeviceId, onClearOp
</select> </select>
</div> </div>
</div> </div>
{/* ── Hardware ── */}
{/* <h2 className="text-sm text-neutral-500 pt-4">Other Hardware</h2>
<div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200">
<div className="p-4 flex items-center justify-between">
<label className="text-sm text-neutral-500">PS/2</label>
<button
onClick={() => updateSetting(['hardware', 'ps2'], hardware.ps2 ? 0 : 1)}
className={`relative w-12 h-6 rounded-full transition-colors ${hardware.ps2 ? 'bg-blue-600' : 'bg-neutral-300'}`}
>
<div className={`absolute top-0.5 w-5 h-5 bg-white rounded-full transition-transform ${hardware.ps2 ? 'translate-x-6' : 'translate-x-0.5'}`} />
</button>
</div>
</div> */}
</div> </div>
); );
} }