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.
Files changed (78) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +32 -0
  3. data/CLAUDE.md +112 -44
  4. data/Gemfile.lock +31 -29
  5. data/README.md +5 -0
  6. data/frontend/index.html +5 -1
  7. data/frontend/package-lock.json +123 -0
  8. data/frontend/package.json +4 -0
  9. data/frontend/src/App.tsx +70 -71
  10. data/frontend/src/canvas/CanvasView.tsx +113 -0
  11. data/frontend/src/canvas/ForceLayout.ts +115 -0
  12. data/frontend/src/canvas/SceneManager.ts +587 -0
  13. data/frontend/src/canvas/canvasStore.ts +47 -0
  14. data/frontend/src/canvas/constants.ts +95 -0
  15. data/frontend/src/canvas/controls/CameraControls.ts +98 -0
  16. data/frontend/src/canvas/edges/MessageArc.ts +149 -0
  17. data/frontend/src/canvas/inspector/InspectorPanel.tsx +31 -0
  18. data/frontend/src/canvas/inspector/POInspector.tsx +262 -0
  19. data/frontend/src/canvas/inspector/ToolCallInspector.tsx +67 -0
  20. data/frontend/src/canvas/nodes/PONode.ts +249 -0
  21. data/frontend/src/canvas/nodes/ToolCallNode.ts +116 -0
  22. data/frontend/src/canvas/types.ts +24 -0
  23. data/frontend/src/components/ContextMenu.tsx +5 -4
  24. data/frontend/src/components/Inspector.tsx +232 -0
  25. data/frontend/src/components/MarkdownMessage.tsx +22 -20
  26. data/frontend/src/components/MethodList.tsx +90 -0
  27. data/frontend/src/components/ModelSelector.tsx +13 -14
  28. data/frontend/src/components/NotificationPanel.tsx +29 -33
  29. data/frontend/src/components/ObjectList.tsx +78 -0
  30. data/frontend/src/components/PaneSlot.tsx +30 -0
  31. data/frontend/src/components/SourcePane.tsx +202 -0
  32. data/frontend/src/components/SystemBar.tsx +74 -0
  33. data/frontend/src/components/Transcript.tsx +76 -0
  34. data/frontend/src/components/UsagePanel.tsx +27 -27
  35. data/frontend/src/components/Workspace.tsx +260 -0
  36. data/frontend/src/components/index.ts +10 -9
  37. data/frontend/src/hooks/useResize.ts +55 -0
  38. data/frontend/src/hooks/useWebSocket.ts +274 -189
  39. data/frontend/src/index.css +69 -3
  40. data/frontend/src/store/index.ts +23 -0
  41. data/frontend/src/types/index.ts +5 -0
  42. data/frontend/tailwind.config.js +28 -9
  43. data/lib/prompt_objects/capability.rb +23 -1
  44. data/lib/prompt_objects/connectors/mcp.rb +5 -22
  45. data/lib/prompt_objects/environment.rb +8 -0
  46. data/lib/prompt_objects/llm/openai_adapter.rb +22 -0
  47. data/lib/prompt_objects/mcp/tools/get_pending_requests.rb +1 -2
  48. data/lib/prompt_objects/mcp/tools/inspect_po.rb +1 -31
  49. data/lib/prompt_objects/mcp/tools/list_prompt_objects.rb +2 -8
  50. data/lib/prompt_objects/primitives/list_files.rb +1 -2
  51. data/lib/prompt_objects/prompt_object.rb +150 -6
  52. data/lib/prompt_objects/server/api/routes.rb +3 -48
  53. data/lib/prompt_objects/server/app.rb +9 -0
  54. data/lib/prompt_objects/server/public/assets/index-D1myxE0l.js +4345 -0
  55. data/lib/prompt_objects/server/public/assets/index-DdCcwC-Z.css +1 -0
  56. data/lib/prompt_objects/server/public/index.html +7 -3
  57. data/lib/prompt_objects/server/websocket_handler.rb +23 -100
  58. data/lib/prompt_objects/server.rb +6 -62
  59. data/prompt_objects.gemspec +1 -1
  60. data/templates/arc-agi-1/primitives/find_objects.rb +1 -1
  61. data/templates/arc-agi-1/primitives/grid_diff.rb +2 -2
  62. data/templates/arc-agi-1/primitives/grid_info.rb +1 -1
  63. data/templates/arc-agi-1/primitives/grid_transform.rb +1 -1
  64. data/templates/arc-agi-1/primitives/render_grid.rb +1 -0
  65. data/templates/arc-agi-1/primitives/test_solution.rb +3 -0
  66. metadata +26 -14
  67. data/frontend/src/components/CapabilitiesPanel.tsx +0 -141
  68. data/frontend/src/components/ChatPanel.tsx +0 -288
  69. data/frontend/src/components/Dashboard.tsx +0 -83
  70. data/frontend/src/components/Header.tsx +0 -141
  71. data/frontend/src/components/MessageBus.tsx +0 -56
  72. data/frontend/src/components/POCard.tsx +0 -56
  73. data/frontend/src/components/PODetail.tsx +0 -124
  74. data/frontend/src/components/PromptPanel.tsx +0 -156
  75. data/frontend/src/components/SessionsPanel.tsx +0 -174
  76. data/frontend/src/components/ThreadsSidebar.tsx +0 -163
  77. data/lib/prompt_objects/server/public/assets/index-Bkme6COu.css +0 -1
  78. 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-lg shadow-2xl w-[420px] max-h-[80vh] overflow-auto"
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 p-4 border-b border-po-border">
44
- <h3 className="font-medium text-white">
45
- Token Usage {usage.include_tree && <span className="text-xs text-gray-400 ml-1">(full tree)</span>}
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-gray-400 hover:text-white transition-colors">
48
- &times;
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-3">
55
- <div className="bg-po-bg rounded-lg p-3 text-center">
56
- <div className="text-lg font-mono text-po-accent">{formatTokens(usage.input_tokens)}</div>
57
- <div className="text-[10px] text-gray-500 uppercase tracking-wider">Input</div>
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-bg rounded-lg p-3 text-center">
60
- <div className="text-lg font-mono text-po-warning">{formatTokens(usage.output_tokens)}</div>
61
- <div className="text-[10px] text-gray-500 uppercase tracking-wider">Output</div>
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-bg rounded-lg p-3 text-center">
64
- <div className="text-lg font-mono text-white">{formatCost(usage.estimated_cost_usd)}</div>
65
- <div className="text-[10px] text-gray-500 uppercase tracking-wider">Est. Cost</div>
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-xs text-gray-400 px-1">
70
- <span>{usage.calls} LLM call{usage.calls !== 1 ? 's' : ''}</span>
71
- <span>{formatTokens(usage.total_tokens)} total tokens</span>
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-xs text-gray-500 uppercase tracking-wider mb-2">By Model</h4>
78
- <div className="space-y-2">
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-bg rounded-lg p-3">
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-white">{model}</span>
83
- <span className="text-xs text-gray-400">{data.calls} call{data.calls !== 1 ? 's' : ''}</span>
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-[10px] text-gray-500">
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-gray-500 text-sm py-4">
98
- No usage data recorded for this thread.
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>