Files
claude-code/src/components/LogoV2/VoiceModeNotice.tsx
claude-code-best 5b1a52b8e0 更新大量 tsx 原始文件; 已经迁移 login panel; 部分 (#121)
* style(B1-1): 格式化 ink/buddy/cli/context/screens/tasks/services/keybindings/state (43 files)

纯格式化:移除分号、React Compiler import、import 多行展开。
修复了 Box.tsx 和 ScrollBox.tsx 中无效的 global.d.ts import。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* style(B1-2): 格式化 commands (79 files)

纯格式化:移除分号、React Compiler import、import 多行展开。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* style(B1-3): 格式化 components/messages,permissions,mcp,sandbox,shell (104 files)

纯格式化:移除分号、React Compiler import、import 多行展开。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* style(B1-4): 格式化 components/PromptInput,FeedbackSurvey,tasks,agents,skills,design-system,wizard (73 files)

纯格式化:移除分号、React Compiler import、import 多行展开。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* style(B1-5): 格式化 components其余 + hooks + tools (232 files)

纯格式化:移除分号、React Compiler import、import 多行展开。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* style(B1-6): 格式化 main/entrypoints/utils/moreright (21 files)

纯格式化:移除分号、React Compiler import、import 多行展开。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* docs: 更新 README,新增 Run.ps1/TODO.md,删除 V6.md

- README.md: 大幅重写,更详细版本历史和配置示例
- Run.ps1: 新增 Windows 启动脚本
- TODO.md: 新增包完成清单
- V6.md: 删除(架构重构规划已不适用)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* fix: 修复以前的问题

* fix: 修复 login 面板的问题

---------

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
2026-04-04 23:24:27 +08:00

52 lines
1.9 KiB
TypeScript

import { feature } from 'bun:bundle'
import * as React from 'react'
import { useEffect, useState } from 'react'
import { Box, Text } from '../../ink.js'
import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'
import { getInitialSettings } from '../../utils/settings/settings.js'
import { isVoiceModeEnabled } from '../../voice/voiceModeEnabled.js'
import { AnimatedAsterisk } from './AnimatedAsterisk.js'
import { shouldShowOpus1mMergeNotice } from './Opus1mMergeNotice.js'
const MAX_SHOW_COUNT = 3
export function VoiceModeNotice(): React.ReactNode {
// Positive ternary pattern — see docs/feature-gating.md.
// All strings must be inside the guarded branch for dead-code elimination.
return feature('VOICE_MODE') ? <VoiceModeNoticeInner /> : null
}
function VoiceModeNoticeInner(): React.ReactNode {
// Capture eligibility once at mount — no reactive subscriptions. This sits
// at the top of the message list and enters scrollback quickly; any
// re-render after it's in scrollback would force a full terminal reset.
// If the user runs /voice this session, the notice stays visible; it won't
// show next session since voiceEnabled will be true on disk.
const [show] = useState(
() =>
isVoiceModeEnabled() &&
getInitialSettings().voiceEnabled !== true &&
(getGlobalConfig().voiceNoticeSeenCount ?? 0) < MAX_SHOW_COUNT &&
!shouldShowOpus1mMergeNotice(),
)
useEffect(() => {
if (!show) return
// Capture outside the updater so StrictMode's second invocation is a no-op.
const newCount = (getGlobalConfig().voiceNoticeSeenCount ?? 0) + 1
saveGlobalConfig(prev => {
if ((prev.voiceNoticeSeenCount ?? 0) >= newCount) return prev
return { ...prev, voiceNoticeSeenCount: newCount }
})
}, [show])
if (!show) return null
return (
<Box paddingLeft={2}>
<AnimatedAsterisk />
<Text dimColor> Voice mode is now available · /voice to enable</Text>
</Box>
)
}