import * as React from 'react'; import { useEffect, useMemo, useState } from 'react'; import { useRegisterOverlay } from '../context/overlayContext.js'; import { getTimestampedHistory, type TimestampedHistoryEntry } from '../history.js'; import { useTerminalSize } from '../hooks/useTerminalSize.js'; import { stringWidth } from '../ink/stringWidth.js'; import { wrapAnsi } from '../ink/wrapAnsi.js'; import { Box, Text } from '../ink.js'; import { logEvent } from '../services/analytics/index.js'; import type { HistoryEntry } from '../utils/config.js'; import { formatRelativeTimeAgo, truncateToWidth } from '../utils/format.js'; import { FuzzyPicker } from './design-system/FuzzyPicker.js'; type Props = { initialQuery?: string; onSelect: (entry: HistoryEntry) => void; onCancel: () => void; }; const PREVIEW_ROWS = 6; const AGE_WIDTH = 8; type Item = { entry: TimestampedHistoryEntry; display: string; lower: string; firstLine: string; age: string; }; export function HistorySearchDialog({ initialQuery, onSelect, onCancel }: Props): React.ReactNode { useRegisterOverlay('history-search'); const { columns } = useTerminalSize(); const [items, setItems] = useState(null); const [query, setQuery] = useState(initialQuery ?? ''); useEffect(() => { let cancelled = false; void (async () => { const reader = getTimestampedHistory(); const loaded: Item[] = []; for await (const entry of reader) { if (cancelled) { void reader.return(undefined); return; } const display = entry.display; const nl = display.indexOf('\n'); const age = formatRelativeTimeAgo(new Date(entry.timestamp)); loaded.push({ entry, display, lower: display.toLowerCase(), firstLine: nl === -1 ? display : display.slice(0, nl), age: age + ' '.repeat(Math.max(0, AGE_WIDTH - stringWidth(age))) }); } if (!cancelled) setItems(loaded); })(); return () => { cancelled = true; }; }, []); const filtered = useMemo(() => { if (!items) return []; const q = query.trim().toLowerCase(); if (!q) return items; const exact: Item[] = []; const fuzzy: Item[] = []; for (const item of items) { if (item.lower.includes(q)) { exact.push(item); } else if (isSubsequence(item.lower, q)) { fuzzy.push(item); } } return exact.concat(fuzzy); }, [items, query]); const previewOnRight = columns >= 100; const listWidth = previewOnRight ? Math.floor((columns - 6) * 0.5) : columns - 6; const rowWidth = Math.max(20, listWidth - AGE_WIDTH - 1); const previewWidth = previewOnRight ? Math.max(20, columns - listWidth - 12) : Math.max(20, columns - 10); return String(item_0.entry.timestamp)} onQueryChange={setQuery} onSelect={item_1 => { logEvent('tengu_history_picker_select', { result_count: filtered.length, query_length: query.length }); void item_1.entry.resolve().then(onSelect); }} onCancel={onCancel} emptyMessage={q_0 => items === null ? 'Loading…' : q_0 ? 'No matching prompts' : 'No history yet'} selectAction="use" direction="up" previewPosition={previewOnRight ? 'right' : 'bottom'} renderItem={(item_2, isFocused) => {item_2.age} {' '} {truncateToWidth(item_2.firstLine, rowWidth)} } renderPreview={item_3 => { const wrapped = wrapAnsi(item_3.display, previewWidth, { hard: true }).split('\n').filter(l => l.trim() !== ''); const overflow = wrapped.length > PREVIEW_ROWS; const shown = wrapped.slice(0, overflow ? PREVIEW_ROWS - 1 : PREVIEW_ROWS); const more = wrapped.length - shown.length; return {shown.map((row, i) => {row} )} {more > 0 && {`… +${more} more lines`}} ; }} />; } function isSubsequence(text: string, query: string): boolean { let j = 0; for (let i = 0; i < text.length && j < query.length; i++) { if (text[i] === query[j]) j++; } return j === query.length; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useEffect","useMemo","useState","useRegisterOverlay","getTimestampedHistory","TimestampedHistoryEntry","useTerminalSize","stringWidth","wrapAnsi","Box","Text","logEvent","HistoryEntry","formatRelativeTimeAgo","truncateToWidth","FuzzyPicker","Props","initialQuery","onSelect","entry","onCancel","PREVIEW_ROWS","AGE_WIDTH","Item","display","lower","firstLine","age","HistorySearchDialog","ReactNode","columns","items","setItems","query","setQuery","cancelled","reader","loaded","return","undefined","nl","indexOf","Date","timestamp","push","toLowerCase","slice","repeat","Math","max","filtered","q","trim","exact","fuzzy","item","includes","isSubsequence","concat","previewOnRight","listWidth","floor","rowWidth","previewWidth","String","result_count","length","query_length","resolve","then","isFocused","wrapped","hard","split","filter","l","overflow","shown","more","map","row","i","text","j"],"sources":["HistorySearchDialog.tsx"],"sourcesContent":["import * as React from 'react'\nimport { useEffect, useMemo, useState } from 'react'\nimport { useRegisterOverlay } from '../context/overlayContext.js'\nimport {\n  getTimestampedHistory,\n  type TimestampedHistoryEntry,\n} from '../history.js'\nimport { useTerminalSize } from '../hooks/useTerminalSize.js'\nimport { stringWidth } from '../ink/stringWidth.js'\nimport { wrapAnsi } from '../ink/wrapAnsi.js'\nimport { Box, Text } from '../ink.js'\nimport { logEvent } from '../services/analytics/index.js'\nimport type { HistoryEntry } from '../utils/config.js'\nimport { formatRelativeTimeAgo, truncateToWidth } from '../utils/format.js'\nimport { FuzzyPicker } from './design-system/FuzzyPicker.js'\n\ntype Props = {\n  initialQuery?: string\n  onSelect: (entry: HistoryEntry) => void\n  onCancel: () => void\n}\n\nconst PREVIEW_ROWS = 6\nconst AGE_WIDTH = 8\n\ntype Item = {\n  entry: TimestampedHistoryEntry\n  display: string\n  lower: string\n  firstLine: string\n  age: string\n}\n\nexport function HistorySearchDialog({\n  initialQuery,\n  onSelect,\n  onCancel,\n}: Props): React.ReactNode {\n  useRegisterOverlay('history-search')\n  const { columns } = useTerminalSize()\n\n  const [items, setItems] = useState<Item[] | null>(null)\n  const [query, setQuery] = useState(initialQuery ?? '')\n\n  useEffect(() => {\n    let cancelled = false\n    void (async () => {\n      const reader = getTimestampedHistory()\n      const loaded: Item[] = []\n      for await (const entry of reader) {\n        if (cancelled) {\n          void reader.return(undefined)\n          return\n        }\n        const display = entry.display\n        const nl = display.indexOf('\\n')\n        const age = formatRelativeTimeAgo(new Date(entry.timestamp))\n        loaded.push({\n          entry,\n          display,\n          lower: display.toLowerCase(),\n          firstLine: nl === -1 ? display : display.slice(0, nl),\n          age: age + ' '.repeat(Math.max(0, AGE_WIDTH - stringWidth(age))),\n        })\n      }\n      if (!cancelled) setItems(loaded)\n    })()\n    return () => {\n      cancelled = true\n    }\n  }, [])\n\n  const filtered = useMemo(() => {\n    if (!items) return []\n    const q = query.trim().toLowerCase()\n    if (!q) return items\n    const exact: Item[] = []\n    const fuzzy: Item[] = []\n    for (const item of items) {\n      if (item.lower.includes(q)) {\n        exact.push(item)\n      } else if (isSubsequence(item.lower, q)) {\n        fuzzy.push(item)\n      }\n    }\n    return exact.concat(fuzzy)\n  }, [items, query])\n\n  const previewOnRight = columns >= 100\n  const listWidth = previewOnRight\n    ? Math.floor((columns - 6) * 0.5)\n    : columns - 6\n  const rowWidth = Math.max(20, listWidth - AGE_WIDTH - 1)\n  const previewWidth = previewOnRight\n    ? Math.max(20, columns - listWidth - 12)\n    : Math.max(20, columns - 10)\n\n  return (\n    <FuzzyPicker\n      title=\"Search prompts\"\n      placeholder=\"Filter history…\"\n      initialQuery={initialQuery}\n      items={filtered}\n      getKey={item => String(item.entry.timestamp)}\n      onQueryChange={setQuery}\n      onSelect={item => {\n        logEvent('tengu_history_picker_select', {\n          result_count: filtered.length,\n          query_length: query.length,\n        })\n        void item.entry.resolve().then(onSelect)\n      }}\n      onCancel={onCancel}\n      emptyMessage={q =>\n        items === null\n          ? 'Loading…'\n          : q\n            ? 'No matching prompts'\n            : 'No history yet'\n      }\n      selectAction=\"use\"\n      direction=\"up\"\n      previewPosition={previewOnRight ? 'right' : 'bottom'}\n      renderItem={(item, isFocused) => (\n        <Text>\n          <Text dimColor>{item.age}</Text>\n          <Text color={isFocused ? 'suggestion' : undefined}>\n            {' '}\n            {truncateToWidth(item.firstLine, rowWidth)}\n          </Text>\n        </Text>\n      )}\n      renderPreview={item => {\n        const wrapped = wrapAnsi(item.display, previewWidth, { hard: true })\n          .split('\\n')\n          .filter(l => l.trim() !== '')\n        const overflow = wrapped.length > PREVIEW_ROWS\n        const shown = wrapped.slice(\n          0,\n          overflow ? PREVIEW_ROWS - 1 : PREVIEW_ROWS,\n        )\n        const more = wrapped.length - shown.length\n        return (\n          <Box\n            flexDirection=\"column\"\n            borderStyle=\"round\"\n            borderDimColor\n            paddingX={1}\n            height={PREVIEW_ROWS + 2}\n          >\n            {shown.map((row, i) => (\n              <Text key={i} dimColor>\n                {row}\n              </Text>\n            ))}\n            {more > 0 && <Text dimColor>{`… +${more} more lines`}</Text>}\n          </Box>\n        )\n      }}\n    />\n  )\n}\n\nfunction isSubsequence(text: string, query: string): boolean {\n  let j = 0\n  for (let i = 0; i < text.length && j < query.length; i++) {\n    if (text[i] === query[j]) j++\n  }\n  return j === query.length\n}\n"],"mappings":"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","ignoreList":[]}