feat(app): scrollback search bar (⌘F) on the focused panel
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
+16
-2
@@ -3,6 +3,7 @@ import { LayoutEngine } from "./LayoutEngine";
|
||||
import { Sidebar } from "./Sidebar";
|
||||
import { TopBar } from "./TopBar";
|
||||
import { CenterToolbar } from "./CenterToolbar";
|
||||
import { SearchBar } from "./SearchBar";
|
||||
import { Wizard } from "./Wizard";
|
||||
import { EventCenter } from "./EventCenter";
|
||||
import { maybeNotify } from "./notify";
|
||||
@@ -24,6 +25,8 @@ export function App() {
|
||||
const [health, setHealth] = useState<DaemonHealth | null>(null);
|
||||
const [connected, setConnected] = useState(false);
|
||||
const [focusedId, setFocusedId] = useState<string | null>(null);
|
||||
const [searchOpen, setSearchOpen] = useState(false);
|
||||
const [searchNonce, setSearchNonce] = useState(0);
|
||||
const activeRef = useRef<string | null>(null);
|
||||
const wsRef = useRef<WorkspaceView[]>([]);
|
||||
activeRef.current = activeId;
|
||||
@@ -91,6 +94,16 @@ export function App() {
|
||||
return () => { void unlisten.then((f) => f()); void reconnect.then((f) => f()); };
|
||||
}, [refresh, seedEvents, loadHealth]);
|
||||
|
||||
useEffect(() => {
|
||||
const onKey = (e: KeyboardEvent) => {
|
||||
if ((e.metaKey || e.ctrlKey) && e.key.toLowerCase() === "f") {
|
||||
if (activeRef.current) { e.preventDefault(); setSearchOpen(true); setSearchNonce((n) => n + 1); }
|
||||
}
|
||||
};
|
||||
window.addEventListener("keydown", onKey);
|
||||
return () => window.removeEventListener("keydown", onKey);
|
||||
}, []);
|
||||
|
||||
const unread = useMemo(() => events.filter((e) => !e.read).length, [events]);
|
||||
const active = workspaces.find((w) => w.id === activeId) ?? null;
|
||||
const leaves = active ? leafIds(active.layout) : [];
|
||||
@@ -109,12 +122,13 @@ export function App() {
|
||||
<Sidebar groups={groups} workspaces={workspaces} activeId={activeId} onSelect={selectWorkspace} onNew={() => setWizard(true)} health={health} connected={connected} />
|
||||
<div style={{ flex: 1, display: "flex", flexDirection: "column", minWidth: 0 }}>
|
||||
{active && (
|
||||
<CenterToolbar selected="" onSelect={(p) => { if (active) void applyPreset(active.id, p, []); }} />
|
||||
<CenterToolbar selected="" onSelect={(p) => { if (active) void applyPreset(active.id, p, []); }} onOpenSearch={() => setSearchOpen(true)} />
|
||||
)}
|
||||
<div style={{ flex: 1, minHeight: 0 }}>
|
||||
<div style={{ flex: 1, minHeight: 0, position: "relative" }}>
|
||||
{active
|
||||
? <LayoutEngine workspaceId={active.id} layout={active.layout} running={running} states={states} surfaces={active.surfaces} focusedId={effectiveFocus} onFocus={setFocusedId} zoomed={active.zoomed} />
|
||||
: <div style={{ color: COLORS.textMuted, padding: 24 }}>No workspace — create one to begin.</div>}
|
||||
{searchOpen && active && <SearchBar surfaceId={effectiveFocus} reopenNonce={searchNonce} onClose={() => setSearchOpen(false)} />}
|
||||
</div>
|
||||
</div>
|
||||
{eventsOpen && (
|
||||
|
||||
@@ -3,13 +3,14 @@ import { COLORS, FONT } from "./theme";
|
||||
import { PresetPicker } from "./PresetPicker";
|
||||
|
||||
/** Top-of-grid toolbar: layout presets on the left, scrollback search on the right (search is a mock). */
|
||||
export function CenterToolbar({ selected, onSelect }: { selected: string; onSelect: (id: string) => void }) {
|
||||
export function CenterToolbar({ selected, onSelect, onOpenSearch }: { selected: string; onSelect: (id: string) => void; onOpenSearch: () => void }) {
|
||||
return (
|
||||
<div style={{ display: "flex", alignItems: "center", gap: 8, padding: "0 12px", height: 46, borderBottom: `1px solid ${COLORS.borderSubtle}` }}>
|
||||
<PresetPicker selected={selected} onSelect={onSelect} />
|
||||
<div style={{ flex: 1 }} />
|
||||
<div
|
||||
title="Search scrollback (mock)"
|
||||
title="Search scrollback"
|
||||
onClick={onOpenSearch}
|
||||
style={{
|
||||
display: "flex", alignItems: "center", gap: 6, height: 24, padding: "0 8px", borderRadius: 6,
|
||||
background: COLORS.bgPanel, border: `1px solid ${COLORS.borderSubtle}`, cursor: "pointer",
|
||||
|
||||
@@ -0,0 +1,133 @@
|
||||
import { useEffect, useRef, useState } from "react";
|
||||
import { ChevronUp, ChevronDown, X } from "lucide-react";
|
||||
import { COLORS, FONT } from "./theme";
|
||||
import { getSearch } from "./searchRegistry";
|
||||
import type { ISearchOptions } from "@xterm/addon-search";
|
||||
|
||||
const SEARCH_OPTS: ISearchOptions = {
|
||||
decorations: {
|
||||
matchBackground: COLORS.searchMatch,
|
||||
matchOverviewRuler: COLORS.stWait,
|
||||
activeMatchBackground: COLORS.stWait,
|
||||
activeMatchColorOverviewRuler: COLORS.stWait,
|
||||
},
|
||||
};
|
||||
|
||||
export function SearchBar({
|
||||
surfaceId,
|
||||
reopenNonce,
|
||||
onClose,
|
||||
}: {
|
||||
surfaceId: string | null;
|
||||
reopenNonce: number;
|
||||
onClose: () => void;
|
||||
}) {
|
||||
const [term, setTerm] = useState("");
|
||||
const [count, setCount] = useState({ index: -1, total: 0 });
|
||||
const inputRef = useRef<HTMLInputElement>(null);
|
||||
|
||||
useEffect(() => {
|
||||
inputRef.current?.focus();
|
||||
inputRef.current?.select();
|
||||
}, [reopenNonce]);
|
||||
|
||||
useEffect(() => {
|
||||
inputRef.current?.focus();
|
||||
if (!surfaceId) return;
|
||||
const addon = getSearch(surfaceId);
|
||||
if (!addon) return;
|
||||
const sub = addon.onDidChangeResults((r) =>
|
||||
setCount({ index: r.resultIndex, total: r.resultCount })
|
||||
);
|
||||
return () => {
|
||||
sub.dispose();
|
||||
addon.clearDecorations();
|
||||
};
|
||||
}, [surfaceId]);
|
||||
|
||||
function run(forward: boolean) {
|
||||
if (!surfaceId) return;
|
||||
const addon = getSearch(surfaceId);
|
||||
if (!addon || !term) {
|
||||
addon?.clearDecorations();
|
||||
setCount({ index: -1, total: 0 });
|
||||
return;
|
||||
}
|
||||
if (forward) addon.findNext(term, SEARCH_OPTS);
|
||||
else addon.findPrevious(term, SEARCH_OPTS);
|
||||
}
|
||||
|
||||
return (
|
||||
<div
|
||||
style={{
|
||||
position: "absolute",
|
||||
top: 8,
|
||||
right: 16,
|
||||
zIndex: 50,
|
||||
display: "flex",
|
||||
alignItems: "center",
|
||||
gap: 6,
|
||||
height: 32,
|
||||
padding: "0 8px",
|
||||
background: COLORS.bgElevated,
|
||||
border: `1px solid ${COLORS.borderStrong}`,
|
||||
borderRadius: 8,
|
||||
boxShadow: "0 4px 16px rgba(0,0,0,0.4)",
|
||||
}}
|
||||
>
|
||||
<input
|
||||
ref={inputRef}
|
||||
value={term}
|
||||
onChange={(e) => setTerm(e.target.value)}
|
||||
onKeyDown={(e) => {
|
||||
if (e.key === "Enter") {
|
||||
e.preventDefault();
|
||||
run(!e.shiftKey);
|
||||
} else if (e.key === "Escape") {
|
||||
e.preventDefault();
|
||||
onClose();
|
||||
}
|
||||
}}
|
||||
placeholder="Search scrollback"
|
||||
style={{
|
||||
width: 200,
|
||||
background: "transparent",
|
||||
border: "none",
|
||||
outline: "none",
|
||||
color: COLORS.textPrimary,
|
||||
fontFamily: FONT.ui,
|
||||
fontSize: 13,
|
||||
}}
|
||||
/>
|
||||
<span
|
||||
style={{
|
||||
fontFamily: FONT.mono,
|
||||
fontSize: 11,
|
||||
color: COLORS.textMuted,
|
||||
minWidth: 40,
|
||||
textAlign: "right",
|
||||
}}
|
||||
>
|
||||
{count.total > 0 ? `${count.index + 1}/${count.total}` : "0/0"}
|
||||
</span>
|
||||
<ChevronUp
|
||||
size={15}
|
||||
color={COLORS.textSecondary}
|
||||
style={{ cursor: "pointer" }}
|
||||
onClick={() => run(false)}
|
||||
/>
|
||||
<ChevronDown
|
||||
size={15}
|
||||
color={COLORS.textSecondary}
|
||||
style={{ cursor: "pointer" }}
|
||||
onClick={() => run(true)}
|
||||
/>
|
||||
<X
|
||||
size={15}
|
||||
color={COLORS.textMuted}
|
||||
style={{ cursor: "pointer" }}
|
||||
onClick={onClose}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -18,6 +18,7 @@ export const COLORS = {
|
||||
stDone: "#3FB950",
|
||||
stError: "#F4544E",
|
||||
stIdle: "#5A6573",
|
||||
searchMatch: "#5A4A1F",
|
||||
} as const;
|
||||
|
||||
export const FONT = {
|
||||
|
||||
Reference in New Issue
Block a user