Adds Z as a keyboard equivalent for the existing double-click zoom-to-team gesture (WCAG 2.1.1). When a team node is selected, pressing Z dispatches molecule:zoom-to-team, which fitBounds to the parent and all children. Input elements are guarded so Z still types normally in text fields. Adds a 6th help panel entry documenting the Dbl-click / Z gesture. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
293 lines
12 KiB
TypeScript
293 lines
12 KiB
TypeScript
"use client";
|
|
|
|
import { useMemo, useState, useCallback, useEffect, useRef } from "react";
|
|
import { api } from "@/lib/api";
|
|
import { useCanvasStore } from "@/store/canvas";
|
|
import { SettingsButton } from "@/components/settings/SettingsButton";
|
|
import { settingsGearRef } from "@/components/settings/SettingsPanel";
|
|
import { ConfirmDialog } from "@/components/ConfirmDialog";
|
|
import { showToast } from "@/components/Toaster";
|
|
|
|
export function Toolbar() {
|
|
const nodes = useCanvasStore((s) => s.nodes);
|
|
const wsStatus = useCanvasStore((s) => s.wsStatus);
|
|
|
|
const [stopping, setStopping] = useState(false);
|
|
const [restartingAll, setRestartingAll] = useState(false);
|
|
const [restartConfirmOpen, setRestartConfirmOpen] = useState(false);
|
|
const [helpOpen, setHelpOpen] = useState(false);
|
|
const helpRef = useRef<HTMLDivElement>(null);
|
|
|
|
// Suppress toast on the very first connect at page load; only fire on reconnects.
|
|
const mountedRef = useRef(false);
|
|
useEffect(() => {
|
|
const t = setTimeout(() => { mountedRef.current = true; }, 2000);
|
|
return () => clearTimeout(t);
|
|
}, []);
|
|
|
|
const prevWsStatus = useRef<string>("connecting");
|
|
useEffect(() => {
|
|
if (prevWsStatus.current === "connecting" && wsStatus === "connected") {
|
|
if (mountedRef.current) {
|
|
showToast("Live updates restored", "success");
|
|
}
|
|
}
|
|
prevWsStatus.current = wsStatus;
|
|
}, [wsStatus]);
|
|
|
|
const counts = useMemo(() => {
|
|
const c = { total: nodes.length, roots: 0, children: 0, online: 0, offline: 0, failed: 0, provisioning: 0, activeTasks: 0 };
|
|
for (const n of nodes) {
|
|
if (n.data.parentId) c.children++; else c.roots++;
|
|
const s = n.data.status;
|
|
if (s === "online") c.online++;
|
|
else if (s === "offline") c.offline++;
|
|
else if (s === "failed") c.failed++;
|
|
else if (s === "provisioning") c.provisioning++;
|
|
if ((n.data.activeTasks as number) > 0) c.activeTasks++;
|
|
}
|
|
return c;
|
|
}, [nodes]);
|
|
|
|
const stopAll = useCallback(async () => {
|
|
setStopping(true);
|
|
const active = nodes.filter((n) => (n.data.activeTasks as number) > 0);
|
|
await Promise.all(
|
|
active.map((n) =>
|
|
api.post(`/workspaces/${n.id}/restart`).catch(() => {})
|
|
)
|
|
);
|
|
setStopping(false);
|
|
}, [nodes]);
|
|
|
|
// Workspaces flagged as needing restart (config edited, global secret changed, etc.)
|
|
const needsRestartNodes = useMemo(
|
|
() => nodes.filter((n) => n.data.needsRestart),
|
|
[nodes]
|
|
);
|
|
|
|
const restartAll = useCallback(async () => {
|
|
setRestartConfirmOpen(false);
|
|
setRestartingAll(true);
|
|
const targets = needsRestartNodes;
|
|
const results = await Promise.allSettled(
|
|
targets.map((n) => api.post(`/workspaces/${n.id}/restart`))
|
|
);
|
|
const failed = results.filter((r) => r.status === "rejected").length;
|
|
setRestartingAll(false);
|
|
// Clear needsRestart on successfully-restarted workspaces
|
|
const store = useCanvasStore.getState();
|
|
targets.forEach((n, i) => {
|
|
if (results[i].status === "fulfilled") {
|
|
store.updateNodeData(n.id, { needsRestart: false });
|
|
}
|
|
});
|
|
if (failed === 0) {
|
|
showToast(`Restarted ${targets.length} workspace${targets.length === 1 ? "" : "s"}`, "success");
|
|
} else if (failed === targets.length) {
|
|
showToast(`Failed to restart any workspaces`, "error");
|
|
} else {
|
|
showToast(`Restarted ${targets.length - failed} of ${targets.length} (${failed} failed)`, "error");
|
|
}
|
|
}, [needsRestartNodes]);
|
|
|
|
useEffect(() => {
|
|
const onPointerDown = (event: MouseEvent) => {
|
|
if (helpRef.current && !helpRef.current.contains(event.target as Node)) {
|
|
setHelpOpen(false);
|
|
}
|
|
};
|
|
const onKeyDown = (event: KeyboardEvent) => {
|
|
if (event.key === "Escape") {
|
|
setHelpOpen(false);
|
|
}
|
|
};
|
|
window.addEventListener("pointerdown", onPointerDown);
|
|
window.addEventListener("keydown", onKeyDown);
|
|
return () => {
|
|
window.removeEventListener("pointerdown", onPointerDown);
|
|
window.removeEventListener("keydown", onKeyDown);
|
|
};
|
|
}, []);
|
|
|
|
return (
|
|
<div className="fixed top-3 left-1/2 -translate-x-1/2 z-20 flex items-center gap-3 bg-zinc-900/80 backdrop-blur-md border border-zinc-800/60 rounded-xl px-4 py-2 shadow-xl shadow-black/20">
|
|
{/* Logo / Title */}
|
|
<div className="flex items-center gap-2 pr-3 border-r border-zinc-800/60">
|
|
<img src="/molecule-icon.png" alt="Molecule AI" className="w-5 h-5" />
|
|
<span className="text-[11px] font-semibold text-zinc-300 tracking-wide">Molecule AI</span>
|
|
</div>
|
|
|
|
{/* Status counts */}
|
|
<div className="flex items-center gap-2.5">
|
|
<StatusPill color="bg-emerald-400" count={counts.online} label="online" />
|
|
{counts.offline > 0 && (
|
|
<StatusPill color="bg-zinc-500" count={counts.offline} label="offline" />
|
|
)}
|
|
{counts.provisioning > 0 && (
|
|
<StatusPill color="bg-sky-400 animate-pulse" count={counts.provisioning} label="starting" />
|
|
)}
|
|
{counts.failed > 0 && (
|
|
<StatusPill color="bg-red-400" count={counts.failed} label="failed" />
|
|
)}
|
|
</div>
|
|
|
|
{/* Total */}
|
|
<div className="pl-3 border-l border-zinc-800/60">
|
|
<span className="text-[10px] text-zinc-500">
|
|
{counts.roots} workspace{counts.roots !== 1 ? "s" : ""}
|
|
{counts.children > 0 && <span className="text-zinc-600"> + {counts.children} sub</span>}
|
|
</span>
|
|
</div>
|
|
|
|
{/* WebSocket connection status */}
|
|
<div className="pl-3 border-l border-zinc-800/60">
|
|
<WsStatusPill status={wsStatus} />
|
|
</div>
|
|
|
|
{/* Stop All — visible when agents have active tasks */}
|
|
{counts.activeTasks > 0 && (
|
|
<button
|
|
onClick={stopAll}
|
|
disabled={stopping}
|
|
className="flex items-center gap-1.5 px-2.5 py-1 bg-red-950/50 hover:bg-red-900/60 border border-red-800/40 rounded-lg transition-colors disabled:opacity-50"
|
|
title={`Stop all running tasks (${counts.activeTasks} active)`}
|
|
>
|
|
<svg width="10" height="10" viewBox="0 0 16 16" fill="currentColor" className="text-red-400">
|
|
<rect x="2" y="2" width="12" height="12" rx="2" />
|
|
</svg>
|
|
<span className="text-[10px] text-red-300 font-medium">
|
|
{stopping ? "Stopping..." : `Stop All (${counts.activeTasks})`}
|
|
</span>
|
|
</button>
|
|
)}
|
|
|
|
{/* Restart All — only shows when workspaces are flagged as needsRestart */}
|
|
{needsRestartNodes.length > 0 && (
|
|
<button
|
|
onClick={() => setRestartConfirmOpen(true)}
|
|
disabled={restartingAll}
|
|
className="flex items-center gap-1.5 px-2.5 py-1 bg-amber-950/40 hover:bg-amber-900/50 border border-amber-800/40 rounded-lg transition-colors disabled:opacity-50"
|
|
title={`Restart ${needsRestartNodes.length} workspace${needsRestartNodes.length === 1 ? "" : "s"} that need to pick up config or secret changes`}
|
|
>
|
|
<svg width="10" height="10" viewBox="0 0 16 16" fill="none" stroke="currentColor" strokeWidth="1.8" className="text-amber-400">
|
|
<path d="M2 8a6 6 0 1 1 1.76 4.24M2 13v-3h3" strokeLinecap="round" strokeLinejoin="round" />
|
|
</svg>
|
|
<span className="text-[10px] text-amber-300 font-medium">
|
|
{restartingAll ? "Restarting..." : `Restart Pending (${needsRestartNodes.length})`}
|
|
</span>
|
|
</button>
|
|
)}
|
|
|
|
{/* Search shortcut */}
|
|
<button
|
|
onClick={() => useCanvasStore.getState().setSearchOpen(true)}
|
|
className="flex items-center gap-1.5 px-2.5 py-1 bg-zinc-800/50 hover:bg-zinc-700/50 border border-zinc-700/40 rounded-lg transition-colors"
|
|
>
|
|
<svg width="12" height="12" viewBox="0 0 16 16" fill="none" className="text-zinc-500">
|
|
<circle cx="7" cy="7" r="5" stroke="currentColor" strokeWidth="1.5" />
|
|
<path d="M11 11l3 3" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round" />
|
|
</svg>
|
|
<span className="text-[10px] text-zinc-500">Search</span>
|
|
<kbd className="text-[8px] text-zinc-600 bg-zinc-900/60 px-1 py-0.5 rounded border border-zinc-700/30">⌘K</kbd>
|
|
</button>
|
|
|
|
{/* Quick help */}
|
|
<div ref={helpRef} className="relative">
|
|
<button
|
|
onClick={() => setHelpOpen((open) => !open)}
|
|
className="flex items-center gap-1.5 px-2.5 py-1 bg-zinc-800/50 hover:bg-zinc-700/50 border border-zinc-700/40 rounded-lg transition-colors"
|
|
aria-expanded={helpOpen}
|
|
aria-label="Open quick help"
|
|
>
|
|
<svg width="12" height="12" viewBox="0 0 16 16" fill="none" className="text-zinc-500">
|
|
<path d="M8 12v.5M6.5 6.3A1.9 1.9 0 1 1 9 8.1c-.7.4-1 .8-1 1.7" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round" />
|
|
<circle cx="8" cy="8" r="6" stroke="currentColor" strokeWidth="1.2" />
|
|
</svg>
|
|
<span className="text-[10px] text-zinc-500">Help</span>
|
|
</button>
|
|
|
|
{helpOpen && (
|
|
<div className="absolute right-0 top-full mt-2 w-72 rounded-xl border border-zinc-700/60 bg-zinc-950/95 p-3 shadow-2xl shadow-black/50 backdrop-blur-md">
|
|
<div className="mb-2 flex items-center justify-between">
|
|
<span className="text-[10px] font-semibold uppercase tracking-[0.24em] text-zinc-400">Quick start</span>
|
|
<button
|
|
onClick={() => setHelpOpen(false)}
|
|
className="text-[10px] text-zinc-600 hover:text-zinc-300 transition-colors"
|
|
>
|
|
Close
|
|
</button>
|
|
</div>
|
|
<div className="space-y-2">
|
|
<HelpRow shortcut="⌘K" text="Search workspaces and jump straight into Details or Chat." />
|
|
<HelpRow shortcut="Palette" text="Open the template palette to deploy a new workspace." />
|
|
<HelpRow shortcut="Right-click" text="Use node actions for expand, duplicate, export, restart, or delete." />
|
|
<HelpRow shortcut="Chat" text="If a task is still running, the chat tab resumes that session automatically." />
|
|
<HelpRow shortcut="Config" text="Use the Config tab for skills, model, secrets, and runtime settings." />
|
|
<HelpRow shortcut="Dbl-click / Z" text="Zoom canvas to fit a team node and all its sub-workspaces." />
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Settings gear icon */}
|
|
<SettingsButton ref={settingsGearRef} />
|
|
|
|
<ConfirmDialog
|
|
open={restartConfirmOpen}
|
|
title={`Restart ${needsRestartNodes.length} workspace${needsRestartNodes.length === 1 ? "" : "s"}?`}
|
|
message="These workspaces have pending config or secret changes that need a restart to take effect."
|
|
confirmLabel="Restart"
|
|
confirmVariant="warning"
|
|
onConfirm={restartAll}
|
|
onCancel={() => setRestartConfirmOpen(false)}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function StatusPill({ color, count, label }: { color: string; count: number; label: string }) {
|
|
return (
|
|
<div className="flex items-center gap-1.5" title={`${count} ${label}`}>
|
|
<div className={`w-1.5 h-1.5 rounded-full ${color}`} />
|
|
<span className="text-[10px] text-zinc-400 tabular-nums">{count}</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function WsStatusPill({ status }: { status: "connected" | "connecting" | "disconnected" }) {
|
|
if (status === "connected") {
|
|
return (
|
|
<div className="flex items-center gap-1.5" title="Real-time updates: connected">
|
|
<div className="w-1.5 h-1.5 rounded-full bg-emerald-400" />
|
|
<span className="text-[10px] text-zinc-500">Live</span>
|
|
</div>
|
|
);
|
|
}
|
|
if (status === "connecting") {
|
|
return (
|
|
<div className="flex items-center gap-1.5" title="Real-time updates: reconnecting…">
|
|
<div className="w-1.5 h-1.5 rounded-full bg-amber-400 animate-pulse" />
|
|
<span className="text-[10px] text-zinc-500">Reconnecting</span>
|
|
</div>
|
|
);
|
|
}
|
|
return (
|
|
<div className="flex items-center gap-1.5" title="Real-time updates: disconnected">
|
|
<div className="w-1.5 h-1.5 rounded-full bg-red-400" />
|
|
<span className="text-[10px] text-zinc-500">Offline</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function HelpRow({ shortcut, text }: { shortcut: string; text: string }) {
|
|
return (
|
|
<div className="flex items-start gap-3 rounded-lg border border-zinc-800/70 bg-zinc-900/45 px-3 py-2">
|
|
<span className="shrink-0 rounded-md border border-zinc-700/60 bg-zinc-950/70 px-2 py-0.5 text-[9px] font-medium uppercase tracking-[0.18em] text-zinc-400">
|
|
{shortcut}
|
|
</span>
|
|
<p className="text-[11px] leading-relaxed text-zinc-500">{text}</p>
|
|
</div>
|
|
);
|
|
}
|