@djangocfg/ui-tools 2.1.342 → 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 chunk4C4FGPZV_cjs = require('./chunk-4C4FGPZV.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-XJMWMJVD.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 ?? chunk4C4FGPZV_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: chunk4C4FGPZV_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: chunk4C4FGPZV_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 = chunk4C4FGPZV_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: chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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 chunk4C4FGPZV_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
@@ -1553,6 +1553,8 @@ type ChatStreamEvent = {
1553
1553
  type: 'message_start';
1554
1554
  messageId: string;
1555
1555
  sessionId: string;
1556
+ } | {
1557
+ type: 'resume_start';
1556
1558
  } | {
1557
1559
  type: 'chunk';
1558
1560
  delta: string;
@@ -1719,6 +1721,11 @@ type ChatAction = {
1719
1721
  type: 'STREAM_ERROR';
1720
1722
  id?: string;
1721
1723
  message: string;
1724
+ } | {
1725
+ type: 'STREAM_CANCEL_PLACEHOLDER';
1726
+ id: string;
1727
+ } | {
1728
+ type: 'STREAM_RESUME_EXISTING';
1722
1729
  } | {
1723
1730
  type: 'MESSAGE_EDIT';
1724
1731
  id: string;
package/dist/index.d.ts CHANGED
@@ -1553,6 +1553,8 @@ type ChatStreamEvent = {
1553
1553
  type: 'message_start';
1554
1554
  messageId: string;
1555
1555
  sessionId: string;
1556
+ } | {
1557
+ type: 'resume_start';
1556
1558
  } | {
1557
1559
  type: 'chunk';
1558
1560
  delta: string;
@@ -1719,6 +1721,11 @@ type ChatAction = {
1719
1721
  type: 'STREAM_ERROR';
1720
1722
  id?: string;
1721
1723
  message: string;
1724
+ } | {
1725
+ type: 'STREAM_CANCEL_PLACEHOLDER';
1726
+ id: string;
1727
+ } | {
1728
+ type: 'STREAM_RESUME_EXISTING';
1722
1729
  } | {
1723
1730
  type: 'MESSAGE_EDIT';
1724
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-CJ76BM2N.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-CJ76BM2N.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-AFBNSI2A.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.342",
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.342",
105
- "@djangocfg/ui-core": "^2.1.342",
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.342",
158
+ "@djangocfg/i18n": "^2.1.345",
159
159
  "@djangocfg/playground": "workspace:*",
160
- "@djangocfg/typescript-config": "^2.1.342",
161
- "@djangocfg/ui-core": "^2.1.342",
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",
@@ -941,7 +941,80 @@ export const MultiUser = () => (
941
941
  );
942
942
 
943
943
  // ---------------------------------------------------------------------------
944
- // 13) Playgroundknobs
944
+ // 13) WithHideComposeragent-pause pattern (approval gate, human-in-the-loop)
945
+ // ---------------------------------------------------------------------------
946
+
947
+ export const WithHideComposer = () => {
948
+ const [paused, setPaused] = useState(false);
949
+
950
+ const transport = useMemo(
951
+ () =>
952
+ createMockTransport({
953
+ replies: [
954
+ 'Sure — I will send that email. Waiting for your approval before proceeding.',
955
+ ],
956
+ latencyMs: 30,
957
+ }),
958
+ [],
959
+ );
960
+
961
+ return (
962
+ <Frame h={480}>
963
+ <ChatRoot
964
+ transport={transport}
965
+ config={{
966
+ greeting: 'Agent-pause demo',
967
+ placeholder: 'Ask to do something requiring approval…',
968
+ }}
969
+ hideComposer={paused}
970
+ footer={
971
+ paused ? (
972
+ <div className="flex flex-col gap-2 border-t bg-background px-3 py-2">
973
+ <div className="rounded-lg border border-amber-200 bg-amber-50 dark:border-amber-800 dark:bg-amber-950/40 p-3 text-sm">
974
+ <div className="mb-1 font-medium text-amber-900 dark:text-amber-200">
975
+ Approval required
976
+ <span className="ml-2 font-mono text-xs text-amber-700 dark:text-amber-400">send_email</span>
977
+ </div>
978
+ <pre className="mb-2 rounded bg-muted px-2 py-1.5 text-[11px] text-muted-foreground">
979
+ {`{ "to": "mark@example.com", "subject": "Hello" }`}
980
+ </pre>
981
+ <div className="flex gap-2">
982
+ <button
983
+ type="button"
984
+ onClick={() => setPaused(false)}
985
+ className="flex-1 rounded bg-amber-600 hover:bg-amber-700 px-3 py-1.5 text-xs font-medium text-white transition-colors"
986
+ >
987
+ Approve
988
+ </button>
989
+ <button
990
+ type="button"
991
+ onClick={() => setPaused(false)}
992
+ className="flex-1 rounded border border-amber-300 hover:bg-amber-100 px-3 py-1.5 text-xs font-medium text-amber-800 transition-colors"
993
+ >
994
+ Deny
995
+ </button>
996
+ </div>
997
+ </div>
998
+ </div>
999
+ ) : (
1000
+ <div className="border-t border-border bg-muted/20 px-3 py-1.5 text-center">
1001
+ <button
1002
+ type="button"
1003
+ onClick={() => setPaused(true)}
1004
+ className="text-[11px] text-muted-foreground hover:text-foreground underline"
1005
+ >
1006
+ Simulate approval gate
1007
+ </button>
1008
+ </div>
1009
+ )
1010
+ }
1011
+ />
1012
+ </Frame>
1013
+ );
1014
+ };
1015
+
1016
+ // ---------------------------------------------------------------------------
1017
+ // 14) Playground — knobs
945
1018
  // ---------------------------------------------------------------------------
946
1019
 
947
1020
  export const Playground = () => {
@@ -93,6 +93,9 @@ Module boundaries:
93
93
  renderInput: (v) => <LazyJsonTree data={v} mode="compact" />,
94
94
  renderOutput: (v) => <LazyJsonTree data={v} mode="compact" />,
95
95
  }}
96
+
97
+ // Hide composer while agent is paused waiting for human input
98
+ hideComposer={hasPendingApprovals}
96
99
  />
97
100
  ```
98
101
 
@@ -106,14 +109,32 @@ Module boundaries:
106
109
  ├───────────────────────────────┤
107
110
  │ messages (jumpToLatest) │
108
111
  ├───────────────────────────────┤
109
- │ composerToolbarStart │
112
+ │ composerToolbarStart │ ← hidden when hideComposer={true}
110
113
  │ [textarea] composerToolbarEnd │
111
114
  │ composerAttachmentTray │
112
115
  ├───────────────────────────────┤
113
- │ footer │
116
+ │ footer │ ← use for approval panels, disclaimers
114
117
  └───────────────────────────────┘
115
118
  ```
116
119
 
120
+ ### hideComposer — agent-pause / human-in-the-loop
121
+
122
+ Set `hideComposer={true}` to remove the composer while the agent is waiting for a human decision (approval gate, HITL). Combine with `footer` to show action buttons:
123
+
124
+ ```tsx
125
+ <ChatRoot
126
+ transport={transport}
127
+ hideComposer={pendingActions.length > 0}
128
+ footer={
129
+ pendingActions.length > 0 ? (
130
+ <AgentActions actions={pendingActions} onResult={handleResult} />
131
+ ) : null
132
+ }
133
+ />
134
+ ```
135
+
136
+ The composer reappears automatically once `hideComposer` returns `false`.
137
+
117
138
  ### ToolCalls — expand behavior
118
139
 
119
140
  Panels are **collapsed by default**. While a tool is `running`, the panel auto-expands so you can see live `streamingText`; on completion it auto-collapses again. Manual user toggles (open or close) are remembered and override the auto-behavior for that call.
@@ -525,4 +546,5 @@ Full implementation plan and rationale lives at [`@dev/@refactoring7-chat/`](../
525
546
  - `WithMapPayload` — `dispatchToolPayload` with `LazyJsonTree` fallback
526
547
  - `WithPersonas` — config-level `user` + `assistant` identity (avatar, name)
527
548
  - `MultiUser` — per-message `sender` overrides (multi-user / multi-bot)
549
+ - `WithHideComposer` — `hideComposer` + `footer` approval gate (HITL / agent-pause pattern)
528
550
  - `Playground` — knobs (latency, streaming, suggestions)
@@ -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' }
@@ -168,7 +170,6 @@ export function reducer(state: ChatState, action: ChatAction): ChatState {
168
170
  case 'STREAM_START': {
169
171
  if (state.isStreaming) {
170
172
  if (typeof console !== 'undefined') {
171
- // Soft guard — UI should disable the composer; this catches bugs.
172
173
  // eslint-disable-next-line no-console
173
174
  console.warn('[chat] STREAM_START while already streaming, ignoring');
174
175
  }
@@ -280,6 +281,28 @@ export function reducer(state: ChatState, action: ChatAction): ChatState {
280
281
  return { ...state, isStreaming: false, error: action.message, messages };
281
282
  }
282
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
+
283
306
  case 'MESSAGE_EDIT': {
284
307
  const messages = patchMessageById(state.messages, action.id, (m) => ({
285
308
  ...m,
@@ -207,6 +207,17 @@ export function useChat(config: UseChatConfig): UseChatReturn {
207
207
  const assistantId = createId('a');
208
208
  streamingMsgIdRef.current = assistantId;
209
209
 
210
+ const iterator = transport.stream(sessionId, content, {
211
+ signal: ctrl.signal,
212
+ attachments,
213
+ metadata: config.metadata,
214
+ });
215
+
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.
219
+ let peekedEvent: ChatStreamEvent | null = null;
220
+
210
221
  dispatch({ type: 'STREAM_START', id: assistantId });
211
222
  config.onStreamStart?.(assistantId);
212
223
  log.stream.info('start', { sessionId, assistantId, chars: content.length });
@@ -221,11 +232,22 @@ export function useChat(config: UseChatConfig): UseChatReturn {
221
232
  const t0 = performance.now();
222
233
 
223
234
  try {
224
- const iterator = transport.stream(sessionId, content, {
225
- signal: ctrl.signal,
226
- attachments,
227
- metadata: config.metadata,
228
- });
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
+
250
+ if (peekedEvent) handleEvent(peekedEvent);
229
251
 
230
252
  for await (const ev of iterator) {
231
253
  if (ctrl.signal.aborted) break;
@@ -200,6 +200,7 @@ export interface SendOptions {
200
200
 
201
201
  export type ChatStreamEvent =
202
202
  | { type: 'message_start'; messageId: string; sessionId: string }
203
+ | { type: 'resume_start' }
203
204
  | { type: 'chunk'; delta: string }
204
205
  | { type: 'tool_activity'; tool: string; status: string }
205
206
  | {
@@ -1,5 +0,0 @@
1
- export { ChatRoot } from './chunk-CJ76BM2N.mjs';
2
- import './chunk-2ZLKZ5VR.mjs';
3
- import './chunk-N2XQF2OL.mjs';
4
- //# sourceMappingURL=ChatRoot-AFBNSI2A.mjs.map
5
- //# sourceMappingURL=ChatRoot-AFBNSI2A.mjs.map
@@ -1,14 +0,0 @@
1
- 'use strict';
2
-
3
- var chunk4C4FGPZV_cjs = require('./chunk-4C4FGPZV.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 chunk4C4FGPZV_cjs.ChatRoot; }
12
- });
13
- //# sourceMappingURL=ChatRoot-XJMWMJVD.cjs.map
14
- //# sourceMappingURL=ChatRoot-XJMWMJVD.cjs.map