prompt_objects 0.3.1 → 0.5.0
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +32 -0
- data/CLAUDE.md +112 -44
- data/Gemfile.lock +31 -29
- data/README.md +5 -0
- data/frontend/index.html +5 -1
- data/frontend/package-lock.json +123 -0
- data/frontend/package.json +4 -0
- data/frontend/src/App.tsx +70 -71
- data/frontend/src/canvas/CanvasView.tsx +113 -0
- data/frontend/src/canvas/ForceLayout.ts +115 -0
- data/frontend/src/canvas/SceneManager.ts +587 -0
- data/frontend/src/canvas/canvasStore.ts +47 -0
- data/frontend/src/canvas/constants.ts +95 -0
- data/frontend/src/canvas/controls/CameraControls.ts +98 -0
- data/frontend/src/canvas/edges/MessageArc.ts +149 -0
- data/frontend/src/canvas/inspector/InspectorPanel.tsx +31 -0
- data/frontend/src/canvas/inspector/POInspector.tsx +262 -0
- data/frontend/src/canvas/inspector/ToolCallInspector.tsx +67 -0
- data/frontend/src/canvas/nodes/PONode.ts +249 -0
- data/frontend/src/canvas/nodes/ToolCallNode.ts +116 -0
- data/frontend/src/canvas/types.ts +24 -0
- data/frontend/src/components/ContextMenu.tsx +5 -4
- data/frontend/src/components/Inspector.tsx +232 -0
- data/frontend/src/components/MarkdownMessage.tsx +22 -20
- data/frontend/src/components/MethodList.tsx +90 -0
- data/frontend/src/components/ModelSelector.tsx +13 -14
- data/frontend/src/components/NotificationPanel.tsx +29 -33
- data/frontend/src/components/ObjectList.tsx +78 -0
- data/frontend/src/components/PaneSlot.tsx +30 -0
- data/frontend/src/components/SourcePane.tsx +202 -0
- data/frontend/src/components/SystemBar.tsx +74 -0
- data/frontend/src/components/Transcript.tsx +76 -0
- data/frontend/src/components/UsagePanel.tsx +27 -27
- data/frontend/src/components/Workspace.tsx +260 -0
- data/frontend/src/components/index.ts +10 -9
- data/frontend/src/hooks/useResize.ts +55 -0
- data/frontend/src/hooks/useWebSocket.ts +274 -189
- data/frontend/src/index.css +69 -3
- data/frontend/src/store/index.ts +23 -0
- data/frontend/src/types/index.ts +5 -0
- data/frontend/tailwind.config.js +28 -9
- data/lib/prompt_objects/capability.rb +23 -1
- data/lib/prompt_objects/connectors/mcp.rb +5 -22
- data/lib/prompt_objects/environment.rb +8 -0
- data/lib/prompt_objects/llm/openai_adapter.rb +22 -0
- data/lib/prompt_objects/mcp/tools/get_pending_requests.rb +1 -2
- data/lib/prompt_objects/mcp/tools/inspect_po.rb +1 -31
- data/lib/prompt_objects/mcp/tools/list_prompt_objects.rb +2 -8
- data/lib/prompt_objects/primitives/list_files.rb +1 -2
- data/lib/prompt_objects/prompt_object.rb +150 -6
- data/lib/prompt_objects/server/api/routes.rb +3 -48
- data/lib/prompt_objects/server/app.rb +9 -0
- data/lib/prompt_objects/server/public/assets/index-D1myxE0l.js +4345 -0
- data/lib/prompt_objects/server/public/assets/index-DdCcwC-Z.css +1 -0
- data/lib/prompt_objects/server/public/index.html +7 -3
- data/lib/prompt_objects/server/websocket_handler.rb +23 -100
- data/lib/prompt_objects/server.rb +6 -62
- data/prompt_objects.gemspec +1 -1
- data/templates/arc-agi-1/primitives/find_objects.rb +1 -1
- data/templates/arc-agi-1/primitives/grid_diff.rb +2 -2
- data/templates/arc-agi-1/primitives/grid_info.rb +1 -1
- data/templates/arc-agi-1/primitives/grid_transform.rb +1 -1
- data/templates/arc-agi-1/primitives/render_grid.rb +1 -0
- data/templates/arc-agi-1/primitives/test_solution.rb +3 -0
- metadata +26 -14
- data/frontend/src/components/CapabilitiesPanel.tsx +0 -141
- data/frontend/src/components/ChatPanel.tsx +0 -288
- data/frontend/src/components/Dashboard.tsx +0 -83
- data/frontend/src/components/Header.tsx +0 -141
- data/frontend/src/components/MessageBus.tsx +0 -56
- data/frontend/src/components/POCard.tsx +0 -56
- data/frontend/src/components/PODetail.tsx +0 -124
- data/frontend/src/components/PromptPanel.tsx +0 -156
- data/frontend/src/components/SessionsPanel.tsx +0 -174
- data/frontend/src/components/ThreadsSidebar.tsx +0 -163
- data/lib/prompt_objects/server/public/assets/index-Bkme6COu.css +0 -1
- data/lib/prompt_objects/server/public/assets/index-CQ7lVDF_.js +0 -77
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
import { useState, useEffect, useCallback, useRef } from 'react'
|
|
2
|
+
import type { PromptObject, CapabilityInfo } from '../types'
|
|
3
|
+
|
|
4
|
+
interface SourcePaneProps {
|
|
5
|
+
po: PromptObject
|
|
6
|
+
selectedCapability: CapabilityInfo | null
|
|
7
|
+
onSave?: (prompt: string) => void
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export function SourcePane({ po, selectedCapability, onSave }: SourcePaneProps) {
|
|
11
|
+
const prompt = po.prompt || ''
|
|
12
|
+
const config = po.config || {}
|
|
13
|
+
const [isEditing, setIsEditing] = useState(false)
|
|
14
|
+
const [editedPrompt, setEditedPrompt] = useState(prompt)
|
|
15
|
+
const [saveStatus, setSaveStatus] = useState<'saved' | 'saving' | 'unsaved'>('saved')
|
|
16
|
+
const saveTimeoutRef = useRef<number | null>(null)
|
|
17
|
+
|
|
18
|
+
// Sync editedPrompt when po.prompt changes from server
|
|
19
|
+
useEffect(() => {
|
|
20
|
+
if (!isEditing) {
|
|
21
|
+
setEditedPrompt(prompt)
|
|
22
|
+
}
|
|
23
|
+
}, [prompt, isEditing])
|
|
24
|
+
|
|
25
|
+
// Debounced auto-save
|
|
26
|
+
const debouncedSave = useCallback((newPrompt: string) => {
|
|
27
|
+
if (saveTimeoutRef.current) {
|
|
28
|
+
clearTimeout(saveTimeoutRef.current)
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
setSaveStatus('unsaved')
|
|
32
|
+
|
|
33
|
+
saveTimeoutRef.current = window.setTimeout(() => {
|
|
34
|
+
if (onSave && newPrompt !== prompt) {
|
|
35
|
+
setSaveStatus('saving')
|
|
36
|
+
onSave(newPrompt)
|
|
37
|
+
setTimeout(() => setSaveStatus('saved'), 500)
|
|
38
|
+
} else {
|
|
39
|
+
setSaveStatus('saved')
|
|
40
|
+
}
|
|
41
|
+
}, 1000)
|
|
42
|
+
}, [onSave, prompt])
|
|
43
|
+
|
|
44
|
+
const handlePromptChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {
|
|
45
|
+
const newPrompt = e.target.value
|
|
46
|
+
setEditedPrompt(newPrompt)
|
|
47
|
+
debouncedSave(newPrompt)
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const handleToggleEdit = () => {
|
|
51
|
+
if (isEditing) {
|
|
52
|
+
// Save on exit
|
|
53
|
+
if (saveTimeoutRef.current) {
|
|
54
|
+
clearTimeout(saveTimeoutRef.current)
|
|
55
|
+
}
|
|
56
|
+
if (editedPrompt !== prompt && onSave) {
|
|
57
|
+
onSave(editedPrompt)
|
|
58
|
+
}
|
|
59
|
+
setIsEditing(false)
|
|
60
|
+
setSaveStatus('saved')
|
|
61
|
+
} else {
|
|
62
|
+
setEditedPrompt(prompt)
|
|
63
|
+
setIsEditing(true)
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Cleanup timeout on unmount
|
|
68
|
+
useEffect(() => {
|
|
69
|
+
return () => {
|
|
70
|
+
if (saveTimeoutRef.current) {
|
|
71
|
+
clearTimeout(saveTimeoutRef.current)
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}, [])
|
|
75
|
+
|
|
76
|
+
// Capability detail view
|
|
77
|
+
if (selectedCapability) {
|
|
78
|
+
return (
|
|
79
|
+
<div className="flex-1 overflow-auto bg-po-bg">
|
|
80
|
+
<div className="px-3 py-2 border-b border-po-border bg-po-surface">
|
|
81
|
+
<span className="font-mono text-xs text-po-accent">{selectedCapability.name}</span>
|
|
82
|
+
</div>
|
|
83
|
+
<div className="p-3 space-y-3">
|
|
84
|
+
<p className="text-xs text-po-text-secondary">{selectedCapability.description}</p>
|
|
85
|
+
{selectedCapability.parameters && (
|
|
86
|
+
<ParametersView parameters={selectedCapability.parameters} />
|
|
87
|
+
)}
|
|
88
|
+
</div>
|
|
89
|
+
</div>
|
|
90
|
+
)
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// Prompt source view
|
|
94
|
+
return (
|
|
95
|
+
<div className="flex-1 overflow-auto bg-po-bg flex flex-col">
|
|
96
|
+
{/* Header with edit toggle and save status */}
|
|
97
|
+
<div className="px-3 py-1.5 border-b border-po-border bg-po-surface flex items-center gap-2 flex-shrink-0">
|
|
98
|
+
<span className="text-2xs text-po-text-ghost uppercase tracking-wider flex-1">Source</span>
|
|
99
|
+
|
|
100
|
+
{isEditing && (
|
|
101
|
+
<span className="flex items-center gap-1">
|
|
102
|
+
<span className={`w-1.5 h-1.5 rounded-full ${
|
|
103
|
+
saveStatus === 'saved' ? 'bg-po-success' :
|
|
104
|
+
saveStatus === 'saving' ? 'bg-po-accent animate-pulse' :
|
|
105
|
+
'bg-po-text-ghost'
|
|
106
|
+
}`} />
|
|
107
|
+
<span className={`text-2xs ${
|
|
108
|
+
saveStatus === 'saved' ? 'text-po-success' :
|
|
109
|
+
saveStatus === 'saving' ? 'text-po-accent' :
|
|
110
|
+
'text-po-text-ghost'
|
|
111
|
+
}`}>
|
|
112
|
+
{saveStatus === 'saved' ? 'saved' : saveStatus === 'saving' ? 'saving' : 'unsaved'}
|
|
113
|
+
</span>
|
|
114
|
+
</span>
|
|
115
|
+
)}
|
|
116
|
+
|
|
117
|
+
<button
|
|
118
|
+
onClick={handleToggleEdit}
|
|
119
|
+
className={`text-2xs px-1.5 py-0.5 rounded transition-colors duration-150 ${
|
|
120
|
+
isEditing
|
|
121
|
+
? 'bg-po-accent text-po-bg'
|
|
122
|
+
: 'text-po-text-tertiary hover:text-po-text-primary hover:bg-po-surface-2'
|
|
123
|
+
}`}
|
|
124
|
+
>
|
|
125
|
+
{isEditing ? 'Done' : 'Edit'}
|
|
126
|
+
</button>
|
|
127
|
+
</div>
|
|
128
|
+
|
|
129
|
+
{/* Config (collapsed) */}
|
|
130
|
+
{Object.keys(config).length > 0 && (
|
|
131
|
+
<details className="border-b border-po-border">
|
|
132
|
+
<summary className="px-3 py-1 text-2xs text-po-text-ghost cursor-pointer hover:text-po-text-secondary transition-colors duration-150">
|
|
133
|
+
Frontmatter
|
|
134
|
+
</summary>
|
|
135
|
+
<pre className="px-3 pb-2 text-2xs text-po-text-tertiary font-mono whitespace-pre-wrap">
|
|
136
|
+
{JSON.stringify(config, null, 2)}
|
|
137
|
+
</pre>
|
|
138
|
+
</details>
|
|
139
|
+
)}
|
|
140
|
+
|
|
141
|
+
{/* Prompt content */}
|
|
142
|
+
{isEditing ? (
|
|
143
|
+
<textarea
|
|
144
|
+
value={editedPrompt}
|
|
145
|
+
onChange={handlePromptChange}
|
|
146
|
+
className="flex-1 w-full p-3 bg-transparent text-po-text-primary font-mono text-xs resize-none focus:outline-none"
|
|
147
|
+
placeholder="Enter prompt markdown..."
|
|
148
|
+
spellCheck={false}
|
|
149
|
+
/>
|
|
150
|
+
) : (
|
|
151
|
+
<pre className="flex-1 p-3 text-xs text-po-text-secondary font-mono whitespace-pre-wrap overflow-auto">
|
|
152
|
+
{prompt || '(empty)'}
|
|
153
|
+
</pre>
|
|
154
|
+
)}
|
|
155
|
+
</div>
|
|
156
|
+
)
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
function ParametersView({ parameters }: { parameters: Record<string, unknown> }) {
|
|
160
|
+
const properties = (parameters.properties as Record<string, unknown>) || {}
|
|
161
|
+
const required = (parameters.required as string[]) || []
|
|
162
|
+
|
|
163
|
+
const propertyNames = Object.keys(properties)
|
|
164
|
+
if (propertyNames.length === 0) return null
|
|
165
|
+
|
|
166
|
+
return (
|
|
167
|
+
<div>
|
|
168
|
+
<div className="text-2xs text-po-text-ghost uppercase tracking-wider mb-1.5">Parameters</div>
|
|
169
|
+
<div className="space-y-1.5">
|
|
170
|
+
{propertyNames.map((propName) => {
|
|
171
|
+
const prop = properties[propName] as Record<string, unknown>
|
|
172
|
+
const isRequired = required.includes(propName)
|
|
173
|
+
const propType = prop.type ? String(prop.type) : null
|
|
174
|
+
const propDescription = prop.description ? String(prop.description) : null
|
|
175
|
+
const propEnum = prop.enum as string[] | undefined
|
|
176
|
+
|
|
177
|
+
return (
|
|
178
|
+
<div key={propName} className="bg-po-surface rounded p-2">
|
|
179
|
+
<div className="flex items-center gap-1.5">
|
|
180
|
+
<span className="font-mono text-2xs text-po-accent">{propName}</span>
|
|
181
|
+
{propType && <span className="text-2xs text-po-text-ghost">({propType})</span>}
|
|
182
|
+
{isRequired && <span className="text-2xs text-po-error">req</span>}
|
|
183
|
+
</div>
|
|
184
|
+
{propDescription && (
|
|
185
|
+
<p className="text-2xs text-po-text-tertiary mt-0.5">{propDescription}</p>
|
|
186
|
+
)}
|
|
187
|
+
{propEnum && propEnum.length > 0 && (
|
|
188
|
+
<div className="mt-1 flex flex-wrap gap-1">
|
|
189
|
+
{propEnum.map((val) => (
|
|
190
|
+
<span key={val} className="text-2xs bg-po-surface-2 px-1 py-0.5 rounded text-po-text-ghost">
|
|
191
|
+
{val}
|
|
192
|
+
</span>
|
|
193
|
+
))}
|
|
194
|
+
</div>
|
|
195
|
+
)}
|
|
196
|
+
</div>
|
|
197
|
+
)
|
|
198
|
+
})}
|
|
199
|
+
</div>
|
|
200
|
+
</div>
|
|
201
|
+
)
|
|
202
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { useStore, useNotificationCount } from '../store'
|
|
2
|
+
import { ModelSelector } from './ModelSelector'
|
|
3
|
+
|
|
4
|
+
interface Props {
|
|
5
|
+
switchLLM: (provider: string, model?: string) => void
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export function SystemBar({ switchLLM }: Props) {
|
|
9
|
+
const { connected, environment, toggleBus, busOpen, currentView, setCurrentView } =
|
|
10
|
+
useStore()
|
|
11
|
+
const notificationCount = useNotificationCount()
|
|
12
|
+
|
|
13
|
+
return (
|
|
14
|
+
<header className="h-8 bg-po-surface-2 border-b border-po-border flex items-center px-3 gap-3 flex-shrink-0">
|
|
15
|
+
{/* Logo / Environment */}
|
|
16
|
+
<div className="flex items-center gap-1.5 text-xs">
|
|
17
|
+
<span className="font-mono text-po-text-primary font-medium">PromptObjects</span>
|
|
18
|
+
{environment && (
|
|
19
|
+
<>
|
|
20
|
+
<span className="text-po-text-ghost">/</span>
|
|
21
|
+
<span className="text-po-text-secondary">{environment.name}</span>
|
|
22
|
+
</>
|
|
23
|
+
)}
|
|
24
|
+
</div>
|
|
25
|
+
|
|
26
|
+
<div className="flex-1" />
|
|
27
|
+
|
|
28
|
+
{/* Connection dot */}
|
|
29
|
+
<div
|
|
30
|
+
className={`w-1.5 h-1.5 rounded-full flex-shrink-0 ${
|
|
31
|
+
connected ? 'bg-po-success' : 'bg-po-error animate-pulse'
|
|
32
|
+
}`}
|
|
33
|
+
title={connected ? 'Connected' : 'Disconnected'}
|
|
34
|
+
/>
|
|
35
|
+
|
|
36
|
+
{/* Model selector */}
|
|
37
|
+
<ModelSelector switchLLM={switchLLM} />
|
|
38
|
+
|
|
39
|
+
{/* Notification count */}
|
|
40
|
+
{notificationCount > 0 && (
|
|
41
|
+
<span
|
|
42
|
+
className="text-2xs font-mono bg-po-warning text-po-bg px-1.5 py-0.5 rounded font-bold"
|
|
43
|
+
title={`${notificationCount} pending requests`}
|
|
44
|
+
>
|
|
45
|
+
{notificationCount}
|
|
46
|
+
</span>
|
|
47
|
+
)}
|
|
48
|
+
|
|
49
|
+
{/* Canvas toggle */}
|
|
50
|
+
<button
|
|
51
|
+
onClick={() => setCurrentView(currentView === 'canvas' ? 'dashboard' : 'canvas')}
|
|
52
|
+
className={`text-xs px-2 py-0.5 rounded transition-colors duration-150 ${
|
|
53
|
+
currentView === 'canvas'
|
|
54
|
+
? 'bg-po-accent text-po-bg font-medium'
|
|
55
|
+
: 'text-po-text-secondary hover:text-po-text-primary hover:bg-po-surface-3'
|
|
56
|
+
}`}
|
|
57
|
+
>
|
|
58
|
+
Canvas
|
|
59
|
+
</button>
|
|
60
|
+
|
|
61
|
+
{/* Transcript toggle */}
|
|
62
|
+
<button
|
|
63
|
+
onClick={toggleBus}
|
|
64
|
+
className={`text-xs px-2 py-0.5 rounded transition-colors duration-150 ${
|
|
65
|
+
busOpen
|
|
66
|
+
? 'bg-po-accent text-po-bg font-medium'
|
|
67
|
+
: 'text-po-text-secondary hover:text-po-text-primary hover:bg-po-surface-3'
|
|
68
|
+
}`}
|
|
69
|
+
>
|
|
70
|
+
Transcript
|
|
71
|
+
</button>
|
|
72
|
+
</header>
|
|
73
|
+
)
|
|
74
|
+
}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { useRef, useEffect, useState } from 'react'
|
|
2
|
+
import { useStore } from '../store'
|
|
3
|
+
|
|
4
|
+
export function Transcript() {
|
|
5
|
+
const { busMessages, toggleBus } = useStore()
|
|
6
|
+
const messagesEndRef = useRef<HTMLDivElement>(null)
|
|
7
|
+
|
|
8
|
+
useEffect(() => {
|
|
9
|
+
messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' })
|
|
10
|
+
}, [busMessages])
|
|
11
|
+
|
|
12
|
+
return (
|
|
13
|
+
<div className="h-full flex flex-col border-t border-po-border">
|
|
14
|
+
{/* Header */}
|
|
15
|
+
<div className="h-6 bg-po-surface-2 border-b border-po-border flex items-center px-3 flex-shrink-0">
|
|
16
|
+
<span className="text-2xs font-medium text-po-text-ghost uppercase tracking-wider flex-1">
|
|
17
|
+
Transcript
|
|
18
|
+
</span>
|
|
19
|
+
<button
|
|
20
|
+
onClick={toggleBus}
|
|
21
|
+
className="text-2xs text-po-text-ghost hover:text-po-text-secondary transition-colors duration-150"
|
|
22
|
+
>
|
|
23
|
+
{'\u2715'}
|
|
24
|
+
</button>
|
|
25
|
+
</div>
|
|
26
|
+
|
|
27
|
+
{/* Messages */}
|
|
28
|
+
<div className="flex-1 overflow-auto px-3 py-1 font-mono">
|
|
29
|
+
{busMessages.length === 0 ? (
|
|
30
|
+
<div className="text-2xs text-po-text-ghost text-center py-2">
|
|
31
|
+
No messages
|
|
32
|
+
</div>
|
|
33
|
+
) : (
|
|
34
|
+
busMessages.map((msg, index) => (
|
|
35
|
+
<TranscriptRow key={index} msg={msg} />
|
|
36
|
+
))
|
|
37
|
+
)}
|
|
38
|
+
<div ref={messagesEndRef} />
|
|
39
|
+
</div>
|
|
40
|
+
</div>
|
|
41
|
+
)
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function TranscriptRow({ msg }: { msg: { from: string; to: string; content: string | Record<string, unknown>; summary?: string; timestamp: string } }) {
|
|
45
|
+
const [expanded, setExpanded] = useState(false)
|
|
46
|
+
const fullText = msg.summary || (typeof msg.content === 'string' ? msg.content : JSON.stringify(msg.content, null, 2))
|
|
47
|
+
const isLong = fullText.length > 120
|
|
48
|
+
const truncated = isLong ? fullText.slice(0, 120) + '...' : fullText
|
|
49
|
+
|
|
50
|
+
return (
|
|
51
|
+
<div>
|
|
52
|
+
<div
|
|
53
|
+
onClick={() => isLong && setExpanded(!expanded)}
|
|
54
|
+
className={`text-2xs leading-relaxed flex items-baseline gap-1.5 ${isLong ? 'cursor-pointer hover:bg-po-surface-2' : ''} ${expanded ? 'bg-po-surface-2' : ''}`}
|
|
55
|
+
>
|
|
56
|
+
<span className="text-po-text-ghost flex-shrink-0 whitespace-nowrap">
|
|
57
|
+
{new Date(msg.timestamp).toLocaleTimeString('en-US', { hour12: false })}
|
|
58
|
+
</span>
|
|
59
|
+
<span className="text-po-accent flex-shrink-0">{msg.from}</span>
|
|
60
|
+
<span className="text-po-text-ghost">{'\u2192'}</span>
|
|
61
|
+
<span className="text-po-status-calling flex-shrink-0">{msg.to}</span>
|
|
62
|
+
{!expanded && (
|
|
63
|
+
<span className="text-po-text-secondary truncate">{truncated}</span>
|
|
64
|
+
)}
|
|
65
|
+
{isLong && (
|
|
66
|
+
<span className="text-po-text-ghost flex-shrink-0 ml-auto">{expanded ? '\u25BC' : '\u25B8'}</span>
|
|
67
|
+
)}
|
|
68
|
+
</div>
|
|
69
|
+
{expanded && (
|
|
70
|
+
<pre className="text-2xs text-po-text-secondary whitespace-pre-wrap break-all pl-[4.5rem] pb-1.5 bg-po-surface-2 rounded-b mb-0.5">
|
|
71
|
+
{fullText}
|
|
72
|
+
</pre>
|
|
73
|
+
)}
|
|
74
|
+
</div>
|
|
75
|
+
)
|
|
76
|
+
}
|
|
@@ -37,52 +37,52 @@ export function UsagePanel({ usage, onClose }: UsagePanelProps) {
|
|
|
37
37
|
return (
|
|
38
38
|
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50" onClick={onClose}>
|
|
39
39
|
<div
|
|
40
|
-
className="bg-po-surface border border-po-border rounded
|
|
40
|
+
className="bg-po-surface-2 border border-po-border rounded shadow-2xl w-[400px] max-h-[80vh] overflow-auto"
|
|
41
41
|
onClick={(e) => e.stopPropagation()}
|
|
42
42
|
>
|
|
43
|
-
<div className="flex items-center justify-between
|
|
44
|
-
<h3 className="font-
|
|
45
|
-
Token Usage {usage.include_tree && <span className="text-
|
|
43
|
+
<div className="flex items-center justify-between px-4 py-2.5 border-b border-po-border">
|
|
44
|
+
<h3 className="font-mono text-xs text-po-text-primary">
|
|
45
|
+
Token Usage {usage.include_tree && <span className="text-po-text-ghost ml-1">(full tree)</span>}
|
|
46
46
|
</h3>
|
|
47
|
-
<button onClick={onClose} className="text-
|
|
48
|
-
|
|
47
|
+
<button onClick={onClose} className="text-po-text-ghost hover:text-po-text-primary transition-colors duration-150">
|
|
48
|
+
{'\u2715'}
|
|
49
49
|
</button>
|
|
50
50
|
</div>
|
|
51
51
|
|
|
52
52
|
<div className="p-4 space-y-4">
|
|
53
53
|
{/* Summary */}
|
|
54
|
-
<div className="grid grid-cols-3 gap-
|
|
55
|
-
<div className="bg-po-
|
|
56
|
-
<div className="text-
|
|
57
|
-
<div className="text-
|
|
54
|
+
<div className="grid grid-cols-3 gap-2">
|
|
55
|
+
<div className="bg-po-surface rounded p-2.5 text-center">
|
|
56
|
+
<div className="text-sm font-mono text-po-accent">{formatTokens(usage.input_tokens)}</div>
|
|
57
|
+
<div className="text-2xs text-po-text-ghost uppercase tracking-wider mt-0.5">Input</div>
|
|
58
58
|
</div>
|
|
59
|
-
<div className="bg-po-
|
|
60
|
-
<div className="text-
|
|
61
|
-
<div className="text-
|
|
59
|
+
<div className="bg-po-surface rounded p-2.5 text-center">
|
|
60
|
+
<div className="text-sm font-mono text-po-warning">{formatTokens(usage.output_tokens)}</div>
|
|
61
|
+
<div className="text-2xs text-po-text-ghost uppercase tracking-wider mt-0.5">Output</div>
|
|
62
62
|
</div>
|
|
63
|
-
<div className="bg-po-
|
|
64
|
-
<div className="text-
|
|
65
|
-
<div className="text-
|
|
63
|
+
<div className="bg-po-surface rounded p-2.5 text-center">
|
|
64
|
+
<div className="text-sm font-mono text-po-text-primary">{formatCost(usage.estimated_cost_usd)}</div>
|
|
65
|
+
<div className="text-2xs text-po-text-ghost uppercase tracking-wider mt-0.5">Est. Cost</div>
|
|
66
66
|
</div>
|
|
67
67
|
</div>
|
|
68
68
|
|
|
69
|
-
<div className="flex justify-between text-
|
|
70
|
-
<span>{usage.calls}
|
|
71
|
-
<span>{formatTokens(usage.total_tokens)} total
|
|
69
|
+
<div className="flex justify-between text-2xs text-po-text-ghost px-1 font-mono">
|
|
70
|
+
<span>{usage.calls} call{usage.calls !== 1 ? 's' : ''}</span>
|
|
71
|
+
<span>{formatTokens(usage.total_tokens)} total</span>
|
|
72
72
|
</div>
|
|
73
73
|
|
|
74
74
|
{/* Per-model breakdown */}
|
|
75
75
|
{models.length > 0 && (
|
|
76
76
|
<div>
|
|
77
|
-
<h4 className="text-
|
|
78
|
-
<div className="space-y-
|
|
77
|
+
<h4 className="text-2xs text-po-text-ghost uppercase tracking-wider mb-2">By Model</h4>
|
|
78
|
+
<div className="space-y-1.5">
|
|
79
79
|
{models.map(([model, data]) => (
|
|
80
|
-
<div key={model} className="bg-po-
|
|
80
|
+
<div key={model} className="bg-po-surface rounded p-2.5">
|
|
81
81
|
<div className="flex items-center justify-between mb-1">
|
|
82
|
-
<span className="text-xs font-mono text-
|
|
83
|
-
<span className="text-
|
|
82
|
+
<span className="text-xs font-mono text-po-text-primary">{model}</span>
|
|
83
|
+
<span className="text-2xs text-po-text-ghost">{data.calls} call{data.calls !== 1 ? 's' : ''}</span>
|
|
84
84
|
</div>
|
|
85
|
-
<div className="flex justify-between text-
|
|
85
|
+
<div className="flex justify-between text-2xs text-po-text-ghost font-mono">
|
|
86
86
|
<span>In: {formatTokens(data.input_tokens)}</span>
|
|
87
87
|
<span>Out: {formatTokens(data.output_tokens)}</span>
|
|
88
88
|
<span>{formatCost(data.estimated_cost_usd)}</span>
|
|
@@ -94,8 +94,8 @@ export function UsagePanel({ usage, onClose }: UsagePanelProps) {
|
|
|
94
94
|
)}
|
|
95
95
|
|
|
96
96
|
{usage.calls === 0 && (
|
|
97
|
-
<div className="text-center text-
|
|
98
|
-
No usage data recorded
|
|
97
|
+
<div className="text-center text-po-text-ghost text-xs py-4 font-mono">
|
|
98
|
+
No usage data recorded.
|
|
99
99
|
</div>
|
|
100
100
|
)}
|
|
101
101
|
</div>
|