forked from iarv/Remote-Terminal-for-MeshCore
93 lines
3.0 KiB
TypeScript
93 lines
3.0 KiB
TypeScript
import { useState } from 'react';
|
|
import { Menu } from 'lucide-react';
|
|
import type { HealthStatus, RadioConfig } from '../types';
|
|
import { api } from '../api';
|
|
import { toast } from './ui/sonner';
|
|
|
|
interface StatusBarProps {
|
|
health: HealthStatus | null;
|
|
config: RadioConfig | null;
|
|
onSettingsClick: () => void;
|
|
onMenuClick?: () => void;
|
|
}
|
|
|
|
export function StatusBar({ health, config, onSettingsClick, onMenuClick }: StatusBarProps) {
|
|
const connected = health?.radio_connected ?? false;
|
|
const [reconnecting, setReconnecting] = useState(false);
|
|
|
|
const handleReconnect = async () => {
|
|
setReconnecting(true);
|
|
try {
|
|
const result = await api.reconnectRadio();
|
|
if (result.connected) {
|
|
toast.success('Reconnected', { description: result.message });
|
|
}
|
|
} catch (err) {
|
|
toast.error('Reconnection failed', {
|
|
description: err instanceof Error ? err.message : 'Check radio connection and power',
|
|
});
|
|
} finally {
|
|
setReconnecting(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="flex items-center gap-4 px-4 py-2 bg-[#252525] border-b border-[#333] text-xs">
|
|
{/* Mobile menu button - only visible on small screens */}
|
|
{onMenuClick && (
|
|
<button
|
|
onClick={onMenuClick}
|
|
className="md:hidden p-1 bg-transparent border-none text-[#e0e0e0] cursor-pointer"
|
|
aria-label="Open menu"
|
|
>
|
|
<Menu className="h-5 w-5" />
|
|
</button>
|
|
)}
|
|
|
|
<h1 className="text-base font-semibold mr-auto">RemoteTerm</h1>
|
|
|
|
<div className="flex items-center gap-1 text-[#888]">
|
|
<div className={`w-2 h-2 rounded-full ${connected ? 'bg-[#4caf50]' : 'bg-[#666]'}`} />
|
|
<span className="hidden lg:inline text-[#e0e0e0]">
|
|
{connected ? 'Connected' : 'Disconnected'}
|
|
</span>
|
|
</div>
|
|
|
|
{config && (
|
|
<div className="hidden lg:flex items-center gap-2 text-[#888]">
|
|
<span className="text-[#e0e0e0]">{config.name || 'Unnamed'}</span>
|
|
<span
|
|
className="font-mono text-[#888] cursor-pointer hover:text-[#4a9eff]"
|
|
onClick={() => {
|
|
navigator.clipboard.writeText(config.public_key);
|
|
toast.success('Public key copied!');
|
|
}}
|
|
title="Click to copy public key"
|
|
>
|
|
{config.public_key.toLowerCase()}
|
|
</span>
|
|
</div>
|
|
)}
|
|
|
|
{!connected && (
|
|
<button
|
|
onClick={handleReconnect}
|
|
disabled={reconnecting}
|
|
className="px-3 py-1 bg-[#4a3000] border border-[#6b4500] text-[#ffa500] rounded text-xs cursor-pointer hover:bg-[#5a3a00] disabled:opacity-50 disabled:cursor-not-allowed"
|
|
>
|
|
{reconnecting ? 'Reconnecting...' : 'Reconnect'}
|
|
</button>
|
|
)}
|
|
<button
|
|
onClick={onSettingsClick}
|
|
className="px-3 py-1 bg-[#333] border border-[#444] text-[#e0e0e0] rounded text-xs cursor-pointer hover:bg-[#444]"
|
|
>
|
|
<span role="img" aria-label="Settings">
|
|
🔧
|
|
</span>{' '}
|
|
Radio & Config
|
|
</button>
|
|
</div>
|
|
);
|
|
}
|