@happyvertical/smrt-chat 0.30.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 (169) hide show
  1. package/AGENTS.md +35 -0
  2. package/CLAUDE.md +1 -0
  3. package/LICENSE +7 -0
  4. package/README.md +163 -0
  5. package/dist/__smrt-register__.d.ts +2 -0
  6. package/dist/__smrt-register__.d.ts.map +1 -0
  7. package/dist/chunks/ChatService-Dpzc1Pa5.js +2044 -0
  8. package/dist/chunks/ChatService-Dpzc1Pa5.js.map +1 -0
  9. package/dist/collections/AgentSessionCollection.d.ts +57 -0
  10. package/dist/collections/AgentSessionCollection.d.ts.map +1 -0
  11. package/dist/collections/ChatMessageCollection.d.ts +79 -0
  12. package/dist/collections/ChatMessageCollection.d.ts.map +1 -0
  13. package/dist/collections/ChatParticipantCollection.d.ts +26 -0
  14. package/dist/collections/ChatParticipantCollection.d.ts.map +1 -0
  15. package/dist/collections/ChatReactionCollection.d.ts +23 -0
  16. package/dist/collections/ChatReactionCollection.d.ts.map +1 -0
  17. package/dist/collections/ChatRoomCollection.d.ts +43 -0
  18. package/dist/collections/ChatRoomCollection.d.ts.map +1 -0
  19. package/dist/collections/ChatThreadCollection.d.ts +9 -0
  20. package/dist/collections/ChatThreadCollection.d.ts.map +1 -0
  21. package/dist/index.d.ts +5 -0
  22. package/dist/index.d.ts.map +1 -0
  23. package/dist/index.js +18 -0
  24. package/dist/index.js.map +1 -0
  25. package/dist/internal/agent-runtime.d.ts +16 -0
  26. package/dist/internal/agent-runtime.d.ts.map +1 -0
  27. package/dist/internal/agent-runtime.js +5 -0
  28. package/dist/internal/agent-runtime.js.map +1 -0
  29. package/dist/manifest.json +2811 -0
  30. package/dist/models/AgentSession.d.ts +70 -0
  31. package/dist/models/AgentSession.d.ts.map +1 -0
  32. package/dist/models/ChatMessage.d.ts +55 -0
  33. package/dist/models/ChatMessage.d.ts.map +1 -0
  34. package/dist/models/ChatParticipant.d.ts +32 -0
  35. package/dist/models/ChatParticipant.d.ts.map +1 -0
  36. package/dist/models/ChatReaction.d.ts +19 -0
  37. package/dist/models/ChatReaction.d.ts.map +1 -0
  38. package/dist/models/ChatRoom.d.ts +44 -0
  39. package/dist/models/ChatRoom.d.ts.map +1 -0
  40. package/dist/models/ChatThread.d.ts +24 -0
  41. package/dist/models/ChatThread.d.ts.map +1 -0
  42. package/dist/models/index.d.ts +7 -0
  43. package/dist/models/index.d.ts.map +1 -0
  44. package/dist/playground.d.ts +2 -0
  45. package/dist/playground.d.ts.map +1 -0
  46. package/dist/playground.js +166 -0
  47. package/dist/playground.js.map +1 -0
  48. package/dist/services/ChatService.d.ts +390 -0
  49. package/dist/services/ChatService.d.ts.map +1 -0
  50. package/dist/services/index.d.ts +2 -0
  51. package/dist/services/index.d.ts.map +1 -0
  52. package/dist/smrt-knowledge.json +1507 -0
  53. package/dist/svelte/components/agent/AgentChat.svelte +542 -0
  54. package/dist/svelte/components/agent/AgentChat.svelte.d.ts +21 -0
  55. package/dist/svelte/components/agent/AgentChat.svelte.d.ts.map +1 -0
  56. package/dist/svelte/components/agent/AgentSelector.svelte +175 -0
  57. package/dist/svelte/components/agent/AgentSelector.svelte.d.ts +11 -0
  58. package/dist/svelte/components/agent/AgentSelector.svelte.d.ts.map +1 -0
  59. package/dist/svelte/components/agent/AgentSessionPanel.svelte +322 -0
  60. package/dist/svelte/components/agent/AgentSessionPanel.svelte.d.ts +15 -0
  61. package/dist/svelte/components/agent/AgentSessionPanel.svelte.d.ts.map +1 -0
  62. package/dist/svelte/components/agent/ToolCallDisplay.svelte +335 -0
  63. package/dist/svelte/components/agent/ToolCallDisplay.svelte.d.ts +9 -0
  64. package/dist/svelte/components/agent/ToolCallDisplay.svelte.d.ts.map +1 -0
  65. package/dist/svelte/components/agent/message-blocks.d.ts +12 -0
  66. package/dist/svelte/components/agent/message-blocks.d.ts.map +1 -0
  67. package/dist/svelte/components/agent/message-blocks.js +41 -0
  68. package/dist/svelte/components/agent/message-blocks.test.js +31 -0
  69. package/dist/svelte/components/dialogs/RoomCreateDialog.svelte +403 -0
  70. package/dist/svelte/components/dialogs/RoomCreateDialog.svelte.d.ts +16 -0
  71. package/dist/svelte/components/dialogs/RoomCreateDialog.svelte.d.ts.map +1 -0
  72. package/dist/svelte/components/dialogs/SearchMessages.svelte +457 -0
  73. package/dist/svelte/components/dialogs/SearchMessages.svelte.d.ts +17 -0
  74. package/dist/svelte/components/dialogs/SearchMessages.svelte.d.ts.map +1 -0
  75. package/dist/svelte/components/layout/ChatLayout.svelte +150 -0
  76. package/dist/svelte/components/layout/ChatLayout.svelte.d.ts +18 -0
  77. package/dist/svelte/components/layout/ChatLayout.svelte.d.ts.map +1 -0
  78. package/dist/svelte/components/layout/MemberList.svelte +389 -0
  79. package/dist/svelte/components/layout/MemberList.svelte.d.ts +11 -0
  80. package/dist/svelte/components/layout/MemberList.svelte.d.ts.map +1 -0
  81. package/dist/svelte/components/layout/RoomHeader.svelte +241 -0
  82. package/dist/svelte/components/layout/RoomHeader.svelte.d.ts +15 -0
  83. package/dist/svelte/components/layout/RoomHeader.svelte.d.ts.map +1 -0
  84. package/dist/svelte/components/layout/RoomList.svelte +471 -0
  85. package/dist/svelte/components/layout/RoomList.svelte.d.ts +15 -0
  86. package/dist/svelte/components/layout/RoomList.svelte.d.ts.map +1 -0
  87. package/dist/svelte/components/messages/MessageInput.svelte +232 -0
  88. package/dist/svelte/components/messages/MessageInput.svelte.d.ts +20 -0
  89. package/dist/svelte/components/messages/MessageInput.svelte.d.ts.map +1 -0
  90. package/dist/svelte/components/messages/MessageItem.svelte +431 -0
  91. package/dist/svelte/components/messages/MessageItem.svelte.d.ts +19 -0
  92. package/dist/svelte/components/messages/MessageItem.svelte.d.ts.map +1 -0
  93. package/dist/svelte/components/messages/MessageList.svelte +129 -0
  94. package/dist/svelte/components/messages/MessageList.svelte.d.ts +17 -0
  95. package/dist/svelte/components/messages/MessageList.svelte.d.ts.map +1 -0
  96. package/dist/svelte/components/messages/ThreadPanel.svelte +156 -0
  97. package/dist/svelte/components/messages/ThreadPanel.svelte.d.ts +17 -0
  98. package/dist/svelte/components/messages/ThreadPanel.svelte.d.ts.map +1 -0
  99. package/dist/svelte/components/messages/__tests__/MessageInput.test.js +38 -0
  100. package/dist/svelte/components/shared/Avatar.svelte +30 -0
  101. package/dist/svelte/components/shared/Avatar.svelte.d.ts +14 -0
  102. package/dist/svelte/components/shared/Avatar.svelte.d.ts.map +1 -0
  103. package/dist/svelte/components/shared/FileUpload.svelte +382 -0
  104. package/dist/svelte/components/shared/FileUpload.svelte.d.ts +14 -0
  105. package/dist/svelte/components/shared/FileUpload.svelte.d.ts.map +1 -0
  106. package/dist/svelte/components/shared/LinkPreview.svelte +108 -0
  107. package/dist/svelte/components/shared/LinkPreview.svelte.d.ts +18 -0
  108. package/dist/svelte/components/shared/LinkPreview.svelte.d.ts.map +1 -0
  109. package/dist/svelte/components/shared/MentionAutocomplete.svelte +168 -0
  110. package/dist/svelte/components/shared/MentionAutocomplete.svelte.d.ts +18 -0
  111. package/dist/svelte/components/shared/MentionAutocomplete.svelte.d.ts.map +1 -0
  112. package/dist/svelte/components/shared/MessageBubble.svelte +81 -0
  113. package/dist/svelte/components/shared/MessageBubble.svelte.d.ts +16 -0
  114. package/dist/svelte/components/shared/MessageBubble.svelte.d.ts.map +1 -0
  115. package/dist/svelte/components/shared/ReactionPicker.svelte +103 -0
  116. package/dist/svelte/components/shared/ReactionPicker.svelte.d.ts +10 -0
  117. package/dist/svelte/components/shared/ReactionPicker.svelte.d.ts.map +1 -0
  118. package/dist/svelte/components/shared/ReadReceipts.svelte +127 -0
  119. package/dist/svelte/components/shared/ReadReceipts.svelte.d.ts +13 -0
  120. package/dist/svelte/components/shared/ReadReceipts.svelte.d.ts.map +1 -0
  121. package/dist/svelte/components/shared/TypingIndicator.svelte +90 -0
  122. package/dist/svelte/components/shared/TypingIndicator.svelte.d.ts +12 -0
  123. package/dist/svelte/components/shared/TypingIndicator.svelte.d.ts.map +1 -0
  124. package/dist/svelte/components/shared/UserPresence.svelte +65 -0
  125. package/dist/svelte/components/shared/UserPresence.svelte.d.ts +13 -0
  126. package/dist/svelte/components/shared/UserPresence.svelte.d.ts.map +1 -0
  127. package/dist/svelte/components/shared/__tests__/Avatar.test.js +20 -0
  128. package/dist/svelte/components/shared/__tests__/LinkPreview.test.js +29 -0
  129. package/dist/svelte/components/shared/__tests__/MessageBubble.test.js +21 -0
  130. package/dist/svelte/components/shared/__tests__/ReactionPicker.test.js +35 -0
  131. package/dist/svelte/components/shared/__tests__/ReadReceipts.test.js +28 -0
  132. package/dist/svelte/components/shared/__tests__/TypingIndicator.test.js +27 -0
  133. package/dist/svelte/components/shared/__tests__/UserPresence.test.js +23 -0
  134. package/dist/svelte/components/tabs/ChatTab.svelte +240 -0
  135. package/dist/svelte/components/tabs/ChatTab.svelte.d.ts +21 -0
  136. package/dist/svelte/components/tabs/ChatTab.svelte.d.ts.map +1 -0
  137. package/dist/svelte/components/tabs/ChatTabList.svelte +158 -0
  138. package/dist/svelte/components/tabs/ChatTabList.svelte.d.ts +13 -0
  139. package/dist/svelte/components/tabs/ChatTabList.svelte.d.ts.map +1 -0
  140. package/dist/svelte/components/tabs/ChatTabs.svelte +88 -0
  141. package/dist/svelte/components/tabs/ChatTabs.svelte.d.ts +21 -0
  142. package/dist/svelte/components/tabs/ChatTabs.svelte.d.ts.map +1 -0
  143. package/dist/svelte/components/tabs/MiniChat.svelte +253 -0
  144. package/dist/svelte/components/tabs/MiniChat.svelte.d.ts +15 -0
  145. package/dist/svelte/components/tabs/MiniChat.svelte.d.ts.map +1 -0
  146. package/dist/svelte/i18n.d.ts +51 -0
  147. package/dist/svelte/i18n.d.ts.map +1 -0
  148. package/dist/svelte/i18n.js +72 -0
  149. package/dist/svelte/i18n.messages.d.ts +50 -0
  150. package/dist/svelte/i18n.messages.d.ts.map +1 -0
  151. package/dist/svelte/i18n.messages.js +69 -0
  152. package/dist/svelte/index.d.ts +48 -0
  153. package/dist/svelte/index.d.ts.map +1 -0
  154. package/dist/svelte/index.js +117 -0
  155. package/dist/svelte/playground.d.ts +171 -0
  156. package/dist/svelte/playground.d.ts.map +1 -0
  157. package/dist/svelte/playground.js +161 -0
  158. package/dist/svelte/types.d.ts +116 -0
  159. package/dist/svelte/types.d.ts.map +1 -0
  160. package/dist/svelte/types.js +1 -0
  161. package/dist/types.d.ts +99 -0
  162. package/dist/types.d.ts.map +1 -0
  163. package/dist/types.js +2 -0
  164. package/dist/types.js.map +1 -0
  165. package/dist/ui.d.ts +4 -0
  166. package/dist/ui.d.ts.map +1 -0
  167. package/dist/ui.js +92 -0
  168. package/dist/ui.js.map +1 -0
  169. package/package.json +95 -0
