bingocode 1.0.1 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/bingo-win.cjs +34 -3
- package/desktop/README.md +30 -0
- package/desktop/bunfig.toml +1 -0
- package/desktop/index.html +17 -0
- package/desktop/package.json +55 -0
- package/desktop/pnpm-lock.yaml +3832 -0
- package/desktop/public/app-icon.jpg +0 -0
- package/desktop/public/fonts/inter-latin-ext.woff2 +0 -0
- package/desktop/public/fonts/inter-latin.woff2 +0 -0
- package/desktop/public/fonts/jetbrains-mono-latin-ext.woff2 +0 -0
- package/desktop/public/fonts/jetbrains-mono-latin.woff2 +0 -0
- package/desktop/public/fonts/manrope-latin-ext.woff2 +0 -0
- package/desktop/public/fonts/manrope-latin.woff2 +0 -0
- package/desktop/public/fonts/material-symbols-outlined.woff2 +0 -0
- package/desktop/public/icons/bilibili.svg +1 -0
- package/desktop/public/icons/douyin.svg +1 -0
- package/desktop/public/icons/github.svg +3 -0
- package/desktop/public/icons/xiaohongshu.svg +1 -0
- package/desktop/scripts/build-macos-arm64.sh +270 -0
- package/desktop/scripts/build-sidecars.ts +183 -0
- package/desktop/scripts/build-windows-x64.ps1 +295 -0
- package/desktop/scripts/scan-missing-imports.ts +235 -0
- package/desktop/sidecars/claude-sidecar.ts +156 -0
- package/desktop/src/App.tsx +5 -0
- package/desktop/src/__tests__/agentsSettings.test.tsx +349 -0
- package/desktop/src/__tests__/pages.test.tsx +290 -0
- package/desktop/src/__tests__/skillsSettings.test.tsx +205 -0
- package/desktop/src/api/adapters.ts +12 -0
- package/desktop/src/api/agents.ts +36 -0
- package/desktop/src/api/cliTasks.ts +28 -0
- package/desktop/src/api/client.ts +63 -0
- package/desktop/src/api/computerUse.ts +76 -0
- package/desktop/src/api/filesystem.ts +30 -0
- package/desktop/src/api/hahaOAuth.ts +38 -0
- package/desktop/src/api/models.ts +28 -0
- package/desktop/src/api/providers.ts +63 -0
- package/desktop/src/api/search.ts +29 -0
- package/desktop/src/api/sessions.ts +56 -0
- package/desktop/src/api/settings.ts +20 -0
- package/desktop/src/api/skills.ts +19 -0
- package/desktop/src/api/tasks.ts +36 -0
- package/desktop/src/api/teams.ts +44 -0
- package/desktop/src/api/websocket.ts +164 -0
- package/desktop/src/components/chat/AskUserQuestion.tsx +268 -0
- package/desktop/src/components/chat/AssistantMessage.tsx +29 -0
- package/desktop/src/components/chat/AttachmentGallery.tsx +113 -0
- package/desktop/src/components/chat/ChatInput.tsx +622 -0
- package/desktop/src/components/chat/CodeViewer.tsx +161 -0
- package/desktop/src/components/chat/ComputerUsePermissionModal.test.tsx +174 -0
- package/desktop/src/components/chat/ComputerUsePermissionModal.tsx +311 -0
- package/desktop/src/components/chat/DiffViewer.tsx +157 -0
- package/desktop/src/components/chat/FileSearchMenu.tsx +198 -0
- package/desktop/src/components/chat/ImageGalleryModal.tsx +91 -0
- package/desktop/src/components/chat/InlineImageGallery.tsx +106 -0
- package/desktop/src/components/chat/InlineTaskSummary.tsx +60 -0
- package/desktop/src/components/chat/MermaidRenderer.test.tsx +98 -0
- package/desktop/src/components/chat/MermaidRenderer.tsx +361 -0
- package/desktop/src/components/chat/MessageActionBar.tsx +27 -0
- package/desktop/src/components/chat/MessageList.test.tsx +313 -0
- package/desktop/src/components/chat/MessageList.tsx +249 -0
- package/desktop/src/components/chat/PermissionDialog.tsx +262 -0
- package/desktop/src/components/chat/SessionTaskBar.test.tsx +99 -0
- package/desktop/src/components/chat/SessionTaskBar.tsx +159 -0
- package/desktop/src/components/chat/StreamingIndicator.tsx +41 -0
- package/desktop/src/components/chat/TerminalChrome.tsx +35 -0
- package/desktop/src/components/chat/ThinkingBlock.tsx +87 -0
- package/desktop/src/components/chat/ToolCallBlock.tsx +247 -0
- package/desktop/src/components/chat/ToolCallGroup.tsx +617 -0
- package/desktop/src/components/chat/ToolResultBlock.tsx +107 -0
- package/desktop/src/components/chat/UserMessage.tsx +38 -0
- package/desktop/src/components/chat/chatBlocks.test.tsx +136 -0
- package/desktop/src/components/chat/clipboard.ts +25 -0
- package/desktop/src/components/chat/composerUtils.test.ts +55 -0
- package/desktop/src/components/chat/composerUtils.ts +149 -0
- package/desktop/src/components/controls/ModelSelector.tsx +156 -0
- package/desktop/src/components/controls/PermissionModeSelector.tsx +229 -0
- package/desktop/src/components/layout/AppShell.tsx +107 -0
- package/desktop/src/components/layout/ContentRouter.tsx +27 -0
- package/desktop/src/components/layout/ProjectFilter.tsx +126 -0
- package/desktop/src/components/layout/Sidebar.test.tsx +158 -0
- package/desktop/src/components/layout/Sidebar.tsx +384 -0
- package/desktop/src/components/layout/StatusBar.tsx +31 -0
- package/desktop/src/components/layout/TabBar.test.tsx +136 -0
- package/desktop/src/components/layout/TabBar.tsx +318 -0
- package/desktop/src/components/layout/TitleBar.tsx +96 -0
- package/desktop/src/components/layout/WindowControls.test.tsx +69 -0
- package/desktop/src/components/layout/WindowControls.tsx +89 -0
- package/desktop/src/components/markdown/MarkdownRenderer.test.tsx +100 -0
- package/desktop/src/components/markdown/MarkdownRenderer.tsx +229 -0
- package/desktop/src/components/settings/ClaudeOfficialLogin.tsx +107 -0
- package/desktop/src/components/shared/Button.tsx +63 -0
- package/desktop/src/components/shared/CopyButton.tsx +58 -0
- package/desktop/src/components/shared/DirectoryPicker.tsx +316 -0
- package/desktop/src/components/shared/Dropdown.tsx +91 -0
- package/desktop/src/components/shared/Input.tsx +38 -0
- package/desktop/src/components/shared/Modal.tsx +65 -0
- package/desktop/src/components/shared/ProjectContextChip.tsx +30 -0
- package/desktop/src/components/shared/Spinner.tsx +30 -0
- package/desktop/src/components/shared/Textarea.tsx +38 -0
- package/desktop/src/components/shared/Toast.tsx +47 -0
- package/desktop/src/components/shared/UpdateChecker.tsx +90 -0
- package/desktop/src/components/skills/SkillDetail.test.tsx +89 -0
- package/desktop/src/components/skills/SkillDetail.tsx +403 -0
- package/desktop/src/components/skills/SkillList.tsx +254 -0
- package/desktop/src/components/tasks/DayOfWeekPicker.tsx +57 -0
- package/desktop/src/components/tasks/NewTaskModal.tsx +407 -0
- package/desktop/src/components/tasks/PromptEditor.tsx +74 -0
- package/desktop/src/components/tasks/TaskEmptyState.tsx +30 -0
- package/desktop/src/components/tasks/TaskList.tsx +46 -0
- package/desktop/src/components/tasks/TaskRow.tsx +253 -0
- package/desktop/src/components/tasks/TaskRunsPanel.tsx +195 -0
- package/desktop/src/components/teams/TeamStatusBar.tsx +147 -0
- package/desktop/src/config/providerPresets.ts +78 -0
- package/desktop/src/config/spinnerVerbs.ts +193 -0
- package/desktop/src/hooks/useKeyboardShortcuts.ts +60 -0
- package/desktop/src/i18n/index.ts +54 -0
- package/desktop/src/i18n/locales/en.ts +670 -0
- package/desktop/src/i18n/locales/zh.ts +670 -0
- package/desktop/src/lib/__tests__/cronDescribe.test.ts +93 -0
- package/desktop/src/lib/cronDescribe.ts +188 -0
- package/desktop/src/lib/desktopRuntime.ts +54 -0
- package/desktop/src/lib/parseRunOutput.ts +79 -0
- package/desktop/src/main.tsx +13 -0
- package/desktop/src/mocks/data.ts +202 -0
- package/desktop/src/pages/ActiveSession.test.tsx +181 -0
- package/desktop/src/pages/ActiveSession.tsx +219 -0
- package/desktop/src/pages/AdapterSettings.tsx +375 -0
- package/desktop/src/pages/AgentTeams.tsx +200 -0
- package/desktop/src/pages/ComputerUseSettings.tsx +420 -0
- package/desktop/src/pages/EmptySession.tsx +518 -0
- package/desktop/src/pages/NewTaskModal.tsx +346 -0
- package/desktop/src/pages/ScheduledTasks.tsx +66 -0
- package/desktop/src/pages/ScheduledTasksEmpty.tsx +152 -0
- package/desktop/src/pages/ScheduledTasksList.tsx +416 -0
- package/desktop/src/pages/SessionControls.tsx +460 -0
- package/desktop/src/pages/Settings.tsx +1448 -0
- package/desktop/src/pages/ToolInspection.tsx +235 -0
- package/desktop/src/stores/adapterStore.ts +106 -0
- package/desktop/src/stores/agentStore.ts +34 -0
- package/desktop/src/stores/chatStore.test.ts +505 -0
- package/desktop/src/stores/chatStore.ts +850 -0
- package/desktop/src/stores/cliTaskStore.ts +152 -0
- package/desktop/src/stores/hahaOAuthStore.test.ts +77 -0
- package/desktop/src/stores/hahaOAuthStore.ts +97 -0
- package/desktop/src/stores/providerStore.ts +101 -0
- package/desktop/src/stores/sessionStore.test.ts +63 -0
- package/desktop/src/stores/sessionStore.ts +102 -0
- package/desktop/src/stores/settingsStore.ts +120 -0
- package/desktop/src/stores/skillStore.ts +51 -0
- package/desktop/src/stores/tabStore.ts +169 -0
- package/desktop/src/stores/taskStore.ts +68 -0
- package/desktop/src/stores/teamStore.ts +344 -0
- package/desktop/src/stores/uiStore.ts +100 -0
- package/desktop/src/stores/updateStore.test.ts +71 -0
- package/desktop/src/stores/updateStore.ts +221 -0
- package/desktop/src/theme/globals.css +465 -0
- package/desktop/src/types/adapter.ts +33 -0
- package/desktop/src/types/chat.ts +152 -0
- package/desktop/src/types/cliTask.ts +24 -0
- package/desktop/src/types/provider.ts +62 -0
- package/desktop/src/types/session.ts +27 -0
- package/desktop/src/types/settings.ts +22 -0
- package/desktop/src/types/skill.ts +38 -0
- package/desktop/src/types/task.ts +56 -0
- package/desktop/src/types/team.ts +38 -0
- package/desktop/src-tauri/Cargo.lock +5549 -0
- package/desktop/src-tauri/Cargo.toml +20 -0
- package/desktop/src-tauri/app-icon.svg +13 -0
- package/desktop/src-tauri/build.rs +3 -0
- package/desktop/src-tauri/capabilities/default.json +106 -0
- package/desktop/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml +5 -0
- package/desktop/src-tauri/icons/android/values/ic_launcher_background.xml +4 -0
- package/desktop/src-tauri/icons/icon.icns +0 -0
- package/desktop/src-tauri/icons/icon.ico +0 -0
- package/desktop/src-tauri/src/lib.rs +408 -0
- package/desktop/src-tauri/src/main.rs +6 -0
- package/desktop/src-tauri/tauri.conf.json +78 -0
- package/desktop/src-tauri/tauri.macos.conf.json +18 -0
- package/desktop/src-tauri/tauri.release-ci.json +5 -0
- package/desktop/src-tauri/tauri.windows.conf.json +16 -0
- package/desktop/src-tauri/windows-installer-hooks.nsh +17 -0
- package/desktop/tsconfig.json +25 -0
- package/desktop/vite.config.ts +26 -0
- package/desktop/vitest.config.ts +18 -0
- package/package.json +1 -1
- package/src/commands/desktop/desktop.tsx +9 -0
- package/src/commands/desktop/index.ts +26 -0
|
@@ -0,0 +1,460 @@
|
|
|
1
|
+
import { useState } from 'react'
|
|
2
|
+
import {
|
|
3
|
+
mockPermissionModes,
|
|
4
|
+
mockModels,
|
|
5
|
+
mockEffortLevels,
|
|
6
|
+
mockSessions,
|
|
7
|
+
mockStatusBar,
|
|
8
|
+
} from '../mocks/data'
|
|
9
|
+
|
|
10
|
+
/* ────────────────────────────────────────────────────────────────────
|
|
11
|
+
Icon helpers for the permission modes (matching HTML prototype exactly)
|
|
12
|
+
──────────────────────────────────────────────────────────────────── */
|
|
13
|
+
const permissionIcons: Record<string, { icon: string; color: string; selectedColor?: string }> = {
|
|
14
|
+
ask: { icon: 'verified_user', color: 'text-outline' },
|
|
15
|
+
auto: { icon: 'bolt', color: 'text-outline' },
|
|
16
|
+
plan: { icon: 'architecture', color: 'text-tertiary' },
|
|
17
|
+
bypass: { icon: 'gavel', color: 'text-error' },
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const modelIcons: Record<string, string> = {
|
|
21
|
+
opus: 'psychology',
|
|
22
|
+
sonnet: 'smart_toy',
|
|
23
|
+
haiku: 'auto_awesome',
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/* ════════════════════════════════════════════════════════════════════
|
|
27
|
+
SessionControls — full-page component (sidebar + header + chat +
|
|
28
|
+
two open dropdown panels + composer + footer)
|
|
29
|
+
════════════════════════════════════════════════════════════════════ */
|
|
30
|
+
export default function SessionControls() {
|
|
31
|
+
const [selectedPermission, setSelectedPermission] = useState('ask')
|
|
32
|
+
const [selectedModel, setSelectedModel] = useState('sonnet')
|
|
33
|
+
const [selectedEffort, setSelectedEffort] = useState('Medium')
|
|
34
|
+
const [showPermissions, setShowPermissions] = useState(true)
|
|
35
|
+
const [showModelConfig, setShowModelConfig] = useState(true)
|
|
36
|
+
|
|
37
|
+
const activeModel = mockModels.find((m) => m.id === selectedModel)
|
|
38
|
+
|
|
39
|
+
return (
|
|
40
|
+
<div className="h-screen w-screen bg-background text-on-surface font-body selection:bg-primary-fixed overflow-hidden relative">
|
|
41
|
+
{/* ─── TopAppBar ─────────────────────────────────────────── */}
|
|
42
|
+
<header className="bg-[#FAF9F5] font-headline font-semibold tracking-wide text-sm fixed top-0 left-0 right-0 flex justify-between items-center px-6 h-12 z-40">
|
|
43
|
+
<div className="flex items-center gap-6">
|
|
44
|
+
<span className="text-sm font-bold text-[#1B1C1A] uppercase tracking-tighter">
|
|
45
|
+
Claude Code Companion
|
|
46
|
+
</span>
|
|
47
|
+
<nav className="hidden md:flex gap-4">
|
|
48
|
+
<a className="text-[#1B1C1A] border-b-2 border-[#8F482F] pb-1 cursor-pointer active:opacity-70">
|
|
49
|
+
Code
|
|
50
|
+
</a>
|
|
51
|
+
<a className="text-[#87736D] hover:text-[#8F482F] transition-colors cursor-pointer active:opacity-70">
|
|
52
|
+
Terminal
|
|
53
|
+
</a>
|
|
54
|
+
<a className="text-[#87736D] hover:text-[#8F482F] transition-colors cursor-pointer active:opacity-70">
|
|
55
|
+
History
|
|
56
|
+
</a>
|
|
57
|
+
</nav>
|
|
58
|
+
</div>
|
|
59
|
+
<div className="flex items-center gap-3">
|
|
60
|
+
<span className="material-symbols-outlined text-[#87736D] cursor-pointer">
|
|
61
|
+
arrow_back_ios
|
|
62
|
+
</span>
|
|
63
|
+
<span className="material-symbols-outlined text-[#87736D] cursor-pointer">
|
|
64
|
+
arrow_forward_ios
|
|
65
|
+
</span>
|
|
66
|
+
<button className="ml-2 px-3 py-1 bg-surface-container-high rounded text-[#8F482F] hover:bg-surface-container-highest transition-colors">
|
|
67
|
+
Settings
|
|
68
|
+
</button>
|
|
69
|
+
</div>
|
|
70
|
+
</header>
|
|
71
|
+
|
|
72
|
+
{/* Separator line */}
|
|
73
|
+
<div className="bg-[#F4F4F0] h-[1px] w-full fixed top-12 z-40" />
|
|
74
|
+
|
|
75
|
+
{/* ─── SideNavBar ────────────────────────────────────────── */}
|
|
76
|
+
<aside className="bg-[#F4F4F0] font-body text-sm font-medium fixed left-0 top-0 h-full w-[280px] hidden md:flex flex-col p-4 gap-2 pt-16 z-30">
|
|
77
|
+
{/* Project header */}
|
|
78
|
+
<div className="px-2 mb-4">
|
|
79
|
+
<div className="flex items-center gap-3 mb-1">
|
|
80
|
+
<img
|
|
81
|
+
className="w-8 h-8 rounded-full"
|
|
82
|
+
src="https://lh3.googleusercontent.com/aida-public/AB6AXuCjFls9oDx5jX7Zv8P7BA9QbodBzvDJFhVNIiVjAhp_OhnXT6lmE-uYCDDZvNS4kWHssfxAYuiH05KsXLBWgLd4K-8prrjodVjSsKAG1LhvKWN90nyVzDBSrreWkpW7reNC1N_T4J_Pdr9mgAYVwYRS10nvUMZs_ajpTg2CoTtMkQRRGZGZXLk_gU94EoaeDEPNbvwaxOeeTeGgOxwnzcPIUn6EFzqc5Bjug00IDIrhRYiuwEaGNkTuz39mNFxJl2bKiHES5HxUM60"
|
|
83
|
+
alt="project avatar"
|
|
84
|
+
/>
|
|
85
|
+
<div>
|
|
86
|
+
<h3 className="text-on-surface font-bold leading-none">All projects</h3>
|
|
87
|
+
<p className="text-[10px] text-outline uppercase tracking-widest mt-1">
|
|
88
|
+
Active Session
|
|
89
|
+
</p>
|
|
90
|
+
</div>
|
|
91
|
+
</div>
|
|
92
|
+
</div>
|
|
93
|
+
|
|
94
|
+
{/* Nav items */}
|
|
95
|
+
<button className="w-full text-left p-2.5 bg-[#FAF9F5] text-[#1B1C1A] rounded-lg relative before:content-[''] before:absolute before:left-[-8px] before:w-1 before:h-4 before:bg-[#8F482F] before:rounded-full before:top-1/2 before:-translate-y-1/2 transition-all duration-200 ease-in-out flex items-center gap-3">
|
|
96
|
+
<span className="material-symbols-outlined text-[#8F482F]">add</span>
|
|
97
|
+
<span>New session</span>
|
|
98
|
+
</button>
|
|
99
|
+
<button className="w-full text-left p-2.5 text-[#87736D] hover:bg-[#EBEBE6] rounded-lg transition-all duration-200 ease-in-out flex items-center gap-3">
|
|
100
|
+
<span className="material-symbols-outlined">calendar_today</span>
|
|
101
|
+
<span>Scheduled</span>
|
|
102
|
+
</button>
|
|
103
|
+
<button className="w-full text-left p-2.5 text-[#87736D] hover:bg-[#EBEBE6] rounded-lg transition-all duration-200 ease-in-out flex items-center gap-3" data-count={mockSessions.today.length}>
|
|
104
|
+
<span className="material-symbols-outlined">history</span>
|
|
105
|
+
<span>Today</span>
|
|
106
|
+
</button>
|
|
107
|
+
<button className="w-full text-left p-2.5 text-[#87736D] hover:bg-[#EBEBE6] rounded-lg transition-all duration-200 ease-in-out flex items-center gap-3" data-count={mockSessions.previous7Days.length}>
|
|
108
|
+
<span className="material-symbols-outlined">event_note</span>
|
|
109
|
+
<span>Previous 7 Days</span>
|
|
110
|
+
</button>
|
|
111
|
+
<button className="w-full text-left p-2.5 text-[#87736D] hover:bg-[#EBEBE6] rounded-lg transition-all duration-200 ease-in-out flex items-center gap-3" data-count={mockSessions.older.length}>
|
|
112
|
+
<span className="material-symbols-outlined">archive</span>
|
|
113
|
+
<span>Older</span>
|
|
114
|
+
</button>
|
|
115
|
+
|
|
116
|
+
{/* Bottom modes */}
|
|
117
|
+
<div className="mt-auto pt-4 border-t border-outline/10 flex flex-col gap-1">
|
|
118
|
+
<button className="flex items-center gap-3 p-2 text-outline hover:text-primary transition-colors">
|
|
119
|
+
<span className="material-symbols-outlined text-xs">computer</span>
|
|
120
|
+
<span className="text-xs">Local Mode</span>
|
|
121
|
+
</button>
|
|
122
|
+
<button className="flex items-center gap-3 p-2 text-outline hover:text-primary transition-colors">
|
|
123
|
+
<span className="material-symbols-outlined text-xs">cloud</span>
|
|
124
|
+
<span className="text-xs">Remote Mode</span>
|
|
125
|
+
</button>
|
|
126
|
+
</div>
|
|
127
|
+
</aside>
|
|
128
|
+
|
|
129
|
+
{/* ─── Main Content Area (blurred / dimmed behind overlays) ── */}
|
|
130
|
+
<main className="md:ml-[280px] pt-12 pb-8 min-h-screen blur-[2px] opacity-60">
|
|
131
|
+
<div className="max-w-4xl mx-auto px-8 py-12">
|
|
132
|
+
<div className="grid grid-cols-12 gap-6">
|
|
133
|
+
{/* Main Thread */}
|
|
134
|
+
<div className="col-span-8 space-y-8">
|
|
135
|
+
{/* AI message */}
|
|
136
|
+
<div className="bg-surface-container-low rounded-xl p-6">
|
|
137
|
+
<div className="flex items-center gap-3 mb-4">
|
|
138
|
+
<div className="w-8 h-8 rounded-full bg-primary flex items-center justify-center text-on-primary font-bold text-xs">
|
|
139
|
+
AI
|
|
140
|
+
</div>
|
|
141
|
+
<span className="font-semibold text-sm">Claude 4.6 Sonnet</span>
|
|
142
|
+
</div>
|
|
143
|
+
<p className="text-on-surface-variant leading-relaxed">
|
|
144
|
+
I've analyzed the{' '}
|
|
145
|
+
<code className="bg-surface-dim px-1.5 py-0.5 rounded font-mono text-sm">
|
|
146
|
+
auth_provider.go
|
|
147
|
+
</code>{' '}
|
|
148
|
+
file. The race condition occurs during the token refresh cycle. I recommend
|
|
149
|
+
wrapping the session update in a mutex lock.
|
|
150
|
+
</p>
|
|
151
|
+
</div>
|
|
152
|
+
|
|
153
|
+
{/* User message */}
|
|
154
|
+
<div className="flex justify-end">
|
|
155
|
+
<div className="bg-surface-container-highest rounded-xl p-6 max-w-[80%]">
|
|
156
|
+
<p className="text-on-surface leading-relaxed">
|
|
157
|
+
Can you implement that? Also check if this affects the WebSocket connection
|
|
158
|
+
longevity.
|
|
159
|
+
</p>
|
|
160
|
+
</div>
|
|
161
|
+
</div>
|
|
162
|
+
|
|
163
|
+
{/* Code Block Preview */}
|
|
164
|
+
<div className="bg-surface-dim rounded-lg overflow-hidden font-mono text-sm">
|
|
165
|
+
<div className="bg-surface-container-high px-4 py-2 flex justify-between items-center">
|
|
166
|
+
<span className="text-xs text-on-surface-variant">
|
|
167
|
+
internal/auth/provider.go
|
|
168
|
+
</span>
|
|
169
|
+
<span className="material-symbols-outlined text-sm text-outline cursor-pointer">
|
|
170
|
+
content_copy
|
|
171
|
+
</span>
|
|
172
|
+
</div>
|
|
173
|
+
<pre className="p-4 text-on-surface">
|
|
174
|
+
<code>{`func (p *Provider) RefreshToken(ctx context.Context) error {
|
|
175
|
+
p.mu.Lock()
|
|
176
|
+
defer p.mu.Unlock()
|
|
177
|
+
|
|
178
|
+
// logic to refresh token...
|
|
179
|
+
return nil
|
|
180
|
+
}`}</code>
|
|
181
|
+
</pre>
|
|
182
|
+
</div>
|
|
183
|
+
</div>
|
|
184
|
+
|
|
185
|
+
{/* Session Meta */}
|
|
186
|
+
<div className="col-span-4 space-y-6">
|
|
187
|
+
<div className="bg-surface-container-lowest border border-outline-variant/20 rounded-xl p-4">
|
|
188
|
+
<h4 className="text-xs font-bold uppercase tracking-widest text-outline mb-4">
|
|
189
|
+
Context Files
|
|
190
|
+
</h4>
|
|
191
|
+
<ul className="space-y-3">
|
|
192
|
+
<li className="flex items-center gap-2 text-sm text-on-surface-variant">
|
|
193
|
+
<span className="material-symbols-outlined text-sm">description</span>
|
|
194
|
+
auth_provider.go
|
|
195
|
+
</li>
|
|
196
|
+
<li className="flex items-center gap-2 text-sm text-on-surface-variant">
|
|
197
|
+
<span className="material-symbols-outlined text-sm">description</span>
|
|
198
|
+
main.go
|
|
199
|
+
</li>
|
|
200
|
+
<li className="flex items-center gap-2 text-sm text-on-surface-variant">
|
|
201
|
+
<span className="material-symbols-outlined text-sm">description</span>
|
|
202
|
+
session_test.go
|
|
203
|
+
</li>
|
|
204
|
+
</ul>
|
|
205
|
+
</div>
|
|
206
|
+
</div>
|
|
207
|
+
</div>
|
|
208
|
+
</div>
|
|
209
|
+
</main>
|
|
210
|
+
|
|
211
|
+
{/* ─── Overlay Layer (Active Selectors) ──────────────────── */}
|
|
212
|
+
<div className="fixed inset-0 z-50 flex flex-col justify-end items-center pointer-events-none p-8">
|
|
213
|
+
{/* Floating Dropdown Overlay Container */}
|
|
214
|
+
<div className="w-full max-w-2xl flex gap-4 mb-4 pointer-events-auto items-end">
|
|
215
|
+
{/* ── Permissions Dropdown ─────────────────────────────── */}
|
|
216
|
+
{showPermissions && (
|
|
217
|
+
<div
|
|
218
|
+
className="w-80 rounded-xl border border-[#E8E6DC] overflow-hidden flex flex-col"
|
|
219
|
+
style={{
|
|
220
|
+
background: 'rgba(255, 255, 255, 0.85)',
|
|
221
|
+
backdropFilter: 'blur(20px)',
|
|
222
|
+
boxShadow:
|
|
223
|
+
'0 4px 20px rgba(27, 28, 26, 0.04), 0 12px 40px rgba(27, 28, 26, 0.08)',
|
|
224
|
+
}}
|
|
225
|
+
>
|
|
226
|
+
<div className="px-4 py-3 bg-surface-container-low border-b border-[#E8E6DC]">
|
|
227
|
+
<span className="text-[10px] font-bold uppercase tracking-widest text-outline">
|
|
228
|
+
Execution Permissions
|
|
229
|
+
</span>
|
|
230
|
+
</div>
|
|
231
|
+
<div className="p-2 space-y-1">
|
|
232
|
+
{mockPermissionModes.map((mode) => {
|
|
233
|
+
const meta = permissionIcons[mode.id] || {
|
|
234
|
+
icon: mode.icon,
|
|
235
|
+
color: 'text-outline',
|
|
236
|
+
}
|
|
237
|
+
const isSelected = selectedPermission === mode.id
|
|
238
|
+
const isPlan = mode.id === 'plan'
|
|
239
|
+
const isBypass = mode.id === 'bypass'
|
|
240
|
+
|
|
241
|
+
let hoverClass = 'hover:bg-surface-container-high'
|
|
242
|
+
if (isPlan) hoverClass = 'hover:bg-tertiary-container/10'
|
|
243
|
+
if (isBypass) hoverClass = 'hover:bg-error-container/20'
|
|
244
|
+
|
|
245
|
+
let labelColor = ''
|
|
246
|
+
if (isPlan) labelColor = 'text-tertiary'
|
|
247
|
+
if (isBypass) labelColor = 'text-error'
|
|
248
|
+
|
|
249
|
+
return (
|
|
250
|
+
<button
|
|
251
|
+
key={mode.id}
|
|
252
|
+
onClick={() => setSelectedPermission(mode.id)}
|
|
253
|
+
className={`w-full text-left p-3 rounded-lg ${hoverClass} transition-colors flex gap-3 group`}
|
|
254
|
+
>
|
|
255
|
+
<span
|
|
256
|
+
className={`material-symbols-outlined ${meta.color} mt-0.5`}
|
|
257
|
+
>
|
|
258
|
+
{meta.icon}
|
|
259
|
+
</span>
|
|
260
|
+
<div className="flex-1">
|
|
261
|
+
<div className="flex items-center justify-between">
|
|
262
|
+
<span className={`text-sm font-semibold ${labelColor}`}>
|
|
263
|
+
{mode.label}
|
|
264
|
+
</span>
|
|
265
|
+
{isSelected && (
|
|
266
|
+
<span className="material-symbols-outlined text-primary text-sm">
|
|
267
|
+
check_circle
|
|
268
|
+
</span>
|
|
269
|
+
)}
|
|
270
|
+
</div>
|
|
271
|
+
<p className="text-xs text-on-surface-variant">{mode.description}</p>
|
|
272
|
+
</div>
|
|
273
|
+
</button>
|
|
274
|
+
)
|
|
275
|
+
})}
|
|
276
|
+
</div>
|
|
277
|
+
</div>
|
|
278
|
+
)}
|
|
279
|
+
|
|
280
|
+
{/* ── Model & Effort Dropdown ─────────────────────────── */}
|
|
281
|
+
{showModelConfig && (
|
|
282
|
+
<div
|
|
283
|
+
className="w-64 rounded-xl border border-[#E8E6DC] overflow-hidden flex flex-col"
|
|
284
|
+
style={{
|
|
285
|
+
background: 'rgba(255, 255, 255, 0.85)',
|
|
286
|
+
backdropFilter: 'blur(20px)',
|
|
287
|
+
boxShadow:
|
|
288
|
+
'0 4px 20px rgba(27, 28, 26, 0.04), 0 12px 40px rgba(27, 28, 26, 0.08)',
|
|
289
|
+
}}
|
|
290
|
+
>
|
|
291
|
+
<div className="px-4 py-3 bg-surface-container-low border-b border-[#E8E6DC]">
|
|
292
|
+
<span className="text-[10px] font-bold uppercase tracking-widest text-outline">
|
|
293
|
+
Model Configuration
|
|
294
|
+
</span>
|
|
295
|
+
</div>
|
|
296
|
+
|
|
297
|
+
{/* Models */}
|
|
298
|
+
<div className="p-2">
|
|
299
|
+
{mockModels.map((model) => {
|
|
300
|
+
const isActive = selectedModel === model.id
|
|
301
|
+
const icon = modelIcons[model.id] || 'smart_toy'
|
|
302
|
+
|
|
303
|
+
return (
|
|
304
|
+
<button
|
|
305
|
+
key={model.id}
|
|
306
|
+
onClick={() => setSelectedModel(model.id)}
|
|
307
|
+
className={`w-full text-left p-2.5 rounded-lg flex items-center justify-between transition-colors ${
|
|
308
|
+
isActive
|
|
309
|
+
? 'bg-primary/5 text-primary'
|
|
310
|
+
: 'hover:bg-surface-container-high'
|
|
311
|
+
}`}
|
|
312
|
+
>
|
|
313
|
+
<div
|
|
314
|
+
className={`flex items-center gap-2 ${
|
|
315
|
+
isActive ? 'font-semibold' : ''
|
|
316
|
+
}`}
|
|
317
|
+
>
|
|
318
|
+
<span
|
|
319
|
+
className={`material-symbols-outlined text-sm ${
|
|
320
|
+
isActive ? '' : 'text-outline'
|
|
321
|
+
}`}
|
|
322
|
+
>
|
|
323
|
+
{icon}
|
|
324
|
+
</span>
|
|
325
|
+
<span className="text-sm">{model.name}</span>
|
|
326
|
+
</div>
|
|
327
|
+
{isActive && (
|
|
328
|
+
<span className="material-symbols-outlined text-sm">
|
|
329
|
+
radio_button_checked
|
|
330
|
+
</span>
|
|
331
|
+
)}
|
|
332
|
+
</button>
|
|
333
|
+
)
|
|
334
|
+
})}
|
|
335
|
+
</div>
|
|
336
|
+
|
|
337
|
+
{/* Divider */}
|
|
338
|
+
<div className="mx-4 h-[1px] bg-[#E8E6DC]" />
|
|
339
|
+
|
|
340
|
+
{/* Effort levels */}
|
|
341
|
+
<div className="p-2">
|
|
342
|
+
<div className="px-2 mb-2">
|
|
343
|
+
<span className="text-[9px] font-bold text-outline uppercase tracking-tighter">
|
|
344
|
+
Thinking Effort
|
|
345
|
+
</span>
|
|
346
|
+
</div>
|
|
347
|
+
<div className="grid grid-cols-2 gap-1">
|
|
348
|
+
{mockEffortLevels.map((level) => {
|
|
349
|
+
const isActive = selectedEffort === level
|
|
350
|
+
/* The HTML uses "Med" for the label display of "Medium" */
|
|
351
|
+
const displayLabel =
|
|
352
|
+
level === 'Medium' ? 'Med' : level
|
|
353
|
+
|
|
354
|
+
return (
|
|
355
|
+
<button
|
|
356
|
+
key={level}
|
|
357
|
+
onClick={() => setSelectedEffort(level)}
|
|
358
|
+
className={`text-xs py-2 px-3 rounded-md transition-all ${
|
|
359
|
+
isActive
|
|
360
|
+
? 'bg-primary text-on-primary'
|
|
361
|
+
: 'border border-outline-variant/30 hover:border-primary'
|
|
362
|
+
} ${level === 'Max' ? 'font-bold' : ''}`}
|
|
363
|
+
>
|
|
364
|
+
{displayLabel}
|
|
365
|
+
</button>
|
|
366
|
+
)
|
|
367
|
+
})}
|
|
368
|
+
</div>
|
|
369
|
+
</div>
|
|
370
|
+
</div>
|
|
371
|
+
)}
|
|
372
|
+
</div>
|
|
373
|
+
|
|
374
|
+
{/* ── The Composer (Anchor) ─────────────────────────────── */}
|
|
375
|
+
<div
|
|
376
|
+
className="w-full max-w-2xl p-4 rounded-xl border border-outline-variant/15 pointer-events-auto flex flex-col gap-3"
|
|
377
|
+
style={{
|
|
378
|
+
background: 'rgba(255, 255, 255, 0.85)',
|
|
379
|
+
backdropFilter: 'blur(20px)',
|
|
380
|
+
boxShadow:
|
|
381
|
+
'0 4px 20px rgba(27, 28, 26, 0.04), 0 12px 40px rgba(27, 28, 26, 0.08)',
|
|
382
|
+
}}
|
|
383
|
+
>
|
|
384
|
+
<textarea
|
|
385
|
+
className="w-full bg-transparent border-none focus:ring-0 focus:outline-none resize-none font-body text-on-surface placeholder:text-outline"
|
|
386
|
+
placeholder="Reply to Claude..."
|
|
387
|
+
rows={2}
|
|
388
|
+
/>
|
|
389
|
+
<div className="flex justify-between items-center">
|
|
390
|
+
<div className="flex items-center gap-2">
|
|
391
|
+
{/* Permission pill */}
|
|
392
|
+
<button
|
|
393
|
+
onClick={() => setShowPermissions((v) => !v)}
|
|
394
|
+
className="flex items-center gap-1.5 px-2.5 py-1.5 rounded-lg bg-surface-container-low border border-[#E8E6DC] text-xs font-medium hover:bg-surface-container-high transition-all"
|
|
395
|
+
>
|
|
396
|
+
<span className="material-symbols-outlined text-base">
|
|
397
|
+
{permissionIcons[selectedPermission]?.icon || 'verified_user'}
|
|
398
|
+
</span>
|
|
399
|
+
{mockPermissionModes.find((m) => m.id === selectedPermission)?.label ||
|
|
400
|
+
'Ask permissions'}
|
|
401
|
+
</button>
|
|
402
|
+
|
|
403
|
+
{/* Model pill */}
|
|
404
|
+
<button
|
|
405
|
+
onClick={() => setShowModelConfig((v) => !v)}
|
|
406
|
+
className="flex items-center gap-1.5 px-2.5 py-1.5 rounded-lg bg-surface-container-low border border-[#E8E6DC] text-xs font-medium hover:bg-surface-container-high transition-all"
|
|
407
|
+
>
|
|
408
|
+
<span className="material-symbols-outlined text-base">
|
|
409
|
+
{modelIcons[selectedModel] || 'smart_toy'}
|
|
410
|
+
</span>
|
|
411
|
+
{activeModel?.name || 'Sonnet 4.6'}
|
|
412
|
+
</button>
|
|
413
|
+
|
|
414
|
+
{/* Attach file button */}
|
|
415
|
+
<button className="p-1.5 rounded-lg text-outline hover:bg-surface-container-low transition-colors">
|
|
416
|
+
<span className="material-symbols-outlined">attach_file</span>
|
|
417
|
+
</button>
|
|
418
|
+
</div>
|
|
419
|
+
|
|
420
|
+
{/* Run button */}
|
|
421
|
+
<button className="bg-primary text-on-primary px-4 py-1.5 rounded-lg font-semibold text-sm flex items-center gap-2 hover:opacity-90 transition-opacity">
|
|
422
|
+
Run
|
|
423
|
+
<span className="material-symbols-outlined text-base">send</span>
|
|
424
|
+
</button>
|
|
425
|
+
</div>
|
|
426
|
+
</div>
|
|
427
|
+
</div>
|
|
428
|
+
|
|
429
|
+
{/* ─── Footer / Status Bar ───────────────────────────────── */}
|
|
430
|
+
<footer className="bg-[#FAF9F5] font-body text-xs tracking-tight fixed bottom-0 left-0 w-full h-8 border-t border-[#87736D]/20 flex items-center justify-between px-4 z-[60]">
|
|
431
|
+
<div className="flex items-center gap-3">
|
|
432
|
+
<div className="flex items-center gap-1.5">
|
|
433
|
+
<div className="w-4 h-4 rounded-full bg-primary-fixed flex items-center justify-center">
|
|
434
|
+
<span
|
|
435
|
+
className="material-symbols-outlined text-[10px] text-on-primary-fixed"
|
|
436
|
+
style={{ fontVariationSettings: "'FILL' 1" }}
|
|
437
|
+
>
|
|
438
|
+
person
|
|
439
|
+
</span>
|
|
440
|
+
</div>
|
|
441
|
+
<span className="text-outline">
|
|
442
|
+
{mockStatusBar.user} • {mockStatusBar.username} • {mockStatusBar.plan}
|
|
443
|
+
</span>
|
|
444
|
+
</div>
|
|
445
|
+
</div>
|
|
446
|
+
<div className="flex items-center gap-4">
|
|
447
|
+
<button className="text-primary font-bold hover:bg-[#F4F4F0] transition-colors px-2 py-0.5 rounded">
|
|
448
|
+
{mockStatusBar.branch}
|
|
449
|
+
</button>
|
|
450
|
+
<button className="text-[#87736D] hover:bg-[#F4F4F0] transition-colors px-2 py-0.5 rounded">
|
|
451
|
+
{mockStatusBar.worktreeToggle}
|
|
452
|
+
</button>
|
|
453
|
+
<button className="text-[#87736D] hover:bg-[#F4F4F0] transition-colors px-2 py-0.5 rounded">
|
|
454
|
+
{mockStatusBar.localSwitch}
|
|
455
|
+
</button>
|
|
456
|
+
</div>
|
|
457
|
+
</footer>
|
|
458
|
+
</div>
|
|
459
|
+
)
|
|
460
|
+
}
|