@assistant-ui/react 0.5.41 → 0.5.45
Sign up to get free protection for your applications and to get access to all the features.
- package/dist/index.d.mts +69 -37
- package/dist/index.d.ts +69 -37
- package/dist/index.js +305 -122
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +261 -78
- package/dist/index.mjs.map +1 -1
- package/dist/styles/index.css +30 -18
- package/dist/styles/index.css.map +1 -1
- package/dist/styles/tailwindcss/thread.css +4 -4
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
@@ -164,6 +164,7 @@ import { create as create4 } from "zustand";
|
|
164
164
|
var getThreadStateFromRuntime = (runtime) => {
|
165
165
|
const lastMessage = runtime.messages.at(-1);
|
166
166
|
return Object.freeze({
|
167
|
+
threadId: runtime.threadId,
|
167
168
|
capabilities: runtime.capabilities,
|
168
169
|
isDisabled: runtime.isDisabled,
|
169
170
|
isRunning: lastMessage?.role !== "assistant" ? false : lastMessage.status.type === "running"
|
@@ -636,6 +637,9 @@ var useSmooth = (state, smooth = false) => {
|
|
636
637
|
setDisplayedText(text2);
|
637
638
|
useSmoothStatus2?.setState(text2 !== state.part.text ? SMOOTH_STATUS : state.status);
|
638
639
|
});
|
640
|
+
useEffect(() => {
|
641
|
+
useSmoothStatus2?.setState(text !== displayedText ? SMOOTH_STATUS : state.status);
|
642
|
+
}, [useSmoothStatus2, text, displayedText, state.status]);
|
639
643
|
const [animatorRef] = useState2(
|
640
644
|
new TextStreamAnimator(text, setText)
|
641
645
|
);
|
@@ -1072,59 +1076,12 @@ var useEdgeRuntime = ({
|
|
1072
1076
|
// src/runtimes/local/shouldContinue.tsx
|
1073
1077
|
var shouldContinue = (result) => result.status?.type === "requires-action" && result.status.reason === "tool-calls" && result.content.every((c) => c.type !== "tool-call" || !!c.result);
|
1074
1078
|
|
1075
|
-
// src/utils/getThreadMessageText.tsx
|
1076
|
-
var getThreadMessageText = (message) => {
|
1077
|
-
const textParts = message.content.filter(
|
1078
|
-
(part) => part.type === "text"
|
1079
|
-
);
|
1080
|
-
return textParts.map((part) => part.text).join("\n\n");
|
1081
|
-
};
|
1082
|
-
|
1083
|
-
// src/runtimes/speech/WebSpeechSynthesisAdapter.ts
|
1084
|
-
var WebSpeechSynthesisAdapter = class {
|
1085
|
-
speak(message) {
|
1086
|
-
const text = getThreadMessageText(message);
|
1087
|
-
const utterance = new SpeechSynthesisUtterance(text);
|
1088
|
-
let ended = false;
|
1089
|
-
const endHandlers = /* @__PURE__ */ new Set();
|
1090
|
-
const handleEnd = () => {
|
1091
|
-
if (ended) return;
|
1092
|
-
ended = true;
|
1093
|
-
endHandlers.forEach((handler) => handler());
|
1094
|
-
};
|
1095
|
-
utterance.addEventListener("end", handleEnd);
|
1096
|
-
utterance.addEventListener("error", handleEnd);
|
1097
|
-
window.speechSynthesis.speak(utterance);
|
1098
|
-
return {
|
1099
|
-
stop: () => {
|
1100
|
-
window.speechSynthesis.cancel();
|
1101
|
-
handleEnd();
|
1102
|
-
},
|
1103
|
-
onEnd: (callback) => {
|
1104
|
-
if (ended) {
|
1105
|
-
let cancelled = false;
|
1106
|
-
queueMicrotask(() => {
|
1107
|
-
if (!cancelled) callback();
|
1108
|
-
});
|
1109
|
-
return () => {
|
1110
|
-
cancelled = true;
|
1111
|
-
};
|
1112
|
-
} else {
|
1113
|
-
endHandlers.add(callback);
|
1114
|
-
return () => {
|
1115
|
-
endHandlers.delete(callback);
|
1116
|
-
};
|
1117
|
-
}
|
1118
|
-
}
|
1119
|
-
};
|
1120
|
-
}
|
1121
|
-
};
|
1122
|
-
|
1123
1079
|
// src/runtimes/local/LocalThreadRuntime.tsx
|
1124
1080
|
var LocalThreadRuntime = class {
|
1125
1081
|
constructor(configProvider, adapter, { initialMessages, ...options }) {
|
1126
1082
|
this.configProvider = configProvider;
|
1127
1083
|
this.adapter = adapter;
|
1084
|
+
this.threadId = generateId();
|
1128
1085
|
this.options = options;
|
1129
1086
|
if (initialMessages) {
|
1130
1087
|
let parentId = null;
|
@@ -1146,6 +1103,7 @@ var LocalThreadRuntime = class {
|
|
1146
1103
|
unstable_copy: true,
|
1147
1104
|
speak: false
|
1148
1105
|
};
|
1106
|
+
threadId;
|
1149
1107
|
isDisabled = false;
|
1150
1108
|
get messages() {
|
1151
1109
|
return this.repository.getMessages();
|
@@ -1158,6 +1116,9 @@ var LocalThreadRuntime = class {
|
|
1158
1116
|
}
|
1159
1117
|
};
|
1160
1118
|
_options;
|
1119
|
+
get options() {
|
1120
|
+
return this._options;
|
1121
|
+
}
|
1161
1122
|
set options({ initialMessages, ...options }) {
|
1162
1123
|
this._options = options;
|
1163
1124
|
const canSpeak = options.adapters?.speech !== void 0;
|
@@ -1239,7 +1200,7 @@ var LocalThreadRuntime = class {
|
|
1239
1200
|
this.repository.addOrUpdateMessage(parentId, message);
|
1240
1201
|
this.notifySubscribers();
|
1241
1202
|
};
|
1242
|
-
const maxToolRoundtrips = this.
|
1203
|
+
const maxToolRoundtrips = this.options.maxToolRoundtrips ?? 1;
|
1243
1204
|
const toolRoundtrips = message.metadata?.roundtrips?.length ?? 0;
|
1244
1205
|
if (toolRoundtrips > maxToolRoundtrips) {
|
1245
1206
|
updateMessage({
|
@@ -1334,10 +1295,24 @@ var LocalThreadRuntime = class {
|
|
1334
1295
|
this.performRoundtrip(parentId, message);
|
1335
1296
|
}
|
1336
1297
|
}
|
1298
|
+
// TODO lift utterance state to thread runtime
|
1299
|
+
_utterance;
|
1337
1300
|
speak(messageId) {
|
1301
|
+
const adapter = this.options.adapters?.speech;
|
1302
|
+
if (!adapter) throw new Error("Speech adapter not configured");
|
1338
1303
|
const { message } = this.repository.getMessage(messageId);
|
1339
|
-
|
1340
|
-
|
1304
|
+
if (this._utterance) {
|
1305
|
+
this._utterance.cancel();
|
1306
|
+
this._utterance = void 0;
|
1307
|
+
}
|
1308
|
+
const utterance = adapter.speak(message);
|
1309
|
+
utterance.onEnd(() => {
|
1310
|
+
if (this._utterance === utterance) {
|
1311
|
+
this._utterance = void 0;
|
1312
|
+
}
|
1313
|
+
});
|
1314
|
+
this._utterance = utterance;
|
1315
|
+
return this._utterance;
|
1341
1316
|
}
|
1342
1317
|
export() {
|
1343
1318
|
return this.repository.export();
|
@@ -1431,11 +1406,12 @@ var getAutoStatus = (isLast, isRunning) => isLast && isRunning ? AUTO_STATUS_RUN
|
|
1431
1406
|
|
1432
1407
|
// src/runtimes/external-store/ThreadMessageLike.tsx
|
1433
1408
|
var fromThreadMessageLike = (like, fallbackId, fallbackStatus) => {
|
1434
|
-
const { role,
|
1409
|
+
const { role, id, createdAt, status } = like;
|
1435
1410
|
const common = {
|
1436
1411
|
id: id ?? fallbackId,
|
1437
1412
|
createdAt: createdAt ?? /* @__PURE__ */ new Date()
|
1438
1413
|
};
|
1414
|
+
const content = typeof like.content === "string" ? [{ type: "text", text: like.content }] : like.content;
|
1439
1415
|
switch (role) {
|
1440
1416
|
case "assistant":
|
1441
1417
|
return {
|
@@ -1445,6 +1421,8 @@ var fromThreadMessageLike = (like, fallbackId, fallbackStatus) => {
|
|
1445
1421
|
const type = part.type;
|
1446
1422
|
switch (type) {
|
1447
1423
|
case "text":
|
1424
|
+
if (part.text.trim().length === 0) return null;
|
1425
|
+
return part;
|
1448
1426
|
case "ui":
|
1449
1427
|
return part;
|
1450
1428
|
case "tool-call": {
|
@@ -1459,7 +1437,7 @@ var fromThreadMessageLike = (like, fallbackId, fallbackStatus) => {
|
|
1459
1437
|
throw new Error(`Unknown content part type: ${unhandledType}`);
|
1460
1438
|
}
|
1461
1439
|
}
|
1462
|
-
}),
|
1440
|
+
}).filter((c) => !!c),
|
1463
1441
|
status: status ?? fallbackStatus
|
1464
1442
|
};
|
1465
1443
|
case "user":
|
@@ -1497,6 +1475,14 @@ var fromThreadMessageLike = (like, fallbackId, fallbackStatus) => {
|
|
1497
1475
|
}
|
1498
1476
|
};
|
1499
1477
|
|
1478
|
+
// src/utils/getThreadMessageText.tsx
|
1479
|
+
var getThreadMessageText = (message) => {
|
1480
|
+
const textParts = message.content.filter(
|
1481
|
+
(part) => part.type === "text"
|
1482
|
+
);
|
1483
|
+
return textParts.map((part) => part.text).join("\n\n");
|
1484
|
+
};
|
1485
|
+
|
1500
1486
|
// src/runtimes/external-store/ExternalStoreThreadRuntime.tsx
|
1501
1487
|
var hasUpcomingMessage = (isRunning, messages) => {
|
1502
1488
|
return isRunning && messages[messages.length - 1]?.role !== "assistant";
|
@@ -1516,8 +1502,9 @@ var ExternalStoreThreadRuntime = class {
|
|
1516
1502
|
get capabilities() {
|
1517
1503
|
return this._capabilities;
|
1518
1504
|
}
|
1519
|
-
|
1520
|
-
|
1505
|
+
threadId;
|
1506
|
+
messages;
|
1507
|
+
isDisabled;
|
1521
1508
|
converter = new ThreadMessageConverter();
|
1522
1509
|
_store;
|
1523
1510
|
composer = {
|
@@ -1530,8 +1517,12 @@ var ExternalStoreThreadRuntime = class {
|
|
1530
1517
|
constructor(store) {
|
1531
1518
|
this.store = store;
|
1532
1519
|
}
|
1520
|
+
get store() {
|
1521
|
+
return this._store;
|
1522
|
+
}
|
1533
1523
|
set store(store) {
|
1534
1524
|
if (this._store === store) return;
|
1525
|
+
this.threadId = store.threadId ?? this.threadId ?? generateId();
|
1535
1526
|
const isRunning = store.isRunning ?? false;
|
1536
1527
|
this.isDisabled = store.isDisabled ?? false;
|
1537
1528
|
const oldStore = this._store;
|
@@ -1667,6 +1658,9 @@ var ExternalStoreRuntime = class extends BaseAssistantRuntime {
|
|
1667
1658
|
constructor(store) {
|
1668
1659
|
super(new ExternalStoreThreadRuntime(store));
|
1669
1660
|
}
|
1661
|
+
get store() {
|
1662
|
+
return this.thread.store;
|
1663
|
+
}
|
1670
1664
|
set store(store) {
|
1671
1665
|
this.thread.store = store;
|
1672
1666
|
}
|
@@ -1707,6 +1701,153 @@ var useExternalStoreRuntime = (store) => {
|
|
1707
1701
|
return runtime;
|
1708
1702
|
};
|
1709
1703
|
|
1704
|
+
// src/runtimes/external-store/external-message-converter.tsx
|
1705
|
+
import { useMemo as useMemo2 } from "react";
|
1706
|
+
var joinExternalMessages = (messages) => {
|
1707
|
+
const assistantMessage = {
|
1708
|
+
role: "assistant",
|
1709
|
+
content: []
|
1710
|
+
};
|
1711
|
+
for (const output of messages) {
|
1712
|
+
if (output.role === "tool") {
|
1713
|
+
const toolCallIdx = assistantMessage.content.findIndex(
|
1714
|
+
(c) => c.type === "tool-call" && c.toolCallId === output.toolCallId
|
1715
|
+
);
|
1716
|
+
if (toolCallIdx !== -1) {
|
1717
|
+
const toolCall = assistantMessage.content[toolCallIdx];
|
1718
|
+
if (output.toolName) {
|
1719
|
+
if (toolCall.toolName !== output.toolName)
|
1720
|
+
throw new Error(
|
1721
|
+
`Tool call name ${output.toolCallId} ${output.toolName} does not match existing tool call ${toolCall.toolName}`
|
1722
|
+
);
|
1723
|
+
}
|
1724
|
+
assistantMessage.content[toolCallIdx] = {
|
1725
|
+
...toolCall,
|
1726
|
+
result: output.result
|
1727
|
+
};
|
1728
|
+
} else {
|
1729
|
+
throw new Error(
|
1730
|
+
`Tool call ${output.toolCallId} ${output.toolName} not found in assistant message`
|
1731
|
+
);
|
1732
|
+
}
|
1733
|
+
} else {
|
1734
|
+
const content = output.content;
|
1735
|
+
const role = output.role;
|
1736
|
+
switch (role) {
|
1737
|
+
case "system":
|
1738
|
+
case "user":
|
1739
|
+
return { role, content };
|
1740
|
+
case "assistant":
|
1741
|
+
if (assistantMessage.content.length === 0) {
|
1742
|
+
assistantMessage.id = output.id;
|
1743
|
+
assistantMessage.createdAt ??= output.createdAt;
|
1744
|
+
assistantMessage.status ??= output.status;
|
1745
|
+
}
|
1746
|
+
assistantMessage.content.push(...content);
|
1747
|
+
break;
|
1748
|
+
default: {
|
1749
|
+
const unsupportedRole = role;
|
1750
|
+
throw new Error(`Unknown message role: ${unsupportedRole}`);
|
1751
|
+
}
|
1752
|
+
}
|
1753
|
+
}
|
1754
|
+
}
|
1755
|
+
return assistantMessage;
|
1756
|
+
};
|
1757
|
+
var chunkExternalMessages = (callbackResults) => {
|
1758
|
+
const results = [];
|
1759
|
+
let isAssistant = false;
|
1760
|
+
let inputs = [];
|
1761
|
+
let outputs = [];
|
1762
|
+
const flush = () => {
|
1763
|
+
if (outputs.length) {
|
1764
|
+
results.push({
|
1765
|
+
inputs,
|
1766
|
+
outputs
|
1767
|
+
});
|
1768
|
+
}
|
1769
|
+
inputs = [];
|
1770
|
+
outputs = [];
|
1771
|
+
};
|
1772
|
+
for (const callbackResult of callbackResults) {
|
1773
|
+
for (const output of callbackResult.outputs) {
|
1774
|
+
if (!isAssistant || output.role === "user" || output.role === "system") {
|
1775
|
+
flush();
|
1776
|
+
}
|
1777
|
+
isAssistant = output.role === "assistant" || output.role === "tool";
|
1778
|
+
if (inputs.at(-1) !== callbackResult.input) {
|
1779
|
+
inputs.push(callbackResult.input);
|
1780
|
+
}
|
1781
|
+
outputs.push(output);
|
1782
|
+
}
|
1783
|
+
}
|
1784
|
+
flush();
|
1785
|
+
return results;
|
1786
|
+
};
|
1787
|
+
var useExternalMessageConverter = ({
|
1788
|
+
callback,
|
1789
|
+
messages,
|
1790
|
+
isRunning
|
1791
|
+
}) => {
|
1792
|
+
const state = useMemo2(
|
1793
|
+
() => ({
|
1794
|
+
callback,
|
1795
|
+
callbackCache: /* @__PURE__ */ new WeakMap(),
|
1796
|
+
chunkCache: /* @__PURE__ */ new WeakMap(),
|
1797
|
+
converterCache: new ThreadMessageConverter()
|
1798
|
+
}),
|
1799
|
+
[callback]
|
1800
|
+
);
|
1801
|
+
return useMemo2(() => {
|
1802
|
+
const callbackResults = [];
|
1803
|
+
for (const message of messages) {
|
1804
|
+
let result = state.callbackCache.get(message);
|
1805
|
+
if (!result) {
|
1806
|
+
const output = state.callback(message);
|
1807
|
+
const outputs = Array.isArray(output) ? output : [output];
|
1808
|
+
result = { input: message, outputs };
|
1809
|
+
state.callbackCache.set(message, result);
|
1810
|
+
}
|
1811
|
+
callbackResults.push(result);
|
1812
|
+
}
|
1813
|
+
const chunks = chunkExternalMessages(callbackResults).map((m) => {
|
1814
|
+
const key = m.outputs[0];
|
1815
|
+
if (!key) return m;
|
1816
|
+
const cached = state.chunkCache.get(key);
|
1817
|
+
if (cached && shallowArrayEqual(cached.outputs, m.outputs)) return cached;
|
1818
|
+
state.chunkCache.set(key, m);
|
1819
|
+
return m;
|
1820
|
+
});
|
1821
|
+
return state.converterCache.convertMessages(
|
1822
|
+
chunks,
|
1823
|
+
(cache, message, idx) => {
|
1824
|
+
const isLast = idx === chunks.length - 1;
|
1825
|
+
const autoStatus = getAutoStatus(isLast, isRunning);
|
1826
|
+
if (cache && (cache.role !== "assistant" || !isAutoStatus(cache.status) || cache.status === autoStatus)) {
|
1827
|
+
const inputs = getExternalStoreMessage(cache);
|
1828
|
+
if (shallowArrayEqual(inputs, message.inputs)) {
|
1829
|
+
return cache;
|
1830
|
+
}
|
1831
|
+
}
|
1832
|
+
const newMessage = fromThreadMessageLike(
|
1833
|
+
joinExternalMessages(message.outputs),
|
1834
|
+
idx.toString(),
|
1835
|
+
autoStatus
|
1836
|
+
);
|
1837
|
+
newMessage[symbolInnerMessage] = message.inputs;
|
1838
|
+
return newMessage;
|
1839
|
+
}
|
1840
|
+
);
|
1841
|
+
}, [state, messages, isRunning]);
|
1842
|
+
};
|
1843
|
+
var shallowArrayEqual = (a, b) => {
|
1844
|
+
if (a.length !== b.length) return false;
|
1845
|
+
for (let i = 0; i < a.length; i++) {
|
1846
|
+
if (a[i] !== b[i]) return false;
|
1847
|
+
}
|
1848
|
+
return true;
|
1849
|
+
};
|
1850
|
+
|
1710
1851
|
// src/runtimes/dangerous-in-browser/useDangerousInBrowserRuntime.ts
|
1711
1852
|
import { useState as useState6 } from "react";
|
1712
1853
|
|
@@ -1743,6 +1884,47 @@ var useDangerousInBrowserRuntime = ({
|
|
1743
1884
|
return useLocalRuntime(adapter, { initialMessages });
|
1744
1885
|
};
|
1745
1886
|
|
1887
|
+
// src/runtimes/speech/WebSpeechSynthesisAdapter.ts
|
1888
|
+
var WebSpeechSynthesisAdapter = class {
|
1889
|
+
speak(message) {
|
1890
|
+
const text = getThreadMessageText(message);
|
1891
|
+
const utterance = new SpeechSynthesisUtterance(text);
|
1892
|
+
const endHandlers = /* @__PURE__ */ new Set();
|
1893
|
+
const handleEnd = (reason, error) => {
|
1894
|
+
if (res.status.type === "ended") return;
|
1895
|
+
res.status = { type: "ended", reason, error };
|
1896
|
+
endHandlers.forEach((handler) => handler());
|
1897
|
+
};
|
1898
|
+
utterance.addEventListener("end", () => handleEnd("finished"));
|
1899
|
+
utterance.addEventListener("error", (e) => handleEnd("error", e.error));
|
1900
|
+
window.speechSynthesis.speak(utterance);
|
1901
|
+
const res = {
|
1902
|
+
status: { type: "running" },
|
1903
|
+
cancel: () => {
|
1904
|
+
window.speechSynthesis.cancel();
|
1905
|
+
handleEnd("cancelled");
|
1906
|
+
},
|
1907
|
+
onEnd: (callback) => {
|
1908
|
+
if (res.status.type === "ended") {
|
1909
|
+
let cancelled = false;
|
1910
|
+
queueMicrotask(() => {
|
1911
|
+
if (!cancelled) callback();
|
1912
|
+
});
|
1913
|
+
return () => {
|
1914
|
+
cancelled = true;
|
1915
|
+
};
|
1916
|
+
} else {
|
1917
|
+
endHandlers.add(callback);
|
1918
|
+
return () => {
|
1919
|
+
endHandlers.delete(callback);
|
1920
|
+
};
|
1921
|
+
}
|
1922
|
+
}
|
1923
|
+
};
|
1924
|
+
return res;
|
1925
|
+
}
|
1926
|
+
};
|
1927
|
+
|
1746
1928
|
// src/context/providers/ThreadProvider.tsx
|
1747
1929
|
import { jsx as jsx6, jsxs as jsxs2 } from "react/jsx-runtime";
|
1748
1930
|
var ThreadProvider = ({
|
@@ -1770,7 +1952,7 @@ var ThreadProvider = ({
|
|
1770
1952
|
const thread = provider.thread;
|
1771
1953
|
const oldState = context.useThread.getState();
|
1772
1954
|
const state = getThreadStateFromRuntime(thread);
|
1773
|
-
if (oldState.isDisabled !== state.isDisabled || oldState.isRunning !== state.isRunning || // TODO ensure capabilities is memoized
|
1955
|
+
if (oldState.threadId !== state.threadId || oldState.isDisabled !== state.isDisabled || oldState.isRunning !== state.isRunning || // TODO ensure capabilities is memoized
|
1774
1956
|
oldState.capabilities !== state.capabilities) {
|
1775
1957
|
context.useThread.setState(
|
1776
1958
|
state,
|
@@ -1859,7 +2041,7 @@ var AssistantRuntimeProviderImpl = ({ children, runtime }) => {
|
|
1859
2041
|
var AssistantRuntimeProvider = memo(AssistantRuntimeProviderImpl);
|
1860
2042
|
|
1861
2043
|
// src/context/react/ComposerContext.ts
|
1862
|
-
import { useMemo as
|
2044
|
+
import { useMemo as useMemo3 } from "react";
|
1863
2045
|
|
1864
2046
|
// src/context/react/MessageContext.ts
|
1865
2047
|
import { createContext as createContext4, useContext as useContext4 } from "react";
|
@@ -1877,7 +2059,7 @@ function useMessageContext(options) {
|
|
1877
2059
|
var useComposerContext = () => {
|
1878
2060
|
const { useComposer } = useThreadContext();
|
1879
2061
|
const { useEditComposer } = useMessageContext({ optional: true }) ?? {};
|
1880
|
-
return
|
2062
|
+
return useMemo3(
|
1881
2063
|
() => ({
|
1882
2064
|
useComposer: useEditComposer ?? useComposer,
|
1883
2065
|
type: useEditComposer ? "edit" : "new"
|
@@ -2017,7 +2199,7 @@ var useAssistantInstructions = (instruction) => {
|
|
2017
2199
|
import { useCallback as useCallback3 } from "react";
|
2018
2200
|
|
2019
2201
|
// src/utils/combined/useCombinedStore.ts
|
2020
|
-
import { useMemo as
|
2202
|
+
import { useMemo as useMemo4 } from "react";
|
2021
2203
|
|
2022
2204
|
// src/utils/combined/createCombinedStore.ts
|
2023
2205
|
import { useSyncExternalStore } from "react";
|
@@ -2038,7 +2220,7 @@ var createCombinedStore = (stores) => {
|
|
2038
2220
|
|
2039
2221
|
// src/utils/combined/useCombinedStore.ts
|
2040
2222
|
var useCombinedStore = (stores, selector) => {
|
2041
|
-
const useCombined =
|
2223
|
+
const useCombined = useMemo4(() => createCombinedStore(stores), stores);
|
2042
2224
|
return useCombined(selector);
|
2043
2225
|
};
|
2044
2226
|
|
@@ -2900,6 +3082,7 @@ var MessageContentPartComponent = ({
|
|
2900
3082
|
const Tool = by_name[part.toolName] || Fallback2;
|
2901
3083
|
const addResult = (result) => addToolResult({
|
2902
3084
|
messageId: useMessage.getState().message.id,
|
3085
|
+
toolName: part.toolName,
|
2903
3086
|
toolCallId: part.toolCallId,
|
2904
3087
|
result
|
2905
3088
|
});
|
@@ -3335,7 +3518,7 @@ var makeMessageUtilsStore = () => create14((set) => {
|
|
3335
3518
|
},
|
3336
3519
|
isSpeaking: false,
|
3337
3520
|
stopSpeaking: () => {
|
3338
|
-
utterance?.
|
3521
|
+
utterance?.cancel();
|
3339
3522
|
},
|
3340
3523
|
addUtterance: (utt) => {
|
3341
3524
|
utterance = utt;
|
@@ -3537,28 +3720,29 @@ import {
|
|
3537
3720
|
StopCircleIcon
|
3538
3721
|
} from "lucide-react";
|
3539
3722
|
import { Fragment as Fragment4, jsx as jsx32, jsxs as jsxs5 } from "react/jsx-runtime";
|
3540
|
-
var useAllowCopy = () => {
|
3723
|
+
var useAllowCopy = (ensureCapability = false) => {
|
3541
3724
|
const { assistantMessage: { allowCopy = true } = {} } = useThreadConfig();
|
3542
3725
|
const { useThread } = useThreadContext();
|
3543
3726
|
const copySupported = useThread((t) => t.capabilities.unstable_copy);
|
3544
|
-
return
|
3727
|
+
return allowCopy && (!ensureCapability || copySupported);
|
3545
3728
|
};
|
3546
|
-
var useAllowSpeak = () => {
|
3729
|
+
var useAllowSpeak = (ensureCapability = false) => {
|
3547
3730
|
const { assistantMessage: { allowSpeak = true } = {} } = useThreadConfig();
|
3548
3731
|
const { useThread } = useThreadContext();
|
3549
3732
|
const speakSupported = useThread((t) => t.capabilities.speak);
|
3550
|
-
return
|
3733
|
+
return allowSpeak && (!ensureCapability || speakSupported);
|
3551
3734
|
};
|
3552
|
-
var useAllowReload = () => {
|
3735
|
+
var useAllowReload = (ensureCapability = false) => {
|
3553
3736
|
const { assistantMessage: { allowReload = true } = {} } = useThreadConfig();
|
3554
3737
|
const { useThread } = useThreadContext();
|
3555
3738
|
const reloadSupported = useThread((t) => t.capabilities.reload);
|
3556
|
-
return
|
3739
|
+
return allowReload && (!ensureCapability || reloadSupported);
|
3557
3740
|
};
|
3558
3741
|
var AssistantActionBar = () => {
|
3559
|
-
const allowCopy = useAllowCopy();
|
3560
|
-
const allowReload = useAllowReload();
|
3561
|
-
|
3742
|
+
const allowCopy = useAllowCopy(true);
|
3743
|
+
const allowReload = useAllowReload(true);
|
3744
|
+
const allowSpeak = useAllowSpeak(true);
|
3745
|
+
if (!allowCopy && !allowReload && !allowSpeak) return null;
|
3562
3746
|
return /* @__PURE__ */ jsxs5(
|
3563
3747
|
AssistantActionBarRoot,
|
3564
3748
|
{
|
@@ -3566,9 +3750,9 @@ var AssistantActionBar = () => {
|
|
3566
3750
|
autohide: "not-last",
|
3567
3751
|
autohideFloat: "single-branch",
|
3568
3752
|
children: [
|
3569
|
-
/* @__PURE__ */ jsx32(AssistantActionBarSpeechControl, {}),
|
3570
|
-
/* @__PURE__ */ jsx32(AssistantActionBarCopy, {}),
|
3571
|
-
/* @__PURE__ */ jsx32(AssistantActionBarReload, {})
|
3753
|
+
allowSpeak && /* @__PURE__ */ jsx32(AssistantActionBarSpeechControl, {}),
|
3754
|
+
allowCopy && /* @__PURE__ */ jsx32(AssistantActionBarCopy, {}),
|
3755
|
+
allowReload && /* @__PURE__ */ jsx32(AssistantActionBarReload, {})
|
3572
3756
|
]
|
3573
3757
|
}
|
3574
3758
|
);
|
@@ -3584,8 +3768,6 @@ var AssistantActionBarCopy = forwardRef20((props, ref) => {
|
|
3584
3768
|
assistantMessage: { copy: { tooltip = "Copy" } = {} } = {}
|
3585
3769
|
} = {}
|
3586
3770
|
} = useThreadConfig();
|
3587
|
-
const allowCopy = useAllowCopy();
|
3588
|
-
if (!allowCopy) return null;
|
3589
3771
|
return /* @__PURE__ */ jsx32(actionBar_exports.Copy, { asChild: true, children: /* @__PURE__ */ jsx32(TooltipIconButton, { tooltip, ...props, ref, children: props.children ?? /* @__PURE__ */ jsxs5(Fragment4, { children: [
|
3590
3772
|
/* @__PURE__ */ jsx32(message_exports.If, { copied: true, children: /* @__PURE__ */ jsx32(CheckIcon, {}) }),
|
3591
3773
|
/* @__PURE__ */ jsx32(message_exports.If, { copied: false, children: /* @__PURE__ */ jsx32(CopyIcon, {}) })
|
@@ -3979,14 +4161,14 @@ import { forwardRef as forwardRef26 } from "react";
|
|
3979
4161
|
import { forwardRef as forwardRef25 } from "react";
|
3980
4162
|
import { PencilIcon } from "lucide-react";
|
3981
4163
|
import { jsx as jsx40 } from "react/jsx-runtime";
|
3982
|
-
var useAllowEdit = () => {
|
4164
|
+
var useAllowEdit = (ensureCapability = false) => {
|
3983
4165
|
const { userMessage: { allowEdit = true } = {} } = useThreadConfig();
|
3984
4166
|
const { useThread } = useThreadContext();
|
3985
4167
|
const editSupported = useThread((t) => t.capabilities.edit);
|
3986
|
-
return
|
4168
|
+
return allowEdit && (!ensureCapability || editSupported);
|
3987
4169
|
};
|
3988
4170
|
var UserActionBar = () => {
|
3989
|
-
const allowEdit = useAllowEdit();
|
4171
|
+
const allowEdit = useAllowEdit(true);
|
3990
4172
|
if (!allowEdit) return null;
|
3991
4173
|
return /* @__PURE__ */ jsx40(UserActionBarRoot, { hideWhenRunning: true, autohide: "not-last", children: /* @__PURE__ */ jsx40(UserActionBarEdit, {}) });
|
3992
4174
|
};
|
@@ -4310,6 +4492,7 @@ export {
|
|
4310
4492
|
useContentPartText,
|
4311
4493
|
useDangerousInBrowserRuntime,
|
4312
4494
|
useEdgeRuntime,
|
4495
|
+
useExternalMessageConverter,
|
4313
4496
|
useExternalStoreRuntime,
|
4314
4497
|
useLocalRuntime,
|
4315
4498
|
useMessageContext,
|