🎯 Use case
This file lives under “hooks/”, which covers reusable UI or integration hooks. On the API surface it exposes useVoiceEnabled — mainly functions, hooks, or classes. Dependencies touch React UI. It composes internal code from state and voice (relative imports).
Generated from folder role, exports, dependency roots, and inline comments — not hand-reviewed for every path.
🧠 Inline summary
import { useMemo } from 'react' import { useAppState } from '../state/AppState.js' import { hasVoiceAuth, isVoiceGrowthBookEnabled,
📤 Exports (heuristic)
useVoiceEnabled
📚 External import roots
Package roots from from "…" (relative paths omitted).
react
🖥️ Source preview
import { useMemo } from 'react'
import { useAppState } from '../state/AppState.js'
import {
hasVoiceAuth,
isVoiceGrowthBookEnabled,
} from '../voice/voiceModeEnabled.js'
/**
* Combines user intent (settings.voiceEnabled) with auth + GB kill-switch.
* Only the auth half is memoized on authVersion — it's the expensive one
* (cold getClaudeAIOAuthTokens memoize → sync `security` spawn, ~60ms/call,
* ~180ms total in profile v5 when token refresh cleared the cache mid-session).
* GB is a cheap cached-map lookup and stays outside the memo so a mid-session
* kill-switch flip still takes effect on the next render.
*
* authVersion bumps on /login only. Background token refresh leaves it alone
* (user is still authed), so the auth memo stays correct without re-eval.
*/
export function useVoiceEnabled(): boolean {
const userIntent = useAppState(s => s.settings.voiceEnabled === true)
const authVersion = useAppState(s => s.authVersion)
// eslint-disable-next-line react-hooks/exhaustive-deps
const authed = useMemo(hasVoiceAuth, [authVersion])
return userIntent && authed && isVoiceGrowthBookEnabled()
}