@djangocfg/ui-tools 2.1.344 → 2.1.345

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.cjs CHANGED
@@ -7,7 +7,7 @@ var chunkKNDLV4PI_cjs = require('./chunk-KNDLV4PI.cjs');
7
7
  var chunk5I5QNGUG_cjs = require('./chunk-5I5QNGUG.cjs');
8
8
  var chunkYW5IVWHQ_cjs = require('./chunk-YW5IVWHQ.cjs');
9
9
  var chunk76NNDZH6_cjs = require('./chunk-76NNDZH6.cjs');
10
- var chunkAK2VHUJV_cjs = require('./chunk-AK2VHUJV.cjs');
10
+ var chunkHE3N2ACJ_cjs = require('./chunk-HE3N2ACJ.cjs');
11
11
  var chunkYXZ6GU7H_cjs = require('./chunk-YXZ6GU7H.cjs');
12
12
  var chunkIEEAENLX_cjs = require('./chunk-IEEAENLX.cjs');
13
13
  var chunkQW4RBGHN_cjs = require('./chunk-QW4RBGHN.cjs');
@@ -372,7 +372,7 @@ var LazyTree = createLazyComponent(
372
372
  }
373
373
  );
374
374
  var LazyChat = createLazyComponent(
375
- () => import('./ChatRoot-CIOFUESH.cjs').then((m) => ({ default: m.ChatRoot })),
375
+ () => import('./ChatRoot-RU7MQUXF.cjs').then((m) => ({ default: m.ChatRoot })),
376
376
  {
377
377
  displayName: "LazyChat",
378
378
  fallback: /* @__PURE__ */ jsxRuntime.jsx(LoadingFallback, { minHeight: 320, text: "Loading chat\u2026" })
@@ -409,7 +409,7 @@ async function* parseSSE(response, options = {}) {
409
409
  throw new Error("SSE response has no body");
410
410
  }
411
411
  const map = options.map ?? DEFAULT_MAP;
412
- const idleMs = options.idleTimeoutMs ?? chunkAK2VHUJV_cjs.LIMITS.sseIdleMs;
412
+ const idleMs = options.idleTimeoutMs ?? chunkHE3N2ACJ_cjs.LIMITS.sseIdleMs;
413
413
  const reader = response.body.getReader();
414
414
  const decoder = new TextDecoder();
415
415
  let buffer = "";
@@ -612,7 +612,7 @@ function createMockTransport(opts = {}) {
612
612
  async createSession(_opts) {
613
613
  await sleep(latency);
614
614
  return {
615
- sessionId: chunkAK2VHUJV_cjs.createId("s"),
615
+ sessionId: chunkHE3N2ACJ_cjs.createId("s"),
616
616
  messages: history.length ? [...history] : void 0,
617
617
  hasMore: false,
618
618
  cursor: null,
@@ -629,12 +629,12 @@ function createMockTransport(opts = {}) {
629
629
  throw new Error("mock transport scripted failure");
630
630
  }
631
631
  history.push({
632
- id: chunkAK2VHUJV_cjs.createId("u"),
632
+ id: chunkHE3N2ACJ_cjs.createId("u"),
633
633
  role: "user",
634
634
  content,
635
635
  createdAt: Date.now()
636
636
  });
637
- const messageId = chunkAK2VHUJV_cjs.createId("a");
637
+ const messageId = chunkHE3N2ACJ_cjs.createId("a");
638
638
  yield { type: "message_start", messageId, sessionId: _sid };
639
639
  const reply = replies[turn % replies.length];
640
640
  turn += 1;
@@ -663,7 +663,7 @@ function createMockTransport(opts = {}) {
663
663
  turn += 1;
664
664
  const text = typeof reply === "string" ? reply : reply.filter((e) => e.type === "chunk").map((e) => e.delta).join("");
665
665
  return {
666
- id: chunkAK2VHUJV_cjs.createId("a"),
666
+ id: chunkHE3N2ACJ_cjs.createId("a"),
667
667
  role: "assistant",
668
668
  content: text || DEFAULT_REPLY,
669
669
  createdAt: Date.now()
@@ -2019,159 +2019,159 @@ Object.defineProperty(exports, "useCronWeekDays", {
2019
2019
  });
2020
2020
  Object.defineProperty(exports, "Attachments", {
2021
2021
  enumerable: true,
2022
- get: function () { return chunkAK2VHUJV_cjs.Attachments; }
2022
+ get: function () { return chunkHE3N2ACJ_cjs.Attachments; }
2023
2023
  });
2024
2024
  Object.defineProperty(exports, "AttachmentsGrid", {
2025
2025
  enumerable: true,
2026
- get: function () { return chunkAK2VHUJV_cjs.AttachmentsGrid; }
2026
+ get: function () { return chunkHE3N2ACJ_cjs.AttachmentsGrid; }
2027
2027
  });
2028
2028
  Object.defineProperty(exports, "AttachmentsList", {
2029
2029
  enumerable: true,
2030
- get: function () { return chunkAK2VHUJV_cjs.AttachmentsList; }
2030
+ get: function () { return chunkHE3N2ACJ_cjs.AttachmentsList; }
2031
2031
  });
2032
2032
  Object.defineProperty(exports, "CHAT_EVENT_NAME", {
2033
2033
  enumerable: true,
2034
- get: function () { return chunkAK2VHUJV_cjs.CHAT_EVENT_NAME; }
2034
+ get: function () { return chunkHE3N2ACJ_cjs.CHAT_EVENT_NAME; }
2035
2035
  });
2036
2036
  Object.defineProperty(exports, "CSS_VARS", {
2037
2037
  enumerable: true,
2038
- get: function () { return chunkAK2VHUJV_cjs.CSS_VARS; }
2038
+ get: function () { return chunkHE3N2ACJ_cjs.CSS_VARS; }
2039
2039
  });
2040
2040
  Object.defineProperty(exports, "ChatProvider", {
2041
2041
  enumerable: true,
2042
- get: function () { return chunkAK2VHUJV_cjs.ChatProvider; }
2042
+ get: function () { return chunkHE3N2ACJ_cjs.ChatProvider; }
2043
2043
  });
2044
2044
  Object.defineProperty(exports, "ChatRoot", {
2045
2045
  enumerable: true,
2046
- get: function () { return chunkAK2VHUJV_cjs.ChatRoot; }
2046
+ get: function () { return chunkHE3N2ACJ_cjs.ChatRoot; }
2047
2047
  });
2048
2048
  Object.defineProperty(exports, "Composer", {
2049
2049
  enumerable: true,
2050
- get: function () { return chunkAK2VHUJV_cjs.Composer; }
2050
+ get: function () { return chunkHE3N2ACJ_cjs.Composer; }
2051
2051
  });
2052
2052
  Object.defineProperty(exports, "DEFAULT_LABELS", {
2053
2053
  enumerable: true,
2054
- get: function () { return chunkAK2VHUJV_cjs.DEFAULT_LABELS; }
2054
+ get: function () { return chunkHE3N2ACJ_cjs.DEFAULT_LABELS; }
2055
2055
  });
2056
2056
  Object.defineProperty(exports, "DEFAULT_SIDEBAR", {
2057
2057
  enumerable: true,
2058
- get: function () { return chunkAK2VHUJV_cjs.DEFAULT_SIDEBAR; }
2058
+ get: function () { return chunkHE3N2ACJ_cjs.DEFAULT_SIDEBAR; }
2059
2059
  });
2060
2060
  Object.defineProperty(exports, "DEFAULT_Z_INDEX", {
2061
2061
  enumerable: true,
2062
- get: function () { return chunkAK2VHUJV_cjs.DEFAULT_Z_INDEX; }
2062
+ get: function () { return chunkHE3N2ACJ_cjs.DEFAULT_Z_INDEX; }
2063
2063
  });
2064
2064
  Object.defineProperty(exports, "EmptyState", {
2065
2065
  enumerable: true,
2066
- get: function () { return chunkAK2VHUJV_cjs.EmptyState; }
2066
+ get: function () { return chunkHE3N2ACJ_cjs.EmptyState; }
2067
2067
  });
2068
2068
  Object.defineProperty(exports, "ErrorBanner", {
2069
2069
  enumerable: true,
2070
- get: function () { return chunkAK2VHUJV_cjs.ErrorBanner; }
2070
+ get: function () { return chunkHE3N2ACJ_cjs.ErrorBanner; }
2071
2071
  });
2072
2072
  Object.defineProperty(exports, "HOTKEYS", {
2073
2073
  enumerable: true,
2074
- get: function () { return chunkAK2VHUJV_cjs.HOTKEYS; }
2074
+ get: function () { return chunkHE3N2ACJ_cjs.HOTKEYS; }
2075
2075
  });
2076
2076
  Object.defineProperty(exports, "JumpToLatest", {
2077
2077
  enumerable: true,
2078
- get: function () { return chunkAK2VHUJV_cjs.JumpToLatest; }
2078
+ get: function () { return chunkHE3N2ACJ_cjs.JumpToLatest; }
2079
2079
  });
2080
2080
  Object.defineProperty(exports, "LIMITS", {
2081
2081
  enumerable: true,
2082
- get: function () { return chunkAK2VHUJV_cjs.LIMITS; }
2082
+ get: function () { return chunkHE3N2ACJ_cjs.LIMITS; }
2083
2083
  });
2084
2084
  Object.defineProperty(exports, "MessageActions", {
2085
2085
  enumerable: true,
2086
- get: function () { return chunkAK2VHUJV_cjs.MessageActions; }
2086
+ get: function () { return chunkHE3N2ACJ_cjs.MessageActions; }
2087
2087
  });
2088
2088
  Object.defineProperty(exports, "MessageBubble", {
2089
2089
  enumerable: true,
2090
- get: function () { return chunkAK2VHUJV_cjs.MessageBubble; }
2090
+ get: function () { return chunkHE3N2ACJ_cjs.MessageBubble; }
2091
2091
  });
2092
2092
  Object.defineProperty(exports, "MessageList", {
2093
2093
  enumerable: true,
2094
- get: function () { return chunkAK2VHUJV_cjs.MessageList; }
2094
+ get: function () { return chunkHE3N2ACJ_cjs.MessageList; }
2095
2095
  });
2096
2096
  Object.defineProperty(exports, "STORAGE_KEYS", {
2097
2097
  enumerable: true,
2098
- get: function () { return chunkAK2VHUJV_cjs.STORAGE_KEYS; }
2098
+ get: function () { return chunkHE3N2ACJ_cjs.STORAGE_KEYS; }
2099
2099
  });
2100
2100
  Object.defineProperty(exports, "Sources", {
2101
2101
  enumerable: true,
2102
- get: function () { return chunkAK2VHUJV_cjs.Sources; }
2102
+ get: function () { return chunkHE3N2ACJ_cjs.Sources; }
2103
2103
  });
2104
2104
  Object.defineProperty(exports, "StreamingIndicator", {
2105
2105
  enumerable: true,
2106
- get: function () { return chunkAK2VHUJV_cjs.StreamingIndicator; }
2106
+ get: function () { return chunkHE3N2ACJ_cjs.StreamingIndicator; }
2107
2107
  });
2108
2108
  Object.defineProperty(exports, "ToolCalls", {
2109
2109
  enumerable: true,
2110
- get: function () { return chunkAK2VHUJV_cjs.ToolCalls; }
2110
+ get: function () { return chunkHE3N2ACJ_cjs.ToolCalls; }
2111
2111
  });
2112
2112
  Object.defineProperty(exports, "createId", {
2113
2113
  enumerable: true,
2114
- get: function () { return chunkAK2VHUJV_cjs.createId; }
2114
+ get: function () { return chunkHE3N2ACJ_cjs.createId; }
2115
2115
  });
2116
2116
  Object.defineProperty(exports, "createTokenBuffer", {
2117
2117
  enumerable: true,
2118
- get: function () { return chunkAK2VHUJV_cjs.createTokenBuffer; }
2118
+ get: function () { return chunkHE3N2ACJ_cjs.createTokenBuffer; }
2119
2119
  });
2120
2120
  Object.defineProperty(exports, "deriveInitials", {
2121
2121
  enumerable: true,
2122
- get: function () { return chunkAK2VHUJV_cjs.deriveInitials; }
2122
+ get: function () { return chunkHE3N2ACJ_cjs.deriveInitials; }
2123
2123
  });
2124
2124
  Object.defineProperty(exports, "getChatLogger", {
2125
2125
  enumerable: true,
2126
- get: function () { return chunkAK2VHUJV_cjs.getChatLogger; }
2126
+ get: function () { return chunkHE3N2ACJ_cjs.getChatLogger; }
2127
2127
  });
2128
2128
  Object.defineProperty(exports, "initialState", {
2129
2129
  enumerable: true,
2130
- get: function () { return chunkAK2VHUJV_cjs.initialState; }
2130
+ get: function () { return chunkHE3N2ACJ_cjs.initialState; }
2131
2131
  });
2132
2132
  Object.defineProperty(exports, "reducer", {
2133
2133
  enumerable: true,
2134
- get: function () { return chunkAK2VHUJV_cjs.reducer; }
2134
+ get: function () { return chunkHE3N2ACJ_cjs.reducer; }
2135
2135
  });
2136
2136
  Object.defineProperty(exports, "resolvePersona", {
2137
2137
  enumerable: true,
2138
- get: function () { return chunkAK2VHUJV_cjs.resolvePersona; }
2138
+ get: function () { return chunkHE3N2ACJ_cjs.resolvePersona; }
2139
2139
  });
2140
2140
  Object.defineProperty(exports, "useChat", {
2141
2141
  enumerable: true,
2142
- get: function () { return chunkAK2VHUJV_cjs.useChat; }
2142
+ get: function () { return chunkHE3N2ACJ_cjs.useChat; }
2143
2143
  });
2144
2144
  Object.defineProperty(exports, "useChatAudio", {
2145
2145
  enumerable: true,
2146
- get: function () { return chunkAK2VHUJV_cjs.useChatAudio; }
2146
+ get: function () { return chunkHE3N2ACJ_cjs.useChatAudio; }
2147
2147
  });
2148
2148
  Object.defineProperty(exports, "useChatAudioPrefs", {
2149
2149
  enumerable: true,
2150
- get: function () { return chunkAK2VHUJV_cjs.useChatAudioPrefs; }
2150
+ get: function () { return chunkHE3N2ACJ_cjs.useChatAudioPrefs; }
2151
2151
  });
2152
2152
  Object.defineProperty(exports, "useChatComposer", {
2153
2153
  enumerable: true,
2154
- get: function () { return chunkAK2VHUJV_cjs.useChatComposer; }
2154
+ get: function () { return chunkHE3N2ACJ_cjs.useChatComposer; }
2155
2155
  });
2156
2156
  Object.defineProperty(exports, "useChatContext", {
2157
2157
  enumerable: true,
2158
- get: function () { return chunkAK2VHUJV_cjs.useChatContext; }
2158
+ get: function () { return chunkHE3N2ACJ_cjs.useChatContext; }
2159
2159
  });
2160
2160
  Object.defineProperty(exports, "useChatContextOptional", {
2161
2161
  enumerable: true,
2162
- get: function () { return chunkAK2VHUJV_cjs.useChatContextOptional; }
2162
+ get: function () { return chunkHE3N2ACJ_cjs.useChatContextOptional; }
2163
2163
  });
2164
2164
  Object.defineProperty(exports, "useChatHistory", {
2165
2165
  enumerable: true,
2166
- get: function () { return chunkAK2VHUJV_cjs.useChatHistory; }
2166
+ get: function () { return chunkHE3N2ACJ_cjs.useChatHistory; }
2167
2167
  });
2168
2168
  Object.defineProperty(exports, "useChatLayout", {
2169
2169
  enumerable: true,
2170
- get: function () { return chunkAK2VHUJV_cjs.useChatLayout; }
2170
+ get: function () { return chunkHE3N2ACJ_cjs.useChatLayout; }
2171
2171
  });
2172
2172
  Object.defineProperty(exports, "useChatScroll", {
2173
2173
  enumerable: true,
2174
- get: function () { return chunkAK2VHUJV_cjs.useChatScroll; }
2174
+ get: function () { return chunkHE3N2ACJ_cjs.useChatScroll; }
2175
2175
  });
2176
2176
  Object.defineProperty(exports, "TreeError", {
2177
2177
  enumerable: true,
package/dist/index.d.cts CHANGED
@@ -1685,7 +1685,6 @@ type ChatAction = {
1685
1685
  type: 'STREAM_START';
1686
1686
  id: string;
1687
1687
  createdAt?: number;
1688
- continueExisting?: boolean;
1689
1688
  } | {
1690
1689
  type: 'STREAM_CHUNK';
1691
1690
  delta: string;
@@ -1722,6 +1721,11 @@ type ChatAction = {
1722
1721
  type: 'STREAM_ERROR';
1723
1722
  id?: string;
1724
1723
  message: string;
1724
+ } | {
1725
+ type: 'STREAM_CANCEL_PLACEHOLDER';
1726
+ id: string;
1727
+ } | {
1728
+ type: 'STREAM_RESUME_EXISTING';
1725
1729
  } | {
1726
1730
  type: 'MESSAGE_EDIT';
1727
1731
  id: string;
package/dist/index.d.ts CHANGED
@@ -1685,7 +1685,6 @@ type ChatAction = {
1685
1685
  type: 'STREAM_START';
1686
1686
  id: string;
1687
1687
  createdAt?: number;
1688
- continueExisting?: boolean;
1689
1688
  } | {
1690
1689
  type: 'STREAM_CHUNK';
1691
1690
  delta: string;
@@ -1722,6 +1721,11 @@ type ChatAction = {
1722
1721
  type: 'STREAM_ERROR';
1723
1722
  id?: string;
1724
1723
  message: string;
1724
+ } | {
1725
+ type: 'STREAM_CANCEL_PLACEHOLDER';
1726
+ id: string;
1727
+ } | {
1728
+ type: 'STREAM_RESUME_EXISTING';
1725
1729
  } | {
1726
1730
  type: 'MESSAGE_EDIT';
1727
1731
  id: string;
package/dist/index.mjs CHANGED
@@ -5,8 +5,8 @@ export { NativeProvider, StreamProvider, VideoControls, VideoErrorFallback, Vide
5
5
  export { ImageViewer } from './chunk-OBRSGM64.mjs';
6
6
  export { generateContentKey, useAudioCache, useBlobUrlCleanup, useImageCache, useMediaCacheStore, useVideoCache, useVideoPlayerSettings } from './chunk-C6GXVH5J.mjs';
7
7
  export { CronSchedulerProvider, CustomInput, DayChips, MonthDayGrid, SchedulePreview, ScheduleTypeSelector, TimeSelector, buildCron, humanizeCron, isValidCron, parseCron, useCronCustom, useCronMonthDays, useCronPreview, useCronScheduler, useCronSchedulerContext, useCronTime, useCronType, useCronWeekDays } from './chunk-PVAX67JG.mjs';
8
- import { LIMITS, createId } from './chunk-D7ISTFUS.mjs';
9
- export { Attachments, AttachmentsGrid, AttachmentsList, CHAT_EVENT_NAME, CSS_VARS, ChatProvider, ChatRoot, Composer, DEFAULT_LABELS, DEFAULT_SIDEBAR, DEFAULT_Z_INDEX, EmptyState, ErrorBanner, HOTKEYS, JumpToLatest, LIMITS, MessageActions, MessageBubble, MessageList, STORAGE_KEYS, Sources, StreamingIndicator, ToolCalls, createId, createTokenBuffer, deriveInitials, getChatLogger, initialState, reducer, resolvePersona, useChat, useChatAudio, useChatAudioPrefs, useChatComposer, useChatContext, useChatContextOptional, useChatHistory, useChatLayout, useChatScroll } from './chunk-D7ISTFUS.mjs';
8
+ import { LIMITS, createId } from './chunk-OMUJNA42.mjs';
9
+ export { Attachments, AttachmentsGrid, AttachmentsList, CHAT_EVENT_NAME, CSS_VARS, ChatProvider, ChatRoot, Composer, DEFAULT_LABELS, DEFAULT_SIDEBAR, DEFAULT_Z_INDEX, EmptyState, ErrorBanner, HOTKEYS, JumpToLatest, LIMITS, MessageActions, MessageBubble, MessageList, STORAGE_KEYS, Sources, StreamingIndicator, ToolCalls, createId, createTokenBuffer, deriveInitials, getChatLogger, initialState, reducer, resolvePersona, useChat, useChatAudio, useChatAudioPrefs, useChatComposer, useChatContext, useChatContextOptional, useChatHistory, useChatLayout, useChatScroll } from './chunk-OMUJNA42.mjs';
10
10
  export { TreeError, TreeSkeleton, createDemoTree } from './chunk-B6IR5KSC.mjs';
11
11
  export { DEFAULT_TREE_APPEARANCE, DEFAULT_TREE_LABELS, TreeRoot as Tree, TreeChevron, TreeContent, TreeEmpty, TreeIcon, TreeIndentGuides, TreeLabel, TreeProvider, TreeRoot, TreeRow, TreeSearchInput, appearanceToStyle, clearTreeState, createChildCache, flattenTree, loadTreeState, resolveAppearance, resolveChildren, saveTreeState, useTreeActions, useTreeContext, useTreeExpansion, useTreeFocus, useTreeKeyboard, useTreeLabels, useTreeRows, useTreeSearch, useTreeSelection, useTreeTypeAhead } from './chunk-G5IEC7SR.mjs';
12
12
  import { PlaygroundProvider } from './chunk-ZUFTH5IR.mjs';
@@ -347,7 +347,7 @@ var LazyTree = createLazyComponent(
347
347
  }
348
348
  );
349
349
  var LazyChat = createLazyComponent(
350
- () => import('./ChatRoot-OURQVRLD.mjs').then((m) => ({ default: m.ChatRoot })),
350
+ () => import('./ChatRoot-APZXBHDT.mjs').then((m) => ({ default: m.ChatRoot })),
351
351
  {
352
352
  displayName: "LazyChat",
353
353
  fallback: /* @__PURE__ */ jsx(LoadingFallback, { minHeight: 320, text: "Loading chat\u2026" })
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@djangocfg/ui-tools",
3
- "version": "2.1.344",
3
+ "version": "2.1.345",
4
4
  "description": "Heavy React tools with lazy loading - for Electron, Vite, CRA, Next.js apps",
5
5
  "keywords": [
6
6
  "ui-tools",
@@ -101,8 +101,8 @@
101
101
  "check": "tsc --noEmit"
102
102
  },
103
103
  "peerDependencies": {
104
- "@djangocfg/i18n": "^2.1.344",
105
- "@djangocfg/ui-core": "^2.1.344",
104
+ "@djangocfg/i18n": "^2.1.345",
105
+ "@djangocfg/ui-core": "^2.1.345",
106
106
  "consola": "^3.4.2",
107
107
  "lodash-es": "^4.18.1",
108
108
  "lucide-react": "^0.545.0",
@@ -155,10 +155,10 @@
155
155
  "material-file-icons": "^2.4.0"
156
156
  },
157
157
  "devDependencies": {
158
- "@djangocfg/i18n": "^2.1.344",
158
+ "@djangocfg/i18n": "^2.1.345",
159
159
  "@djangocfg/playground": "workspace:*",
160
- "@djangocfg/typescript-config": "^2.1.344",
161
- "@djangocfg/ui-core": "^2.1.344",
160
+ "@djangocfg/typescript-config": "^2.1.345",
161
+ "@djangocfg/ui-core": "^2.1.345",
162
162
  "@types/lodash-es": "^4.17.12",
163
163
  "@types/mapbox__mapbox-gl-draw": "^1.4.8",
164
164
  "@types/node": "^24.7.2",
@@ -58,7 +58,7 @@ export type ChatAction =
58
58
  cursor: string | null;
59
59
  }
60
60
  | { type: 'MESSAGE_USER_ADD'; message: ChatMessage }
61
- | { type: 'STREAM_START'; id: string; createdAt?: number; continueExisting?: boolean }
61
+ | { type: 'STREAM_START'; id: string; createdAt?: number }
62
62
  | { type: 'STREAM_CHUNK'; delta: string }
63
63
  | { type: 'STREAM_TOOL_ACTIVITY'; tool: string }
64
64
  | { type: 'TOOL_CALL_START'; messageId: string; toolCall: ChatToolCall }
@@ -79,6 +79,8 @@ export type ChatAction =
79
79
  }
80
80
  | { type: 'STREAM_CANCELLED'; id: string; partialText: string; label?: string }
81
81
  | { type: 'STREAM_ERROR'; id?: string; message: string }
82
+ | { type: 'STREAM_CANCEL_PLACEHOLDER'; id: string }
83
+ | { type: 'STREAM_RESUME_EXISTING' }
82
84
  | { type: 'MESSAGE_EDIT'; id: string; content: string }
83
85
  | { type: 'MESSAGE_DELETE'; id: string }
84
86
  | { type: 'MESSAGES_CLEAR' }
@@ -173,22 +175,6 @@ export function reducer(state: ChatState, action: ChatAction): ChatState {
173
175
  }
174
176
  return state;
175
177
  }
176
- // continueExisting: resume an agent run — reuse the last assistant message
177
- // instead of creating a new empty placeholder. Used for HITL / approval-gate
178
- // flows where the same logical turn resumes after a human decision.
179
- if (action.continueExisting) {
180
- const lastAssistantIdx = (() => {
181
- for (let i = state.messages.length - 1; i >= 0; i--) {
182
- if (state.messages[i].role === 'assistant') return i;
183
- }
184
- return -1;
185
- })();
186
- if (lastAssistantIdx !== -1) {
187
- const msgs = state.messages.slice();
188
- msgs[lastAssistantIdx] = { ...msgs[lastAssistantIdx], isStreaming: true };
189
- return { ...state, isStreaming: true, messages: msgs };
190
- }
191
- }
192
178
  const placeholder: ChatMessage = {
193
179
  id: action.id,
194
180
  role: 'assistant',
@@ -295,6 +281,28 @@ export function reducer(state: ChatState, action: ChatAction): ChatState {
295
281
  return { ...state, isStreaming: false, error: action.message, messages };
296
282
  }
297
283
 
284
+ case 'STREAM_CANCEL_PLACEHOLDER':
285
+ // Remove the freshly-created empty assistant placeholder (resume path).
286
+ return {
287
+ ...state,
288
+ isStreaming: false,
289
+ messages: state.messages.filter((m) => m.id !== action.id),
290
+ };
291
+
292
+ case 'STREAM_RESUME_EXISTING': {
293
+ // Mark the last assistant message as streaming so resume chunks append to it.
294
+ const lastIdx = (() => {
295
+ for (let i = state.messages.length - 1; i >= 0; i--) {
296
+ if (state.messages[i].role === 'assistant') return i;
297
+ }
298
+ return -1;
299
+ })();
300
+ if (lastIdx === -1) return { ...state, isStreaming: true };
301
+ const msgs = state.messages.slice();
302
+ msgs[lastIdx] = { ...msgs[lastIdx], isStreaming: true };
303
+ return { ...state, isStreaming: true, messages: msgs };
304
+ }
305
+
298
306
  case 'MESSAGE_EDIT': {
299
307
  const messages = patchMessageById(state.messages, action.id, (m) => ({
300
308
  ...m,
@@ -207,30 +207,20 @@ export function useChat(config: UseChatConfig): UseChatReturn {
207
207
  const assistantId = createId('a');
208
208
  streamingMsgIdRef.current = assistantId;
209
209
 
210
- // Peek at the first event to detect resume streams before dispatching STREAM_START.
211
- // A resume stream begins with { type: 'resume_start' } — in that case we reuse
212
- // the last assistant message instead of creating a new empty placeholder.
213
210
  const iterator = transport.stream(sessionId, content, {
214
211
  signal: ctrl.signal,
215
212
  attachments,
216
213
  metadata: config.metadata,
217
214
  });
218
215
 
219
- let continueExisting = false;
216
+ // Peek at the first event — if it's `resume_start` we reuse the last assistant
217
+ // message instead of creating a new empty placeholder. The peek happens inside
218
+ // the try/catch so any transport error (network, 401, etc.) is handled normally.
220
219
  let peekedEvent: ChatStreamEvent | null = null;
221
- const firstResult = await iterator.next();
222
- if (!firstResult.done) {
223
- const ev = firstResult.value as ChatStreamEvent;
224
- if (ev.type === 'resume_start') {
225
- continueExisting = true;
226
- } else {
227
- peekedEvent = ev;
228
- }
229
- }
230
220
 
231
- dispatch({ type: 'STREAM_START', id: assistantId, continueExisting });
221
+ dispatch({ type: 'STREAM_START', id: assistantId });
232
222
  config.onStreamStart?.(assistantId);
233
- log.stream.info('start', { sessionId, assistantId, chars: content.length, continueExisting });
223
+ log.stream.info('start', { sessionId, assistantId, chars: content.length });
234
224
 
235
225
  const tokenBuffer = createTokenBuffer((delta) =>
236
226
  dispatch({ type: 'STREAM_CHUNK', delta }),
@@ -242,7 +232,21 @@ export function useChat(config: UseChatConfig): UseChatReturn {
242
232
  const t0 = performance.now();
243
233
 
244
234
  try {
245
- // iterator was already created above (for peek). Replay peeked event if any.
235
+ // Peek first event to detect resume_start must be inside try/catch so
236
+ // transport errors (network down, 401) are caught and shown as error banners.
237
+ const firstResult = await iterator.next();
238
+ if (!firstResult.done) {
239
+ const ev = firstResult.value as ChatStreamEvent;
240
+ if (ev.type === 'resume_start') {
241
+ // Switch existing placeholder to continueExisting mode: remove the
242
+ // just-created empty placeholder and mark the last assistant msg streaming.
243
+ dispatch({ type: 'STREAM_CANCEL_PLACEHOLDER', id: assistantId });
244
+ dispatch({ type: 'STREAM_RESUME_EXISTING' });
245
+ } else {
246
+ peekedEvent = ev;
247
+ }
248
+ }
249
+
246
250
  if (peekedEvent) handleEvent(peekedEvent);
247
251
 
248
252
  for await (const ev of iterator) {
@@ -1,14 +0,0 @@
1
- 'use strict';
2
-
3
- var chunkAK2VHUJV_cjs = require('./chunk-AK2VHUJV.cjs');
4
- require('./chunk-B5AWZOHJ.cjs');
5
- require('./chunk-OLISEQHS.cjs');
6
-
7
-
8
-
9
- Object.defineProperty(exports, "ChatRoot", {
10
- enumerable: true,
11
- get: function () { return chunkAK2VHUJV_cjs.ChatRoot; }
12
- });
13
- //# sourceMappingURL=ChatRoot-CIOFUESH.cjs.map
14
- //# sourceMappingURL=ChatRoot-CIOFUESH.cjs.map
@@ -1,5 +0,0 @@
1
- export { ChatRoot } from './chunk-D7ISTFUS.mjs';
2
- import './chunk-2ZLKZ5VR.mjs';
3
- import './chunk-N2XQF2OL.mjs';
4
- //# sourceMappingURL=ChatRoot-OURQVRLD.mjs.map
5
- //# sourceMappingURL=ChatRoot-OURQVRLD.mjs.map