@@ -0,0 +1,335 @@
1
+ <script lang="ts">
2
+ /**
3
+ * ToolCallDisplay - Inline tool call/result display
4
+ * Collapsible card showing tool name, arguments (as JSON), status indicator,
5
+ * and result/error. Color-coded by status (pending, running, success, error).
6
+ */
7
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
8
+ import { M } from '../../i18n.js';
9
+ import type { ToolCallDisplayData } from '../../types.js';
10
+
11
+ export interface Props {
12
+ /** Tool call data */
13
+ toolCall: ToolCallDisplayData;
14
+ }
15
+
16
+ const { toolCall }: Props = $props();
17
+
18
+ const { t } = useI18n();
19
+
20
+ let isExpanded = $state(false);
21
+
22
+ const statusIcon = $derived.by(() => {
23
+ switch (toolCall.status) {
24
+ case 'pending':
25
+ return '...';
26
+ case 'running':
27
+ return '...';
28
+ case 'success':
29
+ return '...';
30
+ case 'error':
31
+ return '...';
32
+ default:
33
+ return '';
34
+ }
35
+ });
36
+
37
+ const statusLabel = $derived.by(() => {
38
+ switch (toolCall.status) {
39
+ case 'pending':
40
+ return 'Pending';
41
+ case 'running':
42
+ return 'Running';
43
+ case 'success':
44
+ return 'Completed';
45
+ case 'error':
46
+ return 'Error';
47
+ default:
48
+ return toolCall.status;
49
+ }
50
+ });
51
+
52
+ function formatJson(value: unknown): string {
53
+ try {
54
+ return JSON.stringify(value, null, 2);
55
+ } catch {
56
+ return String(value);
57
+ }
58
+ }
59
+
60
+ function formatDuration(ms: number | undefined): string {
61
+ if (ms === undefined) return '';
62
+ if (ms < 1000) return `${ms}ms`;
63
+ return `${(ms / 1000).toFixed(1)}s`;
64
+ }
65
+ </script>
66
+
67
+ <div class="tool-call tool-call--{toolCall.status}" aria-label={t(M['chat.tool_call_display.tool_call'], { toolName: toolCall.toolName })}>
68
+ <button
69
+ class="tool-call__header"
70
+ type="button"
71
+ onclick={() => isExpanded = !isExpanded}
72
+ aria-expanded={isExpanded}
73
+ aria-controls="tool-call-body-{toolCall.toolCallId}"
74
+ >
75
+ <div class="tool-call__status-dot" aria-label="{statusLabel}"></div>
76
+
77
+ <span class="tool-call__name">{toolCall.toolName}</span>
78
+
79
+ <div class="tool-call__header-right">
80
+ {#if toolCall.duration !== undefined}
81
+ <span class="tool-call__duration">{formatDuration(toolCall.duration)}</span>
82
+ {/if}
83
+ <span class="tool-call__status-label">{statusLabel}</span>
84
+ <svg
85
+ class="tool-call__chevron"
86
+ class:tool-call__chevron--open={isExpanded}
87
+ width="16"
88
+ height="16"
89
+ viewBox="0 0 24 24"
90
+ fill="currentColor"
91
+ aria-hidden="true"
92
+ >
93
+ <path d="M7.41 8.59L12 13.17l4.59-4.58L18 10l-6 6-6-6z" />
94
+ </svg>
95
+ </div>
96
+ </button>
97
+
98
+ {#if isExpanded}
99
+ <div class="tool-call__body" id="tool-call-body-{toolCall.toolCallId}">
100
+ {#if toolCall.arguments && Object.keys(toolCall.arguments).length > 0}
101
+ <div class="tool-call__section">
102
+ <span class="tool-call__section-label">Arguments</span>
103
+ <pre class="tool-call__json">{formatJson(toolCall.arguments)}</pre>
104
+ </div>
105
+ {/if}
106
+
107
+ {#if toolCall.status === 'success' && toolCall.result !== undefined}
108
+ <div class="tool-call__section">
109
+ <span class="tool-call__section-label">Result</span>
110
+ <pre class="tool-call__json tool-call__json--result">{formatJson(toolCall.result)}</pre>
111
+ </div>
112
+ {/if}
113
+
114
+ {#if toolCall.status === 'error' && toolCall.error}
115
+ <div class="tool-call__section">
116
+ <span class="tool-call__section-label">Error</span>
117
+ <pre class="tool-call__json tool-call__json--error">{toolCall.error}</pre>
118
+ </div>
119
+ {/if}
120
+
121
+ {#if toolCall.status === 'running'}
122
+ <div class="tool-call__running">
123
+ <div class="tool-call__spinner" aria-label={t(M['chat.tool_call_display.running'])}></div>
124
+ <span>Executing...</span>
125
+ </div>
126
+ {/if}
127
+ </div>
128
+ {/if}
129
+ </div>
130
+
131
+ <style>
132
+ .tool-call {
133
+ border: 1px solid var(--smrt-color-outline-variant, #c4c6d0);
134
+ border-radius: var(--smrt-radius-medium, 8px);
135
+ overflow: hidden;
136
+ font-family: var(--smrt-font-family, system-ui);
137
+ }
138
+
139
+ .tool-call--pending {
140
+ border-left: 3px solid var(--smrt-color-outline, #74777f);
141
+ }
142
+
143
+ .tool-call--running {
144
+ border-left: 3px solid var(--smrt-color-primary, #005ac1);
145
+ }
146
+
147
+ .tool-call--success {
148
+ border-left: 3px solid var(--smrt-color-success, #4caf50);
149
+ }
150
+
151
+ .tool-call--error {
152
+ border-left: 3px solid var(--smrt-color-error, #ba1a1a);
153
+ }
154
+
155
+ .tool-call__header {
156
+ display: flex;
157
+ align-items: center;
158
+ gap: var(--smrt-spacing-2, 8px);
159
+ width: 100%;
160
+ padding: var(--smrt-spacing-2, 8px) var(--smrt-spacing-3, 12px);
161
+ border: none;
162
+ background: var(--smrt-color-surface-container-low, #f7f7fb);
163
+ cursor: pointer;
164
+ text-align: left;
165
+ color: var(--smrt-color-on-surface, #1a1c1e);
166
+ font: inherit;
167
+ transition: background var(--smrt-duration-short2, 150ms);
168
+ }
169
+
170
+ .tool-call__header:hover {
171
+ background: var(--smrt-color-surface-container, #f0f0f4);
172
+ }
173
+
174
+ .tool-call__header:focus-visible {
175
+ outline: 2px solid var(--smrt-color-primary, #005ac1);
176
+ outline-offset: -2px;
177
+ }
178
+
179
+ .tool-call__status-dot {
180
+ width: 8px;
181
+ height: 8px;
182
+ border-radius: var(--smrt-radius-full, 9999px);
183
+ flex-shrink: 0;
184
+ }
185
+
186
+ .tool-call--pending .tool-call__status-dot {
187
+ background: var(--smrt-color-outline, #74777f);
188
+ }
189
+
190
+ .tool-call--running .tool-call__status-dot {
191
+ background: var(--smrt-color-primary, #005ac1);
192
+ animation: pulse 1.5s ease-in-out infinite;
193
+ }
194
+
195
+ .tool-call--success .tool-call__status-dot {
196
+ background: var(--smrt-color-success, #4caf50);
197
+ }
198
+
199
+ .tool-call--error .tool-call__status-dot {
200
+ background: var(--smrt-color-error, #ba1a1a);
201
+ }
202
+
203
+ .tool-call__name {
204
+ font: var(--smrt-typography-label-large-font, 500 0.875rem/1.25 sans-serif);
205
+ font-family: var(--smrt-font-family-mono, 'SF Mono', 'Fira Code', 'Cascadia Code', monospace);
206
+ flex: 1;
207
+ min-width: 0;
208
+ white-space: nowrap;
209
+ overflow: hidden;
210
+ text-overflow: ellipsis;
211
+ }
212
+
213
+ .tool-call__header-right {
214
+ display: flex;
215
+ align-items: center;
216
+ gap: var(--smrt-spacing-2, 8px);
217
+ flex-shrink: 0;
218
+ }
219
+
220
+ .tool-call__duration {
221
+ font: var(--smrt-typography-label-small-font, 500 0.6875rem/1 sans-serif);
222
+ color: var(--smrt-color-outline, #74777f);
223
+ }
224
+
225
+ .tool-call__status-label {
226
+ font: var(--smrt-typography-label-small-font, 500 0.6875rem/1 sans-serif);
227
+ color: var(--smrt-color-on-surface-variant, #43474e);
228
+ }
229
+
230
+ .tool-call__chevron {
231
+ color: var(--smrt-color-on-surface-variant, #43474e);
232
+ transition: transform var(--smrt-duration-short2, 150ms);
233
+ }
234
+
235
+ .tool-call__chevron--open {
236
+ transform: rotate(180deg);
237
+ }
238
+
239
+ .tool-call__body {
240
+ padding: var(--smrt-spacing-2, 8px) var(--smrt-spacing-3, 12px) var(--smrt-spacing-3, 12px);
241
+ display: flex;
242
+ flex-direction: column;
243
+ gap: var(--smrt-spacing-2, 8px);
244
+ border-top: 1px solid var(--smrt-color-outline-variant, #c4c6d0);
245
+ background: var(--smrt-color-surface, #fefbff);
246
+ }
247
+
248
+ .tool-call__section {
249
+ display: flex;
250
+ flex-direction: column;
251
+ gap: var(--smrt-spacing-1, 4px);
252
+ }
253
+
254
+ .tool-call__section-label {
255
+ font: var(--smrt-typography-label-small-font, 500 0.6875rem/1 sans-serif);
256
+ color: var(--smrt-color-on-surface-variant, #43474e);
257
+ text-transform: uppercase;
258
+ letter-spacing: var(--smrt-typography-label-small-tracking, 0.5px);
259
+ }
260
+
261
+ .tool-call__json {
262
+ margin: 0;
263
+ padding: var(--smrt-spacing-2, 8px);
264
+ border-radius: var(--smrt-radius-small, 4px);
265
+ background: var(--smrt-color-surface-container-low, #f7f7fb);
266
+ color: var(--smrt-color-on-surface, #1a1c1e);
267
+ font-family: var(--smrt-font-family-mono, 'SF Mono', 'Fira Code', 'Cascadia Code', monospace);
268
+ font-size: var(--smrt-typography-body-small-size, 0.75rem);
269
+ line-height: var(--smrt-typography-body-small-line-height, 1.5);
270
+ overflow-x: auto;
271
+ white-space: pre-wrap;
272
+ word-break: break-all;
273
+ max-height: 200px;
274
+ overflow-y: auto;
275
+ }
276
+
277
+ .tool-call__json--result {
278
+ background: var(--smrt-color-success-container, #f1f8e9);
279
+ color: var(--smrt-color-on-success-container, #33691e);
280
+ }
281
+
282
+ .tool-call__json--error {
283
+ background: var(--smrt-color-error-container, #ffdad6);
284
+ color: var(--smrt-color-on-error-container, #410002);
285
+ }
286
+
287
+ .tool-call__running {
288
+ display: flex;
289
+ align-items: center;
290
+ gap: var(--smrt-spacing-2, 8px);
291
+ padding: var(--smrt-spacing-2, 8px);
292
+ font: var(--smrt-typography-body-small-font, 0.8125rem/1.4 sans-serif);
293
+ color: var(--smrt-color-primary, #005ac1);
294
+ }
295
+
296
+ .tool-call__spinner {
297
+ width: 16px;
298
+ height: 16px;
299
+ border: 2px solid var(--smrt-color-primary-container, #d6e3ff);
300
+ border-top-color: var(--smrt-color-primary, #005ac1);
301
+ border-radius: var(--smrt-radius-full, 9999px);
302
+ animation: spin 0.8s linear infinite;
303
+ }
304
+
305
+ @keyframes spin {
306
+ to {
307
+ transform: rotate(360deg);
308
+ }
309
+ }
310
+
311
+ @keyframes pulse {
312
+ 0%, 100% {
313
+ opacity: 1;
314
+ }
315
+ 50% {
316
+ opacity: 0.4;
317
+ }
318
+ }
319
+
320
+ @media (prefers-reduced-motion: reduce) {
321
+ .tool-call__chevron,
322
+ .tool-call__header {
323
+ transition: none;
324
+ }
325
+
326
+ .tool-call__spinner {
327
+ animation: none;
328
+ border-top-color: var(--smrt-color-primary, #005ac1);
329
+ }
330
+
331
+ .tool-call--running .tool-call__status-dot {
332
+ animation: none;
333
+ }
334
+ }
335
+ </style>
@@ -0,0 +1,9 @@
1
+ import type { ToolCallDisplayData } from '../../types.js';
2
+ export interface Props {
3
+ /** Tool call data */
4
+ toolCall: ToolCallDisplayData;
5
+ }
6
+ declare const ToolCallDisplay: import("svelte").Component<Props, {}, "">;
7
+ type ToolCallDisplay = ReturnType<typeof ToolCallDisplay>;
8
+ export default ToolCallDisplay;
9
+ //# sourceMappingURL=ToolCallDisplay.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToolCallDisplay.svelte.d.ts","sourceRoot":"","sources":["../../../../src/svelte/components/agent/ToolCallDisplay.svelte.ts"],"names":[],"mappings":"AAUA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAG1D,MAAM,WAAW,KAAK;IACpB,qBAAqB;IACrB,QAAQ,EAAE,mBAAmB,CAAC;CAC/B;AAiHD,QAAA,MAAM,eAAe,2CAAwC,CAAC;AAC9D,KAAK,eAAe,GAAG,UAAU,CAAC,OAAO,eAAe,CAAC,CAAC;AAC1D,eAAe,eAAe,CAAC"}
@@ -0,0 +1,12 @@
1
+ export type ParsedBlock = {
2
+ type: 'text';
3
+ content: string;
4
+ } | {
5
+ type: 'fields';
6
+ fields: Record<string, string>;
7
+ } | {
8
+ type: 'markdown';
9
+ content: string;
10
+ };
11
+ export declare function parseAgentMessageBlocks(text: string): ParsedBlock[];
12
+ //# sourceMappingURL=message-blocks.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"message-blocks.d.ts","sourceRoot":"","sources":["../../../../src/svelte/components/agent/message-blocks.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,WAAW,GACnB;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,GACjC;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CAAE,GAClD;IAAE,IAAI,EAAE,UAAU,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,CAAC;AAE1C,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,GAAG,WAAW,EAAE,CA2CnE"}
@@ -0,0 +1,41 @@
1
+ export function parseAgentMessageBlocks(text) {
2
+ if (!text)
3
+ return [];
4
+ const regex = /```(json|markdown)[ \t]*\r?\n([\s\S]*?)```/gi;
5
+ const blocks = [];
6
+ let lastIndex = 0;
7
+ let match = regex.exec(text);
8
+ while (match !== null) {
9
+ if (match.index > lastIndex) {
10
+ blocks.push({
11
+ type: 'text',
12
+ content: text.slice(lastIndex, match.index),
13
+ });
14
+ }
15
+ const lang = match[1].toLowerCase();
16
+ const body = match[2].trim();
17
+ if (lang === 'json') {
18
+ try {
19
+ const parsed = JSON.parse(body);
20
+ if (parsed.fields && typeof parsed.fields === 'object') {
21
+ blocks.push({ type: 'fields', fields: parsed.fields });
22
+ }
23
+ else {
24
+ blocks.push({ type: 'text', content: match[0] });
25
+ }
26
+ }
27
+ catch {
28
+ blocks.push({ type: 'text', content: match[0] });
29
+ }
30
+ }
31
+ else {
32
+ blocks.push({ type: 'markdown', content: body });
33
+ }
34
+ lastIndex = regex.lastIndex;
35
+ match = regex.exec(text);
36
+ }
37
+ if (lastIndex < text.length) {
38
+ blocks.push({ type: 'text', content: text.slice(lastIndex) });
39
+ }
40
+ return blocks;
41
+ }
@@ -0,0 +1,31 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { parseAgentMessageBlocks } from './message-blocks';
3
+ describe('parseAgentMessageBlocks', () => {
4
+ it('preserves fenced markdown blocks as readable content', () => {
5
+ expect(parseAgentMessageBlocks('Here is a draft:\n```markdown\n# Heading\n\nBody copy\n```\nThanks.')).toEqual([
6
+ { type: 'text', content: 'Here is a draft:\n' },
7
+ { type: 'markdown', content: '# Heading\n\nBody copy' },
8
+ { type: 'text', content: '\nThanks.' },
9
+ ]);
10
+ });
11
+ it('extracts field updates from json blocks', () => {
12
+ expect(parseAgentMessageBlocks('```json\n{"fields":{"title":"New title","body":"Updated body"}}\n```')).toEqual([
13
+ {
14
+ type: 'fields',
15
+ fields: { title: 'New title', body: 'Updated body' },
16
+ },
17
+ ]);
18
+ });
19
+ it('falls back to plain text for invalid json blocks', () => {
20
+ expect(parseAgentMessageBlocks('```json\n{not valid json}\n```')).toEqual([
21
+ { type: 'text', content: '```json\n{not valid json}\n```' },
22
+ ]);
23
+ });
24
+ it('supports windows newlines and whitespace after the language tag', () => {
25
+ expect(parseAgentMessageBlocks('Intro\r\n```markdown \r\n# Heading\r\n\r\nBody copy\r\n```\r\nDone')).toEqual([
26
+ { type: 'text', content: 'Intro\r\n' },
27
+ { type: 'markdown', content: '# Heading\r\n\r\nBody copy' },
28
+ { type: 'text', content: '\r\nDone' },
29
+ ]);
30
+ });
31
+ });