Spaces:
Running
Running
| import { useCallback, useEffect, useLayoutEffect, useRef, useState } from 'react' | |
| import { createPortal } from 'react-dom' | |
| import ReactMarkdown from 'react-markdown' | |
| import remarkGfm from 'remark-gfm' | |
| import { | |
| AlertTriangle, | |
| ChevronDown, | |
| Copy, | |
| Download, | |
| Loader2, | |
| MessageSquarePlus, | |
| Mic, | |
| Pause, | |
| Play, | |
| RefreshCw, | |
| Search, | |
| Send, | |
| Settings, | |
| Square, | |
| Share2, | |
| Trash2, | |
| SkipBack, | |
| Volume2, | |
| } from 'lucide-react' | |
| const STORAGE_PERSONA = 'askjerry_extra_persona' | |
| const STORAGE_ALWAYS_SPEAK = 'askjerry_always_speak' | |
| const STORAGE_TTS_PRIMED = 'askjerry_tts_primed' | |
| const STORAGE_TTS_SPEED = 'askjerry_tts_speed' | |
| const TTS_SPEED_OPTIONS = [1, 1.5, 2] | |
| const CONTEXT_WINDOW = 8192 | |
| const MAX_REPLY_TOKENS = 4096 | |
| const SUMMARIZE_THRESHOLD = 0.55 | |
| const DEFAULT_EXTRA_PERSONA = `You are Tawish Jerry Huaute, a Chumash man who spent 14+ years at Microsoft as a Senior Field Technical Account Manager starting in 1994. You grew up in Pomona, California. Your grandfather Semu Huaute (1908–2004) was a legendary Chumash medicine man, activist, and founder of the Red Wind Foundation who participated in the 1969 Alcatraz occupation. You are a member of Native Americans at Microsoft, mentoring Indigenous youth in tech careers. | |
| Your core beliefs: "Don't let an opportunity pass you by. You need to invest in yourself." Keep learning, seize chances, and build your value through education and hands-on experience. You are proud of your Chumash heritage and see it as inseparable from your professional identity — not a contradiction. | |
| Your tone is direct, warm, unpretentious, and encouraging. You speak from lived experience, not theory. You naturally mentor younger people, especially Native youth who may not see representation in tech. You know enterprise IT deeply — deployment, troubleshooting, account management, customer relationships, Microsoft technologies. You don't lead with credentials; you lead with real talk. | |
| When discussing heritage, speak with quiet pride about your family and Chumash culture. When giving career advice, be practical and honest — you believe in people but expect them to put in the work.` | |
| function initialExtraPersona() { | |
| try { | |
| const s = sessionStorage.getItem(STORAGE_PERSONA) | |
| if (s !== null) return s | |
| } catch { /* */ } | |
| return DEFAULT_EXTRA_PERSONA | |
| } | |
| function initialAlwaysSpeak() { | |
| try { | |
| return sessionStorage.getItem(STORAGE_ALWAYS_SPEAK) === '1' | |
| } catch { | |
| return false | |
| } | |
| } | |
| function initialTtsSpeed() { | |
| try { | |
| const v = parseFloat(sessionStorage.getItem(STORAGE_TTS_SPEED)) | |
| if (TTS_SPEED_OPTIONS.includes(v)) return v | |
| } catch { /* */ } | |
| return 1 | |
| } | |
| function estimateTokens(text) { | |
| return Math.max(1, Math.ceil((text || '').length / 4)) | |
| } | |
| function estimateMessagesTokens(msgs, systemPrompt, summary) { | |
| let total = estimateTokens(systemPrompt) + 4 | |
| if (summary) total += estimateTokens(summary) + 8 | |
| for (const m of msgs) total += estimateTokens(m.content) + 4 | |
| return total | |
| } | |
| function parseSseBuffer(buf, onEvent) { | |
| const normalized = buf.replace(/\r\n/g, '\n') | |
| const idx = normalized.lastIndexOf('\n') | |
| if (idx === -1) return normalized | |
| const complete = normalized.slice(0, idx + 1) | |
| const remainder = normalized.slice(idx + 1) | |
| for (const line of complete.split('\n')) { | |
| if (!line.startsWith('data: ')) continue | |
| const raw = line.slice(6).trim() | |
| if (raw === '[DONE]') { onEvent({ type: 'done' }); continue } | |
| if (!raw) continue | |
| try { onEvent(JSON.parse(raw)) } catch { /* */ } | |
| } | |
| return remainder | |
| } | |
| /** | |
| * Extract complete sentences from text. Boundaries are stable: once a sentence | |
| * ends with punctuation, it never changes even as more text streams in. | |
| * When includeTrailing is true, any text after the last sentence terminator is | |
| * also returned (used after streaming ends so nothing is silently dropped). | |
| */ | |
| function extractSentences(text, includeTrailing = false) { | |
| const t = (text || '').trim() | |
| if (!t) return [] | |
| const re = /[^.!?\n]+[.!?\n]+/g | |
| const sentences = [] | |
| let match, lastEnd = 0 | |
| while ((match = re.exec(t)) !== null) { | |
| const s = match[0].trim() | |
| if (s) sentences.push(s) | |
| lastEnd = re.lastIndex | |
| } | |
| if (!sentences.length) return includeTrailing ? [t] : [] | |
| if (includeTrailing) { | |
| const remainder = t.slice(lastEnd).trim() | |
| if (remainder) sentences.push(remainder) | |
| } | |
| return sentences | |
| } | |
| function appendTokenContent(acc, ev) { | |
| if (ev.type !== 'token' || ev.content == null) return acc | |
| let piece = ev.content | |
| if (Array.isArray(piece)) { | |
| piece = piece.map((p) => (typeof p === 'string' ? p : p?.text ?? '')).join('') | |
| } else if (typeof piece !== 'string') { | |
| piece = String(piece) | |
| } | |
| return piece ? acc + piece : acc | |
| } | |
| function AssistantSearchBar({ content, show, speak }) { | |
| const [open, setOpen] = useState(false) | |
| const [draft, setDraft] = useState('') | |
| const [loading, setLoading] = useState(false) | |
| const [copied, setCopied] = useState(false) | |
| const [responseCopied, setResponseCopied] = useState(false) | |
| const [placement, setPlacement] = useState('below') | |
| const [popoverStyle, setPopoverStyle] = useState(null) | |
| const wrapRef = useRef(null) | |
| const measurePopoverPosition = useCallback(() => { | |
| if (!open) { | |
| setPopoverStyle(null) | |
| return | |
| } | |
| const bubble = wrapRef.current?.closest('.aj-bubble-wrap') | |
| if (!bubble) return | |
| const br = bubble.getBoundingClientRect() | |
| const vw = window.visualViewport?.width ?? window.innerWidth | |
| const margin = 12 | |
| const gap = 8 | |
| const popoverW = Math.min(360, vw - margin * 2) | |
| const spaceRight = vw - br.right - margin | |
| const canPlaceRight = spaceRight >= popoverW + gap | |
| const vh = window.visualViewport?.height ?? window.innerHeight | |
| let style | |
| if (canPlaceRight) { | |
| setPlacement('right') | |
| const w = Math.min(popoverW, vw - br.right - gap - margin) | |
| style = { position: 'fixed', visibility: 'visible', bottom: vh - br.bottom, left: br.right + gap, width: w } | |
| } else { | |
| setPlacement('below') | |
| const w = Math.min(br.width, popoverW, vw - 2 * margin) | |
| const left = Math.max(margin, Math.min(br.left, vw - w - margin)) | |
| style = { position: 'fixed', visibility: 'visible', top: br.bottom + gap, left, width: w } | |
| } | |
| setPopoverStyle(style) | |
| }, [open]) | |
| useLayoutEffect(() => { | |
| measurePopoverPosition() | |
| }, [open, loading, draft, measurePopoverPosition]) | |
| useEffect(() => { | |
| if (!open) return | |
| const onMove = () => measurePopoverPosition() | |
| window.addEventListener('resize', onMove) | |
| window.addEventListener('scroll', onMove, true) | |
| return () => { | |
| window.removeEventListener('resize', onMove) | |
| window.removeEventListener('scroll', onMove, true) | |
| } | |
| }, [open, measurePopoverPosition]) | |
| useEffect(() => { | |
| if (!open) return | |
| const onDoc = (e) => { | |
| if (wrapRef.current && !wrapRef.current.contains(e.target)) setOpen(false) | |
| } | |
| document.addEventListener('mousedown', onDoc) | |
| return () => document.removeEventListener('mousedown', onDoc) | |
| }, [open]) | |
| const openPanel = async () => { | |
| if (open) { | |
| setOpen(false) | |
| return | |
| } | |
| setOpen(true) | |
| setCopied(false) | |
| setLoading(true) | |
| setDraft('') | |
| const text = (content || '').toString().slice(0, 4000) | |
| try { | |
| const res = await fetch('/api/search-references', { | |
| method: 'POST', | |
| headers: { 'Content-Type': 'application/json' }, | |
| body: JSON.stringify({ statement: text }), | |
| }) | |
| const data = await res.json().catch(() => ({})) | |
| const q = (data.search_query || '').trim() || text.slice(0, 100) | |
| setDraft(`Provide citations for ${q}`) | |
| } catch { | |
| setDraft(`Provide citations for ${text.slice(0, 100)}`) | |
| } finally { | |
| setLoading(false) | |
| } | |
| } | |
| const copyDraft = () => { | |
| navigator.clipboard.writeText(draft).then(() => { | |
| setCopied(true) | |
| setTimeout(() => setCopied(false), 2000) | |
| }).catch(() => {}) | |
| } | |
| const copyResponse = () => { | |
| const text = (content || '').toString().trim() | |
| if (!text) return | |
| navigator.clipboard.writeText(text).then(() => { | |
| setResponseCopied(true) | |
| setTimeout(() => setResponseCopied(false), 2000) | |
| }).catch(() => {}) | |
| } | |
| const openPerplexity = () => { | |
| window.open(`https://www.perplexity.ai/?q=${encodeURIComponent(draft)}`, '_blank', 'noopener,noreferrer') | |
| } | |
| const openWebSearch = () => { | |
| window.open(`https://www.google.com/search?q=${encodeURIComponent(draft)}`, '_blank', 'noopener,noreferrer') | |
| } | |
| if (!show) return null | |
| if ( | |
| !(content || '').trim() | |
| && !speak?.loading | |
| && !speak?.playing | |
| && !speak?.paused | |
| ) { | |
| return null | |
| } | |
| return ( | |
| <div className="aj-msg-actions" ref={wrapRef}> | |
| <div className="aj-msg-search-wrap"> | |
| {speak && ( | |
| <> | |
| <button | |
| type="button" | |
| className="aj-msg-search-btn" | |
| onClick={ | |
| speak.paused | |
| ? speak.onResume | |
| : speak.playing | |
| ? speak.onPause | |
| : speak.onReadAloud | |
| } | |
| disabled={speak.disabled || speak.loading} | |
| data-tip={ | |
| speak.paused | |
| ? 'Resume' | |
| : speak.playing | |
| ? 'Pause' | |
| : speak.loading | |
| ? 'Loading speech…' | |
| : 'Read aloud' | |
| } | |
| aria-label={ | |
| speak.paused | |
| ? 'Resume' | |
| : speak.playing | |
| ? 'Pause' | |
| : speak.loading | |
| ? 'Loading speech' | |
| : 'Read aloud' | |
| } | |
| > | |
| {speak.paused ? ( | |
| <Play size={14} aria-hidden /> | |
| ) : speak.loading ? ( | |
| <Loader2 size={14} className="aj-spin" aria-hidden /> | |
| ) : speak.playing ? ( | |
| <Pause size={14} aria-hidden /> | |
| ) : ( | |
| <Volume2 size={14} aria-hidden /> | |
| )} | |
| </button> | |
| {(speak.playing || speak.paused) && ( | |
| <button | |
| type="button" | |
| className="aj-msg-search-btn" | |
| onClick={speak.onReplay} | |
| data-tip="Skip back" | |
| aria-label="Skip back" | |
| > | |
| <SkipBack size={14} aria-hidden /> | |
| </button> | |
| )} | |
| <button | |
| type="button" | |
| className="aj-msg-search-btn aj-tts-speed-btn" | |
| onClick={speak.onCycleSpeed} | |
| data-tip="Playback speed" | |
| aria-label={`Playback speed: ${speak.speed}x`} | |
| > | |
| <span className="aj-tts-speed-label">{speak.speed}x</span> | |
| </button> | |
| {(speak.playing || speak.paused || (speak.loading && speak.showStop)) && ( | |
| <button | |
| type="button" | |
| className="aj-msg-search-btn" | |
| onClick={speak.onStopReading} | |
| data-tip="Stop reading" | |
| aria-label="Stop reading" | |
| > | |
| <Square size={14} aria-hidden /> | |
| </button> | |
| )} | |
| </> | |
| )} | |
| <button | |
| type="button" | |
| className="aj-msg-search-btn" | |
| onClick={copyResponse} | |
| disabled={!(content || '').trim()} | |
| data-tip={responseCopied ? 'Copied!' : 'Copy response'} | |
| aria-label={responseCopied ? 'Copied' : 'Copy response'} | |
| > | |
| <Copy size={14} aria-hidden /> | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-msg-search-btn" | |
| onClick={openPanel} | |
| data-tip="Search for citations" | |
| aria-label="Search for citations" | |
| > | |
| <Search size={14} aria-hidden /> | |
| </button> | |
| </div> | |
| {open && ( | |
| <div | |
| className={`aj-msg-search-popover aj-msg-search-popover--${placement}`} | |
| style={popoverStyle ?? undefined} | |
| role="dialog" | |
| aria-label="Web search for citations and resources" | |
| > | |
| <div className="aj-msg-search-popover-head"> | |
| <span>Web search for citations and resources</span> | |
| <button type="button" className="aj-msg-search-close" onClick={() => setOpen(false)} aria-label="Close"> | |
| × | |
| </button> | |
| </div> | |
| {loading ? ( | |
| <div className="aj-msg-search-loading">Generating search query…</div> | |
| ) : ( | |
| <> | |
| <textarea | |
| className="aj-msg-search-input" | |
| value={draft} | |
| onChange={(e) => setDraft(e.target.value)} | |
| rows={3} | |
| aria-label="Edit search query" | |
| /> | |
| <div className="aj-msg-search-actions"> | |
| <button type="button" className="aj-btn primary aj-msg-search-action" onClick={openPerplexity}> | |
| Open in Perplexity | |
| </button> | |
| <button type="button" className="aj-btn primary aj-msg-search-action" onClick={copyDraft}> | |
| {copied ? 'Copied!' : 'Copy'} | |
| </button> | |
| <button type="button" className="aj-btn primary aj-msg-search-action" onClick={openWebSearch}> | |
| Search | |
| </button> | |
| </div> | |
| </> | |
| )} | |
| </div> | |
| )} | |
| </div> | |
| ) | |
| } | |
| function AssistantBubbleContent({ content }) { | |
| return ( | |
| <div className="aj-bubble-md"> | |
| <ReactMarkdown | |
| remarkPlugins={[remarkGfm]} | |
| components={{ | |
| a: ({ href, children, ...rest }) => ( | |
| <a href={href} target="_blank" rel="noopener noreferrer" {...rest}> | |
| {children} | |
| </a> | |
| ), | |
| }} | |
| > | |
| {content} | |
| </ReactMarkdown> | |
| </div> | |
| ) | |
| } | |
| export default function App() { | |
| const [optionsOpen, setOptionsOpen] = useState(false) | |
| const [shareOpen, setShareOpen] = useState(false) | |
| const [exportOpen, setExportOpen] = useState(false) | |
| const [extraPersona, setExtraPersona] = useState(initialExtraPersona) | |
| const [messages, setMessages] = useState([]) | |
| const [input, setInput] = useState('') | |
| const [streaming, setStreaming] = useState(false) | |
| const [summary, setSummary] = useState(null) | |
| const [summarizing, setSummarizing] = useState(false) | |
| const [contextOverflow, setContextOverflow] = useState(null) | |
| const [hideTypingCursor, setHideTypingCursor] = useState(false) | |
| const [showJumpToBottom, setShowJumpToBottom] = useState(false) | |
| const abortRef = useRef(null) | |
| const streamingAssistantRowRef = useRef(null) | |
| const didScrollForStreamRef = useRef(false) | |
| const chatScrollRef = useRef(null) | |
| const optionsRef = useRef(null) | |
| const optionsPanelRef = useRef(null) | |
| const shareRef = useRef(null) | |
| const exportRef = useRef(null) | |
| const composerRef = useRef(null) | |
| const lastUserRowRef = useRef(null) | |
| const [ttsLoadingIndex, setTtsLoadingIndex] = useState(null) | |
| const [ttsPlayingIndex, setTtsPlayingIndex] = useState(null) | |
| const [ttsPaused, setTtsPaused] = useState(false) | |
| const [alwaysSpeak, setAlwaysSpeak] = useState(initialAlwaysSpeak) | |
| const [ttsSpeed, setTtsSpeed] = useState(initialTtsSpeed) | |
| const [micListening, setMicListening] = useState(false) | |
| const [micTranscribing, setMicTranscribing] = useState(false) | |
| const [voiceError, setVoiceError] = useState(null) | |
| const audioRef = useRef(null) | |
| const ttsBlobUrlRef = useRef(null) | |
| const ttsSessionRef = useRef(0) | |
| const ttsSpeedRef = useRef(ttsSpeed) | |
| ttsSpeedRef.current = ttsSpeed | |
| const messagesRef = useRef(messages) | |
| messagesRef.current = messages | |
| const streamingRef = useRef(streaming) | |
| streamingRef.current = streaming | |
| const ttsContentResolverRef = useRef(null) | |
| const ttsPlaybackActiveRef = useRef(false) | |
| const ttsSkipRef = useRef(false) | |
| const ttsReplayDeltaRef = useRef(null) | |
| const streamingWasRef = useRef(false) | |
| const mediaRecorderRef = useRef(null) | |
| const audioChunksRef = useRef([]) | |
| useEffect(() => { | |
| const el = document.documentElement | |
| if (hideTypingCursor) el.classList.add('aj-hide-pointer') | |
| else el.classList.remove('aj-hide-pointer') | |
| return () => { | |
| el.classList.remove('aj-hide-pointer') | |
| } | |
| }, [hideTypingCursor]) | |
| useEffect(() => { | |
| if (!hideTypingCursor) return | |
| const show = () => setHideTypingCursor(false) | |
| window.addEventListener('mousemove', show, { passive: true }) | |
| return () => window.removeEventListener('mousemove', show) | |
| }, [hideTypingCursor]) | |
| const updateJumpToBottomVisibility = useCallback(() => { | |
| const el = chatScrollRef.current | |
| if (!el) return | |
| const { scrollHeight, clientHeight, scrollTop } = el | |
| const canScroll = scrollHeight > clientHeight + 2 | |
| const atBottom = scrollTop + clientHeight >= scrollHeight - 48 | |
| setShowJumpToBottom(canScroll && !atBottom) | |
| }, []) | |
| useLayoutEffect(() => { | |
| updateJumpToBottomVisibility() | |
| }, [messages, updateJumpToBottomVisibility]) | |
| useLayoutEffect(() => { | |
| if (!streaming) { | |
| didScrollForStreamRef.current = false | |
| return | |
| } | |
| if (didScrollForStreamRef.current) return | |
| const doScroll = () => { | |
| const container = chatScrollRef.current | |
| const target = lastUserRowRef.current || streamingAssistantRowRef.current | |
| if (!container || !target) return false | |
| const canScroll = container.scrollHeight > container.clientHeight + 2 | |
| if (!canScroll) return false | |
| const offset = target.offsetTop - container.offsetTop | |
| container.scrollTop = offset | |
| const delta = Math.abs( | |
| target.getBoundingClientRect().top - container.getBoundingClientRect().top, | |
| ) | |
| if (delta < 12) didScrollForStreamRef.current = true | |
| updateJumpToBottomVisibility() | |
| return true | |
| } | |
| if (doScroll()) return undefined | |
| const id = requestAnimationFrame(() => { | |
| if (!didScrollForStreamRef.current) doScroll() | |
| }) | |
| return () => cancelAnimationFrame(id) | |
| }, [streaming, messages, updateJumpToBottomVisibility]) | |
| useEffect(() => { | |
| const el = chatScrollRef.current | |
| if (!el) return | |
| const onScroll = () => updateJumpToBottomVisibility() | |
| el.addEventListener('scroll', onScroll, { passive: true }) | |
| window.addEventListener('resize', onScroll) | |
| return () => { | |
| el.removeEventListener('scroll', onScroll) | |
| window.removeEventListener('resize', onScroll) | |
| } | |
| }, [updateJumpToBottomVisibility]) | |
| const scrollToLatestQuestion = useCallback(() => { | |
| const container = chatScrollRef.current | |
| if (!container) return | |
| const userRow = lastUserRowRef.current | |
| if (userRow) { | |
| const offset = userRow.offsetTop - container.offsetTop | |
| container.scrollTo({ top: offset, behavior: 'smooth' }) | |
| } else { | |
| container.scrollTo({ top: container.scrollHeight, behavior: 'smooth' }) | |
| } | |
| requestAnimationFrame(() => { | |
| updateJumpToBottomVisibility() | |
| composerRef.current?.focus() | |
| }) | |
| }, [updateJumpToBottomVisibility]) | |
| const adjustComposerHeight = useCallback(() => { | |
| const el = composerRef.current | |
| if (!el) return | |
| const minPx = 34 | |
| const maxPx = window.innerHeight * 0.5 | |
| el.style.height = '0px' | |
| const sh = Math.max(minPx, el.scrollHeight) | |
| if (sh <= maxPx) { | |
| el.style.height = `${sh}px` | |
| el.style.overflowY = 'hidden' | |
| } else { | |
| el.style.height = `${maxPx}px` | |
| el.style.overflowY = 'auto' | |
| } | |
| }, []) | |
| useLayoutEffect(() => { | |
| adjustComposerHeight() | |
| }, [input, adjustComposerHeight]) | |
| useEffect(() => { | |
| const onResize = () => adjustComposerHeight() | |
| window.addEventListener('resize', onResize) | |
| return () => window.removeEventListener('resize', onResize) | |
| }, [adjustComposerHeight]) | |
| useEffect(() => { | |
| try { sessionStorage.setItem(STORAGE_PERSONA, extraPersona) } catch { /* */ } | |
| }, [extraPersona]) | |
| useEffect(() => { | |
| if (!optionsOpen) return | |
| const onDown = (e) => { | |
| const t = e.target | |
| const inWrap = optionsRef.current?.contains(t) | |
| const inPanel = optionsPanelRef.current?.contains(t) | |
| if (!inWrap && !inPanel) setOptionsOpen(false) | |
| } | |
| const onKey = (e) => { if (e.key === 'Escape') setOptionsOpen(false) } | |
| document.addEventListener('mousedown', onDown) | |
| document.addEventListener('keydown', onKey) | |
| return () => { | |
| document.removeEventListener('mousedown', onDown) | |
| document.removeEventListener('keydown', onKey) | |
| } | |
| }, [optionsOpen]) | |
| /* Options panel: position fixed + CSS vars so it is not clipped by .aj-app overflow:hidden */ | |
| useLayoutEffect(() => { | |
| if (!optionsOpen) { | |
| document.documentElement.style.removeProperty('--aj-options-panel-top') | |
| document.documentElement.style.removeProperty('--aj-options-panel-right') | |
| return | |
| } | |
| const update = () => { | |
| const el = optionsRef.current | |
| if (!el) return | |
| const rect = el.getBoundingClientRect() | |
| document.documentElement.style.setProperty('--aj-options-panel-top', `${Math.round(rect.bottom + 8)}px`) | |
| document.documentElement.style.setProperty('--aj-options-panel-right', `${Math.round(window.innerWidth - rect.right)}px`) | |
| } | |
| update() | |
| window.addEventListener('resize', update) | |
| window.addEventListener('scroll', update, true) | |
| return () => { | |
| window.removeEventListener('resize', update) | |
| window.removeEventListener('scroll', update, true) | |
| document.documentElement.style.removeProperty('--aj-options-panel-top') | |
| document.documentElement.style.removeProperty('--aj-options-panel-right') | |
| } | |
| }, [optionsOpen]) | |
| useEffect(() => { | |
| if (!shareOpen) return | |
| const onDown = (e) => { | |
| if (shareRef.current && !shareRef.current.contains(e.target)) setShareOpen(false) | |
| } | |
| const onKey = (e) => { if (e.key === 'Escape') setShareOpen(false) } | |
| document.addEventListener('mousedown', onDown) | |
| document.addEventListener('keydown', onKey) | |
| return () => { | |
| document.removeEventListener('mousedown', onDown) | |
| document.removeEventListener('keydown', onKey) | |
| } | |
| }, [shareOpen]) | |
| useEffect(() => { | |
| if (!exportOpen) return | |
| const onDown = (e) => { | |
| if (exportRef.current && !exportRef.current.contains(e.target)) setExportOpen(false) | |
| } | |
| const onKey = (e) => { if (e.key === 'Escape') setExportOpen(false) } | |
| document.addEventListener('mousedown', onDown) | |
| document.addEventListener('keydown', onKey) | |
| return () => { | |
| document.removeEventListener('mousedown', onDown) | |
| document.removeEventListener('keydown', onKey) | |
| } | |
| }, [exportOpen]) | |
| const stopTts = useCallback(() => { | |
| ttsSessionRef.current += 1 | |
| ttsSkipRef.current = false | |
| ttsReplayDeltaRef.current = null | |
| if (audioRef.current) { | |
| audioRef.current.pause() | |
| audioRef.current.src = '' | |
| audioRef.current = null | |
| } | |
| if (ttsBlobUrlRef.current) { | |
| URL.revokeObjectURL(ttsBlobUrlRef.current) | |
| ttsBlobUrlRef.current = null | |
| } | |
| setTtsLoadingIndex(null) | |
| setTtsPlayingIndex(null) | |
| setTtsPaused(false) | |
| }, []) | |
| const pauseTts = useCallback(() => { | |
| const a = audioRef.current | |
| if (a && !a.paused) { | |
| a.pause() | |
| setTtsPaused(true) | |
| } | |
| }, []) | |
| const resumeTts = useCallback(async () => { | |
| const a = audioRef.current | |
| if (!a) return | |
| try { | |
| a.playbackRate = ttsSpeedRef.current | |
| await a.play() | |
| setTtsPaused(false) | |
| } catch (e) { | |
| setVoiceError(String(e?.message || e)) | |
| } | |
| }, []) | |
| const replayTts = useCallback(() => { | |
| const a = audioRef.current | |
| if (!a) return | |
| const EARLY_THRESHOLD = 1.0 | |
| if (a.currentTime > EARLY_THRESHOLD) { | |
| a.currentTime = 0 | |
| if (a.paused) a.play().catch(() => {}) | |
| setTtsPaused(false) | |
| } else { | |
| ttsReplayDeltaRef.current = -1 | |
| ttsSkipRef.current = true | |
| } | |
| }, []) | |
| const playAudioUrlUntilDone = useCallback((url, session) => { | |
| return new Promise((resolve) => { | |
| if (session !== ttsSessionRef.current) { | |
| resolve() | |
| return | |
| } | |
| const audio = new Audio(url) | |
| audioRef.current = audio | |
| ttsBlobUrlRef.current = url | |
| let settled = false | |
| let pollAbort = null | |
| const finish = () => { | |
| if (settled) return | |
| settled = true | |
| if (pollAbort != null) clearInterval(pollAbort) | |
| audio.onended = null | |
| audio.onerror = null | |
| resolve() | |
| } | |
| pollAbort = setInterval(() => { | |
| if (session !== ttsSessionRef.current) finish() | |
| if (ttsSkipRef.current) { | |
| audio.pause() | |
| finish() | |
| } | |
| }, 120) | |
| audio.onended = () => { | |
| finish() | |
| } | |
| audio.onerror = () => { | |
| finish() | |
| } | |
| audio.playbackRate = ttsSpeedRef.current | |
| audio.play().catch(() => finish()) | |
| }) | |
| }, []) | |
| const fetchTtsAudio = useCallback(async (chunkText) => { | |
| for (let attempt = 0; attempt < 2; attempt++) { | |
| try { | |
| const res = await fetch('/api/tts', { | |
| method: 'POST', | |
| headers: { 'Content-Type': 'application/json' }, | |
| body: JSON.stringify({ text: chunkText }), | |
| }) | |
| if (!res.ok) { | |
| let detail = `TTS failed (${res.status})` | |
| try { | |
| const ct = res.headers.get('content-type') || '' | |
| if (ct.includes('json')) { | |
| const j = await res.json() | |
| if (j?.detail != null) { | |
| detail = typeof j.detail === 'string' ? j.detail : JSON.stringify(j.detail) | |
| } | |
| } | |
| } catch { /* */ } | |
| if (res.status >= 500 && attempt === 0) { | |
| continue | |
| } | |
| return { error: detail } | |
| } | |
| const blob = await res.blob() | |
| return { url: URL.createObjectURL(blob) } | |
| } catch (e) { | |
| if (attempt === 0) { | |
| continue | |
| } | |
| return { error: String(e?.message || e) } | |
| } | |
| } | |
| return { error: 'TTS failed after retries' } | |
| }, []) | |
| const playTtsForIndex = useCallback( | |
| async (index, initialText) => { | |
| try { sessionStorage.setItem(STORAGE_TTS_PRIMED, '1') } catch { /* */ } | |
| setVoiceError(null) | |
| stopTts() | |
| const session = ttsSessionRef.current | |
| ttsPlaybackActiveRef.current = true | |
| setTtsLoadingIndex(index) | |
| setTtsPlayingIndex(null) | |
| setTtsPaused(false) | |
| const LOOKAHEAD = 2 | |
| let playedCount = 0 | |
| let anyPlayed = false | |
| let lastErr = null | |
| const inFlight = new Map() | |
| const recentUrls = new Map() | |
| const revokeRecent = (keepFrom = Infinity) => { | |
| for (const [idx, u] of recentUrls) { | |
| if (idx < keepFrom) { | |
| URL.revokeObjectURL(u) | |
| recentUrls.delete(idx) | |
| } | |
| } | |
| } | |
| const getSentencesAndLimit = () => { | |
| const live = streamingRef.current | |
| const text = (messagesRef.current[index]?.content || initialText || '').trim() | |
| const sentences = extractSentences(text, !live) | |
| return { sentences, limit: sentences.length, live } | |
| } | |
| try { | |
| // eslint-disable-next-line no-constant-condition | |
| while (true) { | |
| if (session !== ttsSessionRef.current) return | |
| const { sentences, limit, live } = getSentencesAndLimit() | |
| if (playedCount < limit) { | |
| for (let ahead = playedCount; ahead < Math.min(playedCount + LOOKAHEAD, limit); ahead++) { | |
| if (!inFlight.has(ahead) && !recentUrls.has(ahead)) { | |
| inFlight.set(ahead, fetchTtsAudio(sentences[ahead])) | |
| } | |
| } | |
| let result | |
| if (recentUrls.has(playedCount)) { | |
| result = { url: recentUrls.get(playedCount) } | |
| } else { | |
| if (!anyPlayed) setTtsLoadingIndex(index) | |
| result = await (inFlight.get(playedCount) || fetchTtsAudio(sentences[playedCount])) | |
| inFlight.delete(playedCount) | |
| } | |
| if (session !== ttsSessionRef.current) return | |
| if (!result?.url) { | |
| if (result?.error) lastErr = result.error | |
| playedCount++ | |
| continue | |
| } | |
| anyPlayed = true | |
| setTtsLoadingIndex(null) | |
| setTtsPlayingIndex(index) | |
| setTtsPaused(false) | |
| recentUrls.set(playedCount, result.url) | |
| revokeRecent(playedCount - 2) | |
| await playAudioUrlUntilDone(result.url, session) | |
| if (ttsSkipRef.current) { | |
| ttsSkipRef.current = false | |
| const delta = ttsReplayDeltaRef.current ?? 0 | |
| ttsReplayDeltaRef.current = null | |
| playedCount = Math.max(0, playedCount + delta) | |
| setTtsPaused(false) | |
| continue | |
| } | |
| if (ttsBlobUrlRef.current === result.url) { | |
| ttsBlobUrlRef.current = null | |
| audioRef.current = null | |
| } | |
| playedCount++ | |
| if (session !== ttsSessionRef.current) return | |
| continue | |
| } | |
| if (!live) break | |
| setTtsLoadingIndex(index) | |
| await new Promise(resolve => { | |
| ttsContentResolverRef.current = resolve | |
| const rechk = getSentencesAndLimit() | |
| if (rechk.limit > playedCount || !rechk.live) { | |
| ttsContentResolverRef.current = null | |
| resolve() | |
| } | |
| }) | |
| } | |
| setTtsLoadingIndex(null) | |
| setTtsPlayingIndex(null) | |
| setTtsPaused(false) | |
| if (!anyPlayed && lastErr) setVoiceError(lastErr) | |
| } catch (e) { | |
| setVoiceError(String(e?.message || e)) | |
| setTtsLoadingIndex(null) | |
| setTtsPlayingIndex(null) | |
| setTtsPaused(false) | |
| } finally { | |
| ttsPlaybackActiveRef.current = false | |
| revokeRecent() | |
| } | |
| }, | |
| [stopTts, playAudioUrlUntilDone, fetchTtsAudio], | |
| ) | |
| useEffect(() => { | |
| const last = messages[messages.length - 1] | |
| if (streaming && alwaysSpeak && !ttsPlaybackActiveRef.current) { | |
| if (last?.role === 'assistant' && last.content?.trim()) { | |
| const sentences = extractSentences(last.content.trim()) | |
| if (sentences.length >= 2) { | |
| playTtsForIndex(messages.length - 1, last.content) | |
| } | |
| } | |
| } | |
| if (streamingWasRef.current && !streaming && alwaysSpeak && !ttsPlaybackActiveRef.current) { | |
| if (last?.role === 'assistant' && last.content?.trim()) { | |
| playTtsForIndex(messages.length - 1, last.content) | |
| } | |
| } | |
| streamingWasRef.current = streaming | |
| }, [streaming, alwaysSpeak, messages, playTtsForIndex]) | |
| useEffect(() => { | |
| try { sessionStorage.setItem(STORAGE_ALWAYS_SPEAK, alwaysSpeak ? '1' : '0') } catch { /* */ } | |
| }, [alwaysSpeak]) | |
| useEffect(() => { | |
| try { sessionStorage.setItem(STORAGE_TTS_SPEED, String(ttsSpeed)) } catch { /* */ } | |
| const a = audioRef.current | |
| if (a) a.playbackRate = ttsSpeed | |
| }, [ttsSpeed]) | |
| const cycleTtsSpeed = useCallback(() => { | |
| setTtsSpeed(prev => { | |
| const idx = TTS_SPEED_OPTIONS.indexOf(prev) | |
| return TTS_SPEED_OPTIONS[(idx + 1) % TTS_SPEED_OPTIONS.length] | |
| }) | |
| }, []) | |
| useEffect(() => { | |
| if (ttsContentResolverRef.current) { | |
| ttsContentResolverRef.current() | |
| ttsContentResolverRef.current = null | |
| } | |
| }, [messages, streaming]) | |
| useEffect(() => { | |
| return () => { stopTts() } | |
| }, [stopTts]) | |
| const toggleMic = useCallback(async () => { | |
| if (mediaRecorderRef.current && mediaRecorderRef.current.state !== 'inactive') { | |
| mediaRecorderRef.current.stop() | |
| return | |
| } | |
| if (micTranscribing || streaming || contextOverflow) return | |
| try { | |
| const stream = await navigator.mediaDevices.getUserMedia({ audio: true }) | |
| const mimeType = ['audio/webm;codecs=opus', 'audio/webm', 'audio/ogg;codecs=opus', ''] | |
| .find((mt) => mt === '' || MediaRecorder.isTypeSupported(mt)) | |
| const mediaRecorder = new MediaRecorder(stream, mimeType ? { mimeType } : undefined) | |
| audioChunksRef.current = [] | |
| mediaRecorder.ondataavailable = (e) => { | |
| if (e.data && e.data.size > 0) audioChunksRef.current.push(e.data) | |
| } | |
| mediaRecorder.onstart = () => { | |
| setMicListening(true) | |
| } | |
| mediaRecorder.onstop = () => { | |
| stream.getTracks().forEach((t) => t.stop()) | |
| setMicListening(false) | |
| const blobType = mediaRecorder.mimeType || 'audio/webm' | |
| const blob = new Blob(audioChunksRef.current, { type: blobType }) | |
| if (blob.size < 100) return | |
| setMicTranscribing(true) | |
| const form = new FormData() | |
| form.append('audio', blob, 'recording.webm') | |
| fetch('/api/transcribe', { method: 'POST', body: form }) | |
| .then(async (r) => { | |
| const data = await r.json().catch(() => ({})) | |
| if (!r.ok) { | |
| const d = data?.detail | |
| const msg = | |
| typeof d === 'string' | |
| ? d | |
| : d != null | |
| ? JSON.stringify(d) | |
| : `Speech-to-text failed (${r.status})` | |
| setVoiceError(msg) | |
| return null | |
| } | |
| return data | |
| }) | |
| .then((data) => { | |
| if (!data) return | |
| const tx = data?.text?.trim() | |
| if (tx) { | |
| setVoiceError(null) | |
| setInput((prev) => (prev ? `${prev} ${tx}` : tx)) | |
| } | |
| }) | |
| .catch((e) => { | |
| setVoiceError(String(e?.message || e)) | |
| }) | |
| .finally(() => setMicTranscribing(false)) | |
| } | |
| mediaRecorderRef.current = mediaRecorder | |
| mediaRecorder.start(500) | |
| } catch { | |
| /* mic denied */ | |
| } | |
| }, [micTranscribing, streaming, contextOverflow]) | |
| const buildSystemPrompt = useCallback(() => { | |
| const base = | |
| 'You are AI Jerry, a cybersecurity-focused assistant running on the BrainForge Security model. ' + | |
| 'You give clear, practical guidance; you distinguish facts from speculation; you flag risks and ' + | |
| 'compliance considerations when relevant. You are friendly and professional.' | |
| const extra = (extraPersona || '').trim() | |
| return extra ? `${base}\n\nAdditional instructions from the user:\n${extra}` : base | |
| }, [extraPersona]) | |
| const triggerBackgroundSummarize = useCallback( | |
| async (allMessages) => { | |
| const sys = buildSystemPrompt() | |
| const est = estimateMessagesTokens( | |
| allMessages.map(({ role, content }) => ({ role, content })), | |
| sys, | |
| null, | |
| ) | |
| const inputBudget = CONTEXT_WINDOW - MAX_REPLY_TOKENS | |
| const ratio = est / inputBudget | |
| if (ratio < SUMMARIZE_THRESHOLD) return | |
| setSummarizing(true) | |
| try { | |
| const res = await fetch('/api/chat/summarize', { | |
| method: 'POST', | |
| headers: { 'Content-Type': 'application/json' }, | |
| body: JSON.stringify({ | |
| messages: allMessages.map(({ role, content }) => ({ role, content })), | |
| extra_persona: extraPersona, | |
| }), | |
| }) | |
| if (!res.ok) return | |
| const data = await res.json() | |
| if (data.summary) { | |
| setSummary(data.summary) | |
| } | |
| } catch { | |
| /* silent — summarization is best-effort */ | |
| } finally { | |
| setSummarizing(false) | |
| } | |
| }, | |
| [buildSystemPrompt, extraPersona], | |
| ) | |
| const buildApiPayload = useCallback( | |
| (rawMsgs) => { | |
| if (!summary) { | |
| return { messages: rawMsgs, summary: null } | |
| } | |
| const sys = buildSystemPrompt() | |
| const fullEst = estimateMessagesTokens(rawMsgs, sys, null) | |
| const inputBudget = CONTEXT_WINDOW - MAX_REPLY_TOKENS | |
| if (fullEst < inputBudget * 0.7) { | |
| return { messages: rawMsgs, summary: null } | |
| } | |
| const KEEP_RECENT = 4 | |
| const recent = rawMsgs.slice(-KEEP_RECENT) | |
| return { messages: recent, summary } | |
| }, | |
| [summary, buildSystemPrompt], | |
| ) | |
| const runStream = useCallback( | |
| async (rawApiMsgs) => { | |
| const ac = new AbortController() | |
| abortRef.current = ac | |
| setStreaming(true) | |
| setContextOverflow(null) | |
| let acc = '' | |
| const { messages: apiMsgs, summary: useSummary } = buildApiPayload(rawApiMsgs) | |
| try { | |
| const res = await fetch('/api/chat/stream', { | |
| method: 'POST', | |
| headers: { 'Content-Type': 'application/json' }, | |
| body: JSON.stringify({ | |
| messages: apiMsgs, | |
| extra_persona: extraPersona, | |
| summary: useSummary || undefined, | |
| }), | |
| signal: ac.signal, | |
| }) | |
| if (!res.ok) { | |
| const t = await res.text() | |
| throw new Error(t || res.statusText) | |
| } | |
| const reader = res.body.getReader() | |
| const decoder = new TextDecoder() | |
| let buffer = '' | |
| const handleEv = (ev) => { | |
| if (ev.type === 'context_overflow') { | |
| setContextOverflow({ | |
| detail: ev.detail || 'The conversation is too long for the model context window.', | |
| inputTokens: ev.input_tokens, | |
| contextWindow: ev.context_window || CONTEXT_WINDOW, | |
| }) | |
| return | |
| } | |
| if (ev.type === 'token') { | |
| const nextAcc = appendTokenContent(acc, ev) | |
| if (nextAcc !== acc) { | |
| acc = nextAcc | |
| setMessages((prev) => { | |
| const next = [...prev] | |
| const last = next[next.length - 1] | |
| if (last?.role === 'assistant') { | |
| next[next.length - 1] = { ...last, content: acc } | |
| } | |
| return next | |
| }) | |
| } | |
| } | |
| if (ev.type === 'error') { | |
| const detail = ev.detail || '' | |
| if (/context.length|max_tokens.*too large|too many tokens/i.test(detail)) { | |
| setContextOverflow({ detail }) | |
| } else { | |
| throw new Error(detail || 'Stream error') | |
| } | |
| } | |
| } | |
| while (true) { | |
| const { done, value } = await reader.read() | |
| if (done) break | |
| buffer += decoder.decode(value, { stream: true }) | |
| buffer = parseSseBuffer(buffer, handleEv) | |
| } | |
| if (buffer.trim()) { | |
| parseSseBuffer(`${buffer}\n`, handleEv) | |
| } | |
| if (!acc.trim() && !contextOverflow) { | |
| setMessages((prev) => { | |
| const next = [...prev] | |
| const last = next[next.length - 1] | |
| if (last?.role === 'assistant' && !last.content.trim()) { | |
| next[next.length - 1] = { | |
| role: 'assistant', | |
| content: '[No response] The stream finished with no text.', | |
| } | |
| } | |
| return next | |
| }) | |
| } | |
| } catch (e) { | |
| if (e.name === 'AbortError') { | |
| setMessages((prev) => { | |
| const next = [...prev] | |
| const last = next[next.length - 1] | |
| if (last?.role === 'assistant' && !last.content.trim()) next.pop() | |
| return next | |
| }) | |
| } else { | |
| setMessages((prev) => { | |
| const next = [...prev] | |
| const last = next[next.length - 1] | |
| if (last?.role === 'assistant') { | |
| next[next.length - 1] = { | |
| ...last, | |
| content: (last.content ? `${last.content}\n\n` : '') + `[Error] ${e.message}`, | |
| } | |
| } else { | |
| next.push({ role: 'assistant', content: `[Error] ${e.message}` }) | |
| } | |
| return next | |
| }) | |
| } | |
| } finally { | |
| setStreaming(false) | |
| abortRef.current = null | |
| } | |
| }, | |
| [extraPersona, buildApiPayload, contextOverflow], | |
| ) | |
| useEffect(() => { | |
| if (streaming || messages.length < 4) return | |
| const last = messages[messages.length - 1] | |
| if (last.role !== 'assistant' || !last.content.trim()) return | |
| triggerBackgroundSummarize(messages) | |
| }, [streaming, messages, triggerBackgroundSummarize]) | |
| useEffect(() => { | |
| if (!streaming && messages.length > 0) { | |
| composerRef.current?.focus() | |
| } | |
| }, [streaming, messages.length]) | |
| const handleNewChat = () => { | |
| abortRef.current?.abort() | |
| stopTts() | |
| setVoiceError(null) | |
| setMessages([]) | |
| setInput('') | |
| setSummary(null) | |
| setContextOverflow(null) | |
| } | |
| const handleStop = () => { | |
| abortRef.current?.abort() | |
| stopTts() | |
| } | |
| const handleRefresh = () => { | |
| if (streaming) return | |
| stopTts() | |
| let base = [...messages] | |
| if (base.length && base[base.length - 1].role === 'assistant') base = base.slice(0, -1) | |
| if (!base.length || base[base.length - 1].role !== 'user') return | |
| const apiMsgs = base.map(({ role, content }) => ({ role, content })) | |
| setMessages([...base, { role: 'assistant', content: '' }]) | |
| setContextOverflow(null) | |
| runStream(apiMsgs) | |
| } | |
| const handleSend = () => { | |
| const text = input.trim() | |
| if (!text || streaming) return | |
| setHideTypingCursor(false) | |
| stopTts() | |
| const userMsg = { role: 'user', content: text } | |
| const apiMsgs = [...messages, userMsg].map(({ role, content }) => ({ role, content })) | |
| setMessages([...messages, userMsg, { role: 'assistant', content: '' }]) | |
| setInput('') | |
| setContextOverflow(null) | |
| runStream(apiMsgs) | |
| } | |
| const handleOverflowRetry = () => { | |
| setContextOverflow(null) | |
| setMessages((prev) => { | |
| const next = [...prev] | |
| const last = next[next.length - 1] | |
| if (last?.role === 'assistant' && !last.content.trim()) next.pop() | |
| return next | |
| }) | |
| } | |
| const handleOverflowClear = () => { | |
| setContextOverflow(null) | |
| setMessages([]) | |
| setSummary(null) | |
| } | |
| const handleOverflowSummarize = async () => { | |
| setContextOverflow(null) | |
| setSummarizing(true) | |
| try { | |
| const res = await fetch('/api/chat/summarize', { | |
| method: 'POST', | |
| headers: { 'Content-Type': 'application/json' }, | |
| body: JSON.stringify({ | |
| messages: messages.map(({ role, content }) => ({ role, content })), | |
| extra_persona: extraPersona, | |
| }), | |
| }) | |
| if (res.ok) { | |
| const data = await res.json() | |
| if (data.summary) { | |
| setSummary(data.summary) | |
| setMessages([]) | |
| return | |
| } | |
| } | |
| setMessages([]) | |
| setSummary(null) | |
| } catch { | |
| setMessages([]) | |
| setSummary(null) | |
| } finally { | |
| setSummarizing(false) | |
| } | |
| } | |
| const onKeyDown = (e) => { | |
| if (e.key === 'Enter' && !e.shiftKey) { e.preventDefault(); handleSend() } | |
| } | |
| const buildExportText = useCallback(() => { | |
| if (!messages.length) return '' | |
| const lines = messages.map((m) => { | |
| const label = m.role === 'user' ? 'You' : 'AI Jerry' | |
| return `${label}:\n${m.content}\n` | |
| }) | |
| return `Ask Jerry — Chat Export\n${new Date().toLocaleString()}\n${'─'.repeat(40)}\n\n${lines.join('\n')}` | |
| }, [messages]) | |
| const buildExportMarkdown = useCallback(() => { | |
| if (!messages.length) return '' | |
| const ts = new Date().toLocaleString() | |
| const header = `# Ask Jerry — Chat Export\n\n_Exported: ${ts}_\n\n---\n\n` | |
| const blocks = messages.map((m) => { | |
| const heading = m.role === 'user' ? '### You' : '### AI Jerry' | |
| return `${heading}\n\n${m.content}` | |
| }) | |
| return header + blocks.join('\n\n---\n\n') | |
| }, [messages]) | |
| const copyToClipboard = async (text) => { | |
| try { | |
| await navigator.clipboard.writeText(text) | |
| } catch { | |
| const ta = document.createElement('textarea') | |
| ta.value = text | |
| ta.setAttribute('readonly', '') | |
| ta.style.position = 'fixed' | |
| ta.style.left = '-9999px' | |
| document.body.appendChild(ta) | |
| ta.select() | |
| try { | |
| document.execCommand('copy') | |
| } finally { | |
| document.body.removeChild(ta) | |
| } | |
| } | |
| } | |
| const handleExportDownload = (format) => { | |
| const text = format === 'md' ? buildExportMarkdown() : buildExportText() | |
| if (!text) return | |
| const mime = | |
| format === 'md' ? 'text/markdown;charset=utf-8' : 'text/plain;charset=utf-8' | |
| const ext = format === 'md' ? 'md' : 'txt' | |
| const blob = new Blob([text], { type: mime }) | |
| const url = URL.createObjectURL(blob) | |
| const a = document.createElement('a') | |
| a.href = url | |
| a.download = `ask-jerry-chat-${Date.now()}.${ext}` | |
| a.click() | |
| URL.revokeObjectURL(url) | |
| setExportOpen(false) | |
| } | |
| const handleShareCopyClipboard = async () => { | |
| const text = buildExportText() | |
| if (!text) return | |
| await copyToClipboard(text) | |
| setShareOpen(false) | |
| } | |
| const handleShareEmail = () => { | |
| const text = buildExportText() | |
| if (!text) return | |
| const subject = encodeURIComponent('Ask Jerry chat') | |
| const body = encodeURIComponent(text.length > 16000 ? `${text.slice(0, 16000)}\n\n… (truncated)` : text) | |
| const a = document.createElement('a') | |
| a.href = `mailto:?subject=${subject}&body=${body}` | |
| a.rel = 'noopener noreferrer' | |
| a.click() | |
| setShareOpen(false) | |
| } | |
| const handleShareLinkedIn = async () => { | |
| const text = buildExportText() | |
| if (!text) return | |
| await copyToClipboard(text) | |
| window.open('https://www.linkedin.com/feed/?shareActive=true', '_blank', 'noopener,noreferrer') | |
| setShareOpen(false) | |
| } | |
| const handleShareFacebook = async () => { | |
| const text = buildExportText() | |
| if (!text) return | |
| await copyToClipboard(text) | |
| const quote = encodeURIComponent(text.slice(0, 2000)) | |
| const fb = `https://www.facebook.com/sharer/sharer.php?quote=${quote}` | |
| if (fb.length > 2048) { | |
| window.open('https://www.facebook.com/', '_blank', 'noopener,noreferrer') | |
| } else { | |
| window.open(fb, '_blank', 'noopener,noreferrer') | |
| } | |
| setShareOpen(false) | |
| } | |
| const inputBudget = CONTEXT_WINDOW - MAX_REPLY_TOKENS | |
| const currentTokens = estimateMessagesTokens( | |
| messages.map(({ role, content }) => ({ role, content })), | |
| buildSystemPrompt(), | |
| summary && messages.length > 4 ? summary : null, | |
| ) | |
| const usagePct = Math.min(100, Math.round((currentTokens / inputBudget) * 100)) | |
| return ( | |
| <div className="aj-app"> | |
| <header className="aj-header"> | |
| <a | |
| className="aj-header-brand-link" | |
| href="https://www.neon.ai/" | |
| target="_blank" | |
| rel="noopener noreferrer" | |
| data-tip="Neon.ai" | |
| > | |
| <img src="/neon-logo.png" alt="Neon.ai" className="aj-header-logo" /> | |
| <span className="aj-header-brand-name">Neon.ai</span> | |
| </a> | |
| <div className="aj-header-title-wrap"> | |
| <h1 className="aj-header-title">Ask Jerry About Cyber Security</h1> | |
| </div> | |
| <div className="aj-options-wrap" ref={optionsRef}> | |
| {streaming && ( | |
| <button | |
| type="button" | |
| className="aj-btn danger" | |
| onClick={handleStop} | |
| data-tip="Stop" | |
| aria-label="Stop" | |
| > | |
| <Square size={16} aria-hidden /> | |
| </button> | |
| )} | |
| <button | |
| type="button" | |
| className="aj-btn" | |
| onClick={handleRefresh} | |
| disabled={streaming || messages.length === 0} | |
| data-tip="Refresh" | |
| aria-label="Refresh" | |
| > | |
| <RefreshCw size={16} aria-hidden /> | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-btn" | |
| onClick={handleNewChat} | |
| data-tip="New chat" | |
| aria-label="New chat" | |
| > | |
| <MessageSquarePlus size={16} aria-hidden /> | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-btn" | |
| onClick={() => setOptionsOpen((o) => !o)} | |
| aria-expanded={optionsOpen} | |
| aria-haspopup="dialog" | |
| aria-controls="aj-options-panel" | |
| data-tip="Options" | |
| aria-label="Options" | |
| > | |
| <Settings size={16} aria-hidden /> | |
| </button> | |
| {optionsOpen && | |
| createPortal( | |
| <div | |
| ref={optionsPanelRef} | |
| id="aj-options-panel" | |
| className="aj-options-panel" | |
| role="dialog" | |
| aria-label="Options" | |
| onMouseDown={(e) => e.stopPropagation()} | |
| > | |
| <div className="aj-options-section"> | |
| <h4 className="aj-options-section-title">Always speak</h4> | |
| <label className="aj-field aj-field-checkbox"> | |
| <input | |
| type="checkbox" | |
| checked={alwaysSpeak} | |
| onChange={(e) => setAlwaysSpeak(e.target.checked)} | |
| disabled={streaming} | |
| /> | |
| <span>Always speak responses</span> | |
| </label> | |
| <p className="aj-options-hint"> | |
| Read each assistant reply aloud automatically. | |
| </p> | |
| </div> | |
| <div className="aj-options-section"> | |
| <h4 className="aj-options-section-title">Playback speed</h4> | |
| <div className="aj-field" role="group" aria-label="Playback speed"> | |
| <div className="aj-speed-options"> | |
| {TTS_SPEED_OPTIONS.map(s => ( | |
| <button | |
| key={s} | |
| type="button" | |
| className={`aj-speed-option${ttsSpeed === s ? ' aj-speed-option--active' : ''}`} | |
| onClick={() => setTtsSpeed(s)} | |
| > | |
| {s}x | |
| </button> | |
| ))} | |
| </div> | |
| </div> | |
| </div> | |
| <div className="aj-options-section"> | |
| <h4 className="aj-options-section-title">Persona</h4> | |
| <label className="aj-field"> | |
| <span>Additional persona instructions</span> | |
| <textarea | |
| value={extraPersona} | |
| onChange={(e) => setExtraPersona(e.target.value)} | |
| placeholder="Optional — style, tone, domain focus… The assistant stays AI Jerry." | |
| rows={10} | |
| disabled={streaming} | |
| /> | |
| </label> | |
| <p className="aj-options-hint"> | |
| Merged into the system prompt for this session. | |
| </p> | |
| </div> | |
| </div>, | |
| document.body, | |
| )} | |
| </div> | |
| </header> | |
| {voiceError && ( | |
| <div className="aj-voice-error-banner" role="alert"> | |
| <span>{voiceError}</span> | |
| <button type="button" className="aj-voice-error-dismiss" onClick={() => setVoiceError(null)} aria-label="Dismiss"> | |
| × | |
| </button> | |
| </div> | |
| )} | |
| {messages.length > 0 && ( | |
| <div className="aj-context-bar"> | |
| <div className="aj-context-meter"> | |
| <div className="aj-context-fill" style={{ width: `${usagePct}%` }} data-hot={usagePct > 80 ? '' : undefined} /> | |
| </div> | |
| <span className="aj-context-label"> | |
| Context: ~{currentTokens.toLocaleString()} / {inputBudget.toLocaleString()} tokens ({usagePct}%) | |
| {summary && ' · chat history is being summarized'} | |
| {summarizing && ' · summarizing…'} | |
| </span> | |
| </div> | |
| )} | |
| <div className="aj-chat-wrap"> | |
| <div className="aj-messages" ref={chatScrollRef}> | |
| {messages.length === 0 && !summary && ( | |
| <p className="aj-sub" style={{ textAlign: 'center', marginTop: 24 }}> | |
| Message AI Jerry to begin. | |
| </p> | |
| )} | |
| {messages.length === 0 && summary && ( | |
| <div className="aj-summary-banner"> | |
| <p><strong>Continuing from a summary of your previous conversation.</strong></p> | |
| <p className="aj-summary-text">{summary}</p> | |
| </div> | |
| )} | |
| {(() => { | |
| const lastUserIdx = messages.reduce((acc, msg, idx) => msg.role === 'user' ? idx : acc, -1) | |
| return messages.map((m, i) => { | |
| const isUser = m.role === 'user' | |
| const rowRef = | |
| i === lastUserIdx | |
| ? lastUserRowRef | |
| : i === messages.length - 1 && m.role === 'assistant' | |
| ? streamingAssistantRowRef | |
| : undefined | |
| return ( | |
| <div | |
| key={i} | |
| ref={rowRef} | |
| className={`aj-msg-row ${isUser ? 'user' : ''}`} | |
| > | |
| <div className="aj-bubble-wrap"> | |
| <div className="aj-bubble-name">{isUser ? 'You' : 'AI Jerry'}</div> | |
| <div className={`aj-bubble ${!isUser ? 'aj-bubble--assistant-md' : ''}`}> | |
| {isUser ? ( | |
| m.content | |
| ) : ( | |
| <> | |
| {m.content ? <AssistantBubbleContent content={m.content} /> : null} | |
| {streaming && i === messages.length - 1 && !m.content && ( | |
| <span className="aj-typing" /> | |
| )} | |
| </> | |
| )} | |
| </div> | |
| {!isUser && ( | |
| <AssistantSearchBar | |
| content={m.content} | |
| show={!(streaming && i === messages.length - 1)} | |
| speak={{ | |
| loading: ttsLoadingIndex === i, | |
| playing: | |
| ttsPlayingIndex === i | |
| && !ttsPaused | |
| && ttsLoadingIndex !== i, | |
| paused: ttsPlayingIndex === i && ttsPaused, | |
| showStop: ttsLoadingIndex === i || ttsPlayingIndex === i, | |
| disabled: | |
| !(m.content || '').trim() | |
| && ttsLoadingIndex !== i | |
| && ttsPlayingIndex !== i, | |
| onReadAloud: () => playTtsForIndex(i, m.content), | |
| onPause: pauseTts, | |
| onResume: resumeTts, | |
| onReplay: replayTts, | |
| onCycleSpeed: cycleTtsSpeed, | |
| speed: ttsSpeed, | |
| onStopReading: stopTts, | |
| }} | |
| /> | |
| )} | |
| </div> | |
| </div> | |
| ) | |
| }) | |
| })()} | |
| </div> | |
| {contextOverflow && ( | |
| <div className="aj-overflow-banner"> | |
| <div className="aj-overflow-icon"><AlertTriangle size={22} /></div> | |
| <div className="aj-overflow-body"> | |
| <p className="aj-overflow-title">Context window exceeded</p> | |
| <p className="aj-overflow-detail"> | |
| The conversation history is too long for this model's {CONTEXT_WINDOW.toLocaleString()}-token context window. | |
| </p> | |
| <div className="aj-overflow-actions"> | |
| <button type="button" className="aj-btn" onClick={handleOverflowRetry}> | |
| <RefreshCw size={14} aria-hidden /> | |
| Try again (shorter request) | |
| </button> | |
| <button type="button" className="aj-btn" onClick={handleOverflowClear}> | |
| <Trash2 size={14} aria-hidden /> | |
| Clear chat & start fresh | |
| </button> | |
| <button type="button" className="aj-btn primary" onClick={handleOverflowSummarize} disabled={summarizing}> | |
| <MessageSquarePlus size={14} aria-hidden /> | |
| {summarizing ? 'Summarizing…' : 'New chat with summary'} | |
| </button> | |
| </div> | |
| </div> | |
| </div> | |
| )} | |
| </div> | |
| <div className="aj-composer"> | |
| <div className="aj-composer-inner"> | |
| <div className={`aj-composer-field ${micListening ? 'aj-composer-field--listening' : ''}`}> | |
| <textarea | |
| ref={composerRef} | |
| value={input} | |
| onChange={(e) => setInput(e.target.value)} | |
| onInput={(e) => { | |
| const ne = e.nativeEvent | |
| if (ne.inputType?.startsWith('insert') || (ne.data != null && ne.data !== '')) { | |
| setHideTypingCursor(true) | |
| } | |
| }} | |
| onKeyDown={onKeyDown} | |
| placeholder="Ask Jerry about cybersecurity…" | |
| rows={1} | |
| disabled={streaming || !!contextOverflow || micTranscribing} | |
| /> | |
| </div> | |
| <button | |
| type="button" | |
| className="aj-btn" | |
| onClick={toggleMic} | |
| disabled={streaming || !!contextOverflow || micTranscribing} | |
| data-tip={micListening ? 'Stop recording' : micTranscribing ? 'Transcribing…' : 'Speak your message'} | |
| aria-label={micListening ? 'Stop recording' : 'Speak your message'} | |
| > | |
| {micTranscribing ? ( | |
| <Loader2 size={18} className="aj-spin" aria-hidden /> | |
| ) : ( | |
| <Mic size={18} aria-hidden /> | |
| )} | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-btn primary" | |
| onClick={handleSend} | |
| disabled={streaming || !input.trim() || !!contextOverflow} | |
| data-tip="Send" | |
| > | |
| <Send size={18} aria-hidden /> | |
| </button> | |
| <div className="aj-share-wrap" ref={exportRef}> | |
| <button | |
| type="button" | |
| className="aj-btn" | |
| onClick={() => { | |
| setExportOpen((o) => !o) | |
| setShareOpen(false) | |
| }} | |
| disabled={!messages.length || streaming} | |
| data-tip="Export chat" | |
| aria-expanded={exportOpen} | |
| aria-haspopup="menu" | |
| aria-controls="aj-export-panel" | |
| > | |
| <Download size={18} aria-hidden /> | |
| </button> | |
| {exportOpen && ( | |
| <div | |
| id="aj-export-panel" | |
| className="aj-share-panel" | |
| role="menu" | |
| aria-label="Export format" | |
| onMouseDown={(e) => e.stopPropagation()} | |
| > | |
| <button | |
| type="button" | |
| className="aj-share-item" | |
| role="menuitem" | |
| onClick={() => handleExportDownload('md')} | |
| > | |
| Markdown (.md) | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-share-item" | |
| role="menuitem" | |
| onClick={() => handleExportDownload('txt')} | |
| > | |
| Plain text (.txt) | |
| </button> | |
| </div> | |
| )} | |
| </div> | |
| <div className="aj-share-wrap" ref={shareRef}> | |
| <button | |
| type="button" | |
| className="aj-btn" | |
| onClick={() => { | |
| setShareOpen((o) => !o) | |
| setExportOpen(false) | |
| }} | |
| disabled={!messages.length || streaming} | |
| data-tip="Share chat" | |
| aria-expanded={shareOpen} | |
| aria-haspopup="menu" | |
| aria-controls="aj-share-panel" | |
| > | |
| <Share2 size={18} aria-hidden /> | |
| </button> | |
| {shareOpen && ( | |
| <div | |
| id="aj-share-panel" | |
| className="aj-share-panel" | |
| role="menu" | |
| aria-label="Share chat" | |
| onMouseDown={(e) => e.stopPropagation()} | |
| > | |
| <button | |
| type="button" | |
| className="aj-share-item" | |
| role="menuitem" | |
| onClick={handleShareLinkedIn} | |
| > | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-share-item" | |
| role="menuitem" | |
| onClick={handleShareFacebook} | |
| > | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-share-item" | |
| role="menuitem" | |
| onClick={handleShareEmail} | |
| > | |
| </button> | |
| <button | |
| type="button" | |
| className="aj-share-item" | |
| role="menuitem" | |
| onClick={handleShareCopyClipboard} | |
| > | |
| Copy to clipboard | |
| </button> | |
| <p className="aj-share-hint"> | |
| LinkedIn and Facebook copy the conversation to your clipboard, then open the site so you can paste into a post. | |
| </p> | |
| </div> | |
| )} | |
| </div> | |
| </div> | |
| </div> | |
| {showJumpToBottom && ( | |
| <button | |
| type="button" | |
| className="aj-jump-bottom-btn" | |
| onClick={scrollToLatestQuestion} | |
| data-tip="Jump to latest question" | |
| aria-label="Jump to latest question" | |
| > | |
| <ChevronDown size={22} aria-hidden /> | |
| </button> | |
| )} | |
| </div> | |
| ) | |
| } | |