mirror of
https://github.com/jkingsman/Remote-Terminal-for-MeshCore.git
synced 2026-07-02 07:51:09 +02:00
108 lines
3.2 KiB
TypeScript
108 lines
3.2 KiB
TypeScript
import { useState, type ReactNode } from 'react';
|
|
import { toast } from './ui/sonner';
|
|
import { formatTime } from '../utils/messageParser';
|
|
import {
|
|
isValidLocation,
|
|
calculateDistance,
|
|
formatDistance,
|
|
formatRouteLabel,
|
|
getEffectiveContactRoute,
|
|
} from '../utils/pathUtils';
|
|
import { getMapFocusHash } from '../utils/urlHash';
|
|
import { handleKeyboardActivate } from '../utils/a11y';
|
|
import type { Contact } from '../types';
|
|
import { useDistanceUnit } from '../contexts/DistanceUnitContext';
|
|
import { ContactRoutingOverrideModal } from './ContactRoutingOverrideModal';
|
|
|
|
interface ContactStatusInfoProps {
|
|
contact: Contact;
|
|
ourLat: number | null;
|
|
ourLon: number | null;
|
|
}
|
|
|
|
/**
|
|
* Renders the "(Last heard: ..., N hops, lat, lon (dist))" status line
|
|
* shared between ChatHeader and RepeaterDashboard.
|
|
*/
|
|
export function ContactStatusInfo({ contact, ourLat, ourLon }: ContactStatusInfoProps) {
|
|
const { distanceUnit } = useDistanceUnit();
|
|
const [routingModalOpen, setRoutingModalOpen] = useState(false);
|
|
const parts: ReactNode[] = [];
|
|
const effectiveRoute = getEffectiveContactRoute(contact);
|
|
|
|
if (contact.last_seen) {
|
|
parts.push(`Last heard: ${formatTime(contact.last_seen)}`);
|
|
}
|
|
|
|
parts.push(
|
|
<span
|
|
key="path"
|
|
className="cursor-pointer underline underline-offset-2 decoration-muted-foreground/50 hover:text-primary"
|
|
role="button"
|
|
tabIndex={0}
|
|
onKeyDown={handleKeyboardActivate}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
setRoutingModalOpen(true);
|
|
}}
|
|
title="Click to edit routing override"
|
|
>
|
|
{formatRouteLabel(effectiveRoute.pathLen)}
|
|
{effectiveRoute.forced && <span className="text-destructive"> (forced)</span>}
|
|
</span>
|
|
);
|
|
|
|
if (isValidLocation(contact.lat, contact.lon)) {
|
|
const distFromUs =
|
|
ourLat != null && ourLon != null && isValidLocation(ourLat, ourLon)
|
|
? calculateDistance(ourLat, ourLon, contact.lat, contact.lon)
|
|
: null;
|
|
parts.push(
|
|
<span key="coords">
|
|
<span
|
|
className="font-mono cursor-pointer hover:text-primary hover:underline"
|
|
role="button"
|
|
tabIndex={0}
|
|
onKeyDown={handleKeyboardActivate}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
const url =
|
|
window.location.origin +
|
|
window.location.pathname +
|
|
getMapFocusHash(contact.public_key);
|
|
window.open(url, '_blank');
|
|
}}
|
|
title="View on map"
|
|
>
|
|
{contact.lat!.toFixed(3)}, {contact.lon!.toFixed(3)}
|
|
</span>
|
|
{distFromUs !== null && ` (${formatDistance(distFromUs, distanceUnit)})`}
|
|
</span>
|
|
);
|
|
}
|
|
|
|
if (parts.length === 0) return null;
|
|
|
|
return (
|
|
<>
|
|
<span className="font-normal text-sm text-muted-foreground flex-shrink-0">
|
|
(
|
|
{parts.map((part, i) => (
|
|
<span key={i}>
|
|
{i > 0 && ', '}
|
|
{part}
|
|
</span>
|
|
))}
|
|
)
|
|
</span>
|
|
<ContactRoutingOverrideModal
|
|
open={routingModalOpen}
|
|
onClose={() => setRoutingModalOpen(false)}
|
|
contact={contact}
|
|
onSaved={(message) => toast.success(message)}
|
|
onError={(message) => toast.error(message)}
|
|
/>
|
|
</>
|
|
);
|
|
}
|