@super_studio/ecforce-ai-agent-react 1.1.2 → 1.2.0-canary.1

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/README.md CHANGED
@@ -220,6 +220,100 @@ sendAppMessage({
220
220
 
221
221
  これにより、AI Agent 側で現在の画面や対象データに応じた応答をしやすくなります。
222
222
 
223
+ ### 3-5. `useClientTool()` でクライアントツールを登録する
224
+
225
+ ブラウザ上で実行したい処理を AI Agent から呼ばせたい場合は、`useClientTool()` を使います。
226
+
227
+ 典型的には、以下のような用途で使います。
228
+
229
+ - 現在開いている画面の React state を参照して結果を返す
230
+ - ユーザー確認付きでクライアント側の state を更新する
231
+ - クライアント側の UI 操作をトリガーする
232
+
233
+ `useClientTool()` は `ChatbotProvider` 配下で呼び出してください。引数定義には `zod` を使います。
234
+
235
+ ```tsx
236
+ "use client";
237
+
238
+ import { useState } from "react";
239
+ import { z } from "zod";
240
+ import { useClientTool } from "@super_studio/ecforce-ai-agent-react";
241
+
242
+ export function CartToolRegistration() {
243
+ const [selectedSku, setSelectedSku] = useState("sku_001");
244
+ const [isPinned, setIsPinned] = useState(false);
245
+
246
+ useClientTool({
247
+ name: "checkSelectedSku",
248
+ displayName: "選択中SKU確認",
249
+ description: "現在画面で選択中の商品 SKU を返します。",
250
+ summary: "選択中の SKU を返します。",
251
+ parameters: z.object({}),
252
+ execute: () => {
253
+ return {
254
+ selectedSku,
255
+ isPinned,
256
+ };
257
+ },
258
+ });
259
+
260
+ useClientTool({
261
+ name: "togglePinnedState",
262
+ displayName: "ピン留め切り替え",
263
+ description: "現在の商品をピン留めするかどうかを切り替えます。",
264
+ summary: "ピン留め状態を更新します。",
265
+ requireConfirmation: true,
266
+ parameters: z.object({
267
+ pinned: z.boolean().describe("更新後のピン留め状態"),
268
+ }),
269
+ execute: ({ pinned }) => {
270
+ setIsPinned(pinned);
271
+
272
+ return {
273
+ selectedSku,
274
+ isPinned: pinned,
275
+ };
276
+ },
277
+ });
278
+
279
+ return (
280
+ <div>
281
+ <p>selectedSku: {selectedSku}</p>
282
+ <p>isPinned: {String(isPinned)}</p>
283
+ <button onClick={() => setSelectedSku("sku_002")}>SKU を切り替える</button>
284
+ </div>
285
+ );
286
+ }
287
+ ```
288
+
289
+ #### `useClientTool()` の options
290
+
291
+ ```ts
292
+ type UseClientToolOptions<TParameters extends z.ZodTypeAny> = {
293
+ name: string;
294
+ displayName: string;
295
+ description: string;
296
+ summary?: string;
297
+ requireConfirmation?: boolean;
298
+ parameters: TParameters;
299
+ execute: (args: z.output<TParameters>) => Promise<unknown> | unknown;
300
+ };
301
+ ```
302
+
303
+ - `name`: ツール識別子です。内部では自動で `page_` プレフィックスが付きます
304
+ - `displayName`: UI 上で表示するツール名です
305
+ - `description`: AI Agent が参照する詳細説明です。何をするツールか、どんな引数を受けるかを明確に書いてください
306
+ - `summary`: ユーザー向けの短い要約です
307
+ - `requireConfirmation`: `true` の場合、実行前にユーザー確認を要求します
308
+ - `parameters`: ツール引数を表す `zod` スキーマです
309
+ - `execute`: 検証済み引数を受け取ってクライアント側の処理を実行する関数です
310
+
311
+ #### 補足
312
+
313
+ - `name` に `page_` を自分で付けても動きますが、通常は素の名前で問題ありません
314
+ - `execute` の返り値は AI Agent に渡されるため、JSON 化しやすい値を返すのがおすすめです
315
+ - コンポーネントがアンマウントされるとツール登録は自動で解除されます
316
+
223
317
  ## 4. `@super_studio/ecforce-ai-agent-server` のセットアップ
224
318
 
225
319
  ### 4-1. サーバー側で AI Agent セッションを発行する
@@ -1,4 +1,4 @@
1
- import * as react_jsx_runtime1 from "react/jsx-runtime";
1
+ import * as react_jsx_runtime0 from "react/jsx-runtime";
2
2
 
3
3
  //#region src/components/chatbot-frame.d.ts
4
4
  type ChatbotFrameProps = {
@@ -37,7 +37,7 @@ declare function ChatbotFrame({
37
37
  enabled,
38
38
  url,
39
39
  className
40
- }: ChatbotFrameProps): react_jsx_runtime1.JSX.Element;
40
+ }: ChatbotFrameProps): react_jsx_runtime0.JSX.Element;
41
41
  type SessionData = {
42
42
  token: string;
43
43
  expiresAt: string;
@@ -1,6 +1,6 @@
1
1
  import { ChatbotFrameProps } from "./chatbot-frame.mjs";
2
2
  import React from "react";
3
- import * as react_jsx_runtime2 from "react/jsx-runtime";
3
+ import * as react_jsx_runtime1 from "react/jsx-runtime";
4
4
 
5
5
  //#region src/components/chatbot-sheet.d.ts
6
6
  type ChatbotSheetProps = ChatbotFrameProps & {
@@ -9,7 +9,7 @@ type ChatbotSheetProps = ChatbotFrameProps & {
9
9
  declare function ChatbotSheet({
10
10
  sheetStyle,
11
11
  ...props
12
- }: ChatbotSheetProps): react_jsx_runtime2.JSX.Element;
12
+ }: ChatbotSheetProps): react_jsx_runtime1.JSX.Element;
13
13
  //#endregion
14
14
  export { ChatbotSheet };
15
15
  //# sourceMappingURL=chatbot-sheet.d.mts.map
@@ -1,3 +1,4 @@
1
+ import { RegisteredClientTool } from "../../lib/client-tools.mjs";
1
2
  import { InitProps, MCP, SendAppMessagePayload } from "./use-chatbot-frame-handler.mjs";
2
3
  import React from "react";
3
4
  import * as react_jsx_runtime0 from "react/jsx-runtime";
@@ -28,6 +29,10 @@ type ChatbotContextType = {
28
29
  setSessionToken: (sessionToken: string) => void;
29
30
  /** アプリからメッセージを送信 */
30
31
  sendAppMessage: (payload: SendAppMessagePayload) => void;
32
+ /** Client Toolを登録 */
33
+ registerClientTool: (tool: RegisteredClientTool) => void;
34
+ /** Client Toolを解除 */
35
+ unregisterClientTool: (name: string) => void;
31
36
  /** チャットボットが準備できたか */
32
37
  isReady: boolean;
33
38
  /** チャットボットが初期化されたか */
@@ -38,6 +43,34 @@ type ChatbotContextType = {
38
43
  */
39
44
  setIframeEl: (iframeEl: HTMLIFrameElement | null) => void;
40
45
  };
46
+ /**
47
+ * `ChatbotProvider` が提供するチャットボット状態と操作 API を取得します。
48
+ *
49
+ * @returns チャットボットの表示状態、初期化 API、Client Tool の登録 API を含むコンテキスト値。
50
+ *
51
+ * @example
52
+ * ```tsx
53
+ * import { useChatbot } from "@super_studio/ecforce-ai-agent-react";
54
+ *
55
+ * export function OpenChatButton() {
56
+ * const { open, setOpen, sendAppMessage } = useChatbot();
57
+ *
58
+ * return (
59
+ * <button
60
+ * type="button"
61
+ * onClick={() => {
62
+ * setOpen(!open);
63
+ * sendAppMessage({
64
+ * message: "EC Forceの注文状況を確認したいです",
65
+ * });
66
+ * }}
67
+ * >
68
+ * Open chat
69
+ * </button>
70
+ * );
71
+ * }
72
+ * ```
73
+ */
41
74
  declare function useChatbot(): ChatbotContextType;
42
75
  type ChatbotProviderProps = {
43
76
  children: React.ReactNode;
@@ -1 +1 @@
1
- {"version":3,"file":"chatbot-provider.d.mts","names":[],"sources":["../../../src/components/provider/chatbot-provider.tsx"],"sourcesContent":[],"mappings":";;;;;KAWK,kBAAA;;;EAAA;EAgBW,IAAA,EAAA,OAAA;EAEE;EAMU,OAAA,EAAA,CAAA,IAAA,EAAA,OAAA,EAAA,GAAA,IAAA;EASF;EAAiB,UAAA,EAAA,OAAA;EAO3B;EAQX,aAAA,EAAA,CAAA,QAAoB,EAAA,OAAA,EACb,GAAA,IAAM;EAGF;EAAkB,YAAA,EAAA,OAAA;EAAY;EAAoB,eAAA,EAAA,CAAA,UAAA,EAAA,OAAA,EAAA,GAAA,IAAA;EAAA;gBApClD;;kBAEE;;;;;;4BAMU;;;;;;;;;0BASF;;iBAOV,UAAA,CAAA,GAAU;KAQrB,oBAAA;YACO,KAAA,CAAM;;iBAGF,eAAA;;GAA8B,uBAAoB,kBAAA,CAAA,GAAA,CAAA"}
1
+ {"version":3,"file":"chatbot-provider.d.mts","names":[],"sources":["../../../src/components/provider/chatbot-provider.tsx"],"sourcesContent":[],"mappings":";;;;;;KAYK,kBAAA;;;EAAA;EAgBW,IAAA,EAAA,OAAA;EAEE;EAMU,OAAA,EAAA,CAAA,IAAA,EAAA,OAAA,EAAA,GAAA,IAAA;EAEC;EAWH,UAAA,EAAA,OAAA;EAAiB;EAmC3B,aAAU,EAAA,CAAA,QAAA,EAAA,OAAA,EAAA,GAAA,IAAA;EAQrB;EAIW,YAAA,EAAA,OAAe;EAAG;EAAY,eAAA,EAAA,CAAA,UAAA,EAAA,OAAA,EAAA,GAAA,IAAA;EAAoB;EAAA,IAAA,EAAA,CAAA,KAAA,EApElD,SAoEkD,EAAA,GAAA,IAAA;;kBAlEhD;;;;;;4BAMU;;6BAEC;;;;;;;;;;;0BAWH;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAmCV,UAAA,CAAA,GAAU;KAQrB,oBAAA;YACO,KAAA,CAAM;;iBAGF,eAAA;;GAA8B,uBAAoB,kBAAA,CAAA,GAAA,CAAA"}
@@ -7,6 +7,34 @@ import { jsx } from "react/jsx-runtime";
7
7
 
8
8
  //#region src/components/provider/chatbot-provider.tsx
9
9
  const ChatbotContext = React.createContext(void 0);
10
+ /**
11
+ * `ChatbotProvider` が提供するチャットボット状態と操作 API を取得します。
12
+ *
13
+ * @returns チャットボットの表示状態、初期化 API、Client Tool の登録 API を含むコンテキスト値。
14
+ *
15
+ * @example
16
+ * ```tsx
17
+ * import { useChatbot } from "@super_studio/ecforce-ai-agent-react";
18
+ *
19
+ * export function OpenChatButton() {
20
+ * const { open, setOpen, sendAppMessage } = useChatbot();
21
+ *
22
+ * return (
23
+ * <button
24
+ * type="button"
25
+ * onClick={() => {
26
+ * setOpen(!open);
27
+ * sendAppMessage({
28
+ * message: "EC Forceの注文状況を確認したいです",
29
+ * });
30
+ * }}
31
+ * >
32
+ * Open chat
33
+ * </button>
34
+ * );
35
+ * }
36
+ * ```
37
+ */
10
38
  function useChatbot() {
11
39
  const context = React.useContext(ChatbotContext);
12
40
  if (!context) throw new Error("useChatbot must be used within a ChatbotProvider");
@@ -14,7 +42,7 @@ function useChatbot() {
14
42
  }
15
43
  function ChatbotProvider({ children }) {
16
44
  const { hasOpened, open, isExpanded, isFullScreen, setOpen, setIsExpanded, setIsFullScreen } = useChatbotWindowStates();
17
- const { setMcps, setAppName, setSessionToken, sendAppMessage, init, isReady, isInitialized, setIframeEl } = useChatbotFrameHandler({
45
+ const { setMcps, setAppName, setSessionToken, sendAppMessage, registerClientTool, unregisterClientTool, init, isReady, isInitialized, setIframeEl } = useChatbotFrameHandler({
18
46
  setOpen,
19
47
  setIsExpanded,
20
48
  setIsFullScreen
@@ -29,6 +57,8 @@ function ChatbotProvider({ children }) {
29
57
  setIsFullScreen,
30
58
  setSessionToken,
31
59
  sendAppMessage,
60
+ registerClientTool,
61
+ unregisterClientTool,
32
62
  init,
33
63
  setMcps,
34
64
  setAppName,
@@ -1 +1 @@
1
- {"version":3,"file":"chatbot-provider.mjs","names":[],"sources":["../../../src/components/provider/chatbot-provider.tsx"],"sourcesContent":["\"use client\";\n\nimport React from \"react\";\nimport {\n type SendAppMessagePayload,\n useChatbotFrameHandler,\n type InitProps,\n type MCP,\n} from \"./use-chatbot-frame-handler\";\nimport { useChatbotWindowStates } from \"./use-chatbot-window-states\";\n\ntype ChatbotContextType = {\n /** 1回チャットボットを開いたら、固定でtrueになる */\n hasOpened: boolean;\n /** チャットボットを開いているか */\n open: boolean;\n /** チャットボットを開く */\n setOpen: (open: boolean) => void;\n /** チャットボットを展開しているか */\n isExpanded: boolean;\n /** チャットボットを展開する */\n setIsExpanded: (expanded: boolean) => void;\n /** チャットボットをフルスクリーンにしているか */\n isFullScreen: boolean;\n /** チャットボットをフルスクリーンにする */\n setIsFullScreen: (fullScreen: boolean) => void;\n /** チャットボットを初期化 */\n init: (props: InitProps) => void;\n /** MCPを設定 */\n setMcps: (mcps: MCP[]) => void;\n /** アプリ名を設定 */\n setAppName: (appName: string) => void;\n /** セッショントークンを設定 */\n setSessionToken: (sessionToken: string) => void;\n /** アプリからメッセージを送信 */\n sendAppMessage: (payload: SendAppMessagePayload) => void;\n /** チャットボットが準備できたか */\n isReady: boolean;\n /** チャットボットが初期化されたか */\n isInitialized: boolean;\n /**\n * @internal\n * チャットボットのiframe要素を設定\n */\n setIframeEl: (iframeEl: HTMLIFrameElement | null) => void;\n};\n\nconst ChatbotContext = React.createContext<ChatbotContextType | undefined>(\n undefined,\n);\n\nexport function useChatbot() {\n const context = React.useContext(ChatbotContext);\n if (!context) {\n throw new Error(\"useChatbot must be used within a ChatbotProvider\");\n }\n return context;\n}\n\ntype ChatbotProviderProps = {\n children: React.ReactNode;\n};\n\nexport function ChatbotProvider({ children }: ChatbotProviderProps) {\n const {\n hasOpened,\n open,\n isExpanded,\n isFullScreen,\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n } = useChatbotWindowStates();\n const {\n setMcps,\n setAppName,\n setSessionToken,\n sendAppMessage,\n init,\n isReady,\n isInitialized,\n setIframeEl,\n } = useChatbotFrameHandler({\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n });\n\n const value = {\n hasOpened,\n open,\n setOpen,\n isExpanded,\n setIsExpanded,\n isFullScreen,\n setIsFullScreen,\n setSessionToken,\n sendAppMessage,\n init,\n setMcps,\n setAppName,\n isReady,\n isInitialized,\n setIframeEl,\n };\n\n return (\n <ChatbotContext.Provider value={value}>{children}</ChatbotContext.Provider>\n );\n}\n"],"mappings":";;;;;;;;AA+CA,MAAM,iBAAiB,MAAM,cAC3B,OACD;AAED,SAAgB,aAAa;CAC3B,MAAM,UAAU,MAAM,WAAW,eAAe;AAChD,KAAI,CAAC,QACH,OAAM,IAAI,MAAM,mDAAmD;AAErE,QAAO;;AAOT,SAAgB,gBAAgB,EAAE,YAAkC;CAClE,MAAM,EACJ,WACA,MACA,YACA,cACA,SACA,eACA,oBACE,wBAAwB;CAC5B,MAAM,EACJ,SACA,YACA,iBACA,gBACA,MACA,SACA,eACA,gBACE,uBAAuB;EACzB;EACA;EACA;EACD,CAAC;CAEF,MAAM,QAAQ;EACZ;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD;AAED,QACE,oBAAC,eAAe;EAAgB;EAAQ;GAAmC"}
1
+ {"version":3,"file":"chatbot-provider.mjs","names":[],"sources":["../../../src/components/provider/chatbot-provider.tsx"],"sourcesContent":["\"use client\";\n\nimport React from \"react\";\nimport type { RegisteredClientTool } from \"../../lib/client-tools\";\nimport {\n useChatbotFrameHandler,\n type InitProps,\n type MCP,\n type SendAppMessagePayload,\n} from \"./use-chatbot-frame-handler\";\nimport { useChatbotWindowStates } from \"./use-chatbot-window-states\";\n\ntype ChatbotContextType = {\n /** 1回チャットボットを開いたら、固定でtrueになる */\n hasOpened: boolean;\n /** チャットボットを開いているか */\n open: boolean;\n /** チャットボットを開く */\n setOpen: (open: boolean) => void;\n /** チャットボットを展開しているか */\n isExpanded: boolean;\n /** チャットボットを展開する */\n setIsExpanded: (expanded: boolean) => void;\n /** チャットボットをフルスクリーンにしているか */\n isFullScreen: boolean;\n /** チャットボットをフルスクリーンにする */\n setIsFullScreen: (fullScreen: boolean) => void;\n /** チャットボットを初期化 */\n init: (props: InitProps) => void;\n /** MCPを設定 */\n setMcps: (mcps: MCP[]) => void;\n /** アプリ名を設定 */\n setAppName: (appName: string) => void;\n /** セッショントークンを設定 */\n setSessionToken: (sessionToken: string) => void;\n /** アプリからメッセージを送信 */\n sendAppMessage: (payload: SendAppMessagePayload) => void;\n /** Client Toolを登録 */\n registerClientTool: (tool: RegisteredClientTool) => void;\n /** Client Toolを解除 */\n unregisterClientTool: (name: string) => void;\n /** チャットボットが準備できたか */\n isReady: boolean;\n /** チャットボットが初期化されたか */\n isInitialized: boolean;\n /**\n * @internal\n * チャットボットのiframe要素を設定\n */\n setIframeEl: (iframeEl: HTMLIFrameElement | null) => void;\n};\n\nconst ChatbotContext = React.createContext<ChatbotContextType | undefined>(\n undefined,\n);\n\n/**\n * `ChatbotProvider` が提供するチャットボット状態と操作 API を取得します。\n *\n * @returns チャットボットの表示状態、初期化 API、Client Tool の登録 API を含むコンテキスト値。\n *\n * @example\n * ```tsx\n * import { useChatbot } from \"@super_studio/ecforce-ai-agent-react\";\n *\n * export function OpenChatButton() {\n * const { open, setOpen, sendAppMessage } = useChatbot();\n *\n * return (\n * <button\n * type=\"button\"\n * onClick={() => {\n * setOpen(!open);\n * sendAppMessage({\n * message: \"EC Forceの注文状況を確認したいです\",\n * });\n * }}\n * >\n * Open chat\n * </button>\n * );\n * }\n * ```\n */\nexport function useChatbot() {\n const context = React.useContext(ChatbotContext);\n if (!context) {\n throw new Error(\"useChatbot must be used within a ChatbotProvider\");\n }\n return context;\n}\n\ntype ChatbotProviderProps = {\n children: React.ReactNode;\n};\n\nexport function ChatbotProvider({ children }: ChatbotProviderProps) {\n const {\n hasOpened,\n open,\n isExpanded,\n isFullScreen,\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n } = useChatbotWindowStates();\n const {\n setMcps,\n setAppName,\n setSessionToken,\n sendAppMessage,\n registerClientTool,\n unregisterClientTool,\n init,\n isReady,\n isInitialized,\n setIframeEl,\n } = useChatbotFrameHandler({\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n });\n\n const value = {\n hasOpened,\n open,\n setOpen,\n isExpanded,\n setIsExpanded,\n isFullScreen,\n setIsFullScreen,\n setSessionToken,\n sendAppMessage,\n registerClientTool,\n unregisterClientTool,\n init,\n setMcps,\n setAppName,\n isReady,\n isInitialized,\n setIframeEl,\n };\n\n return (\n <ChatbotContext.Provider value={value}>{children}</ChatbotContext.Provider>\n );\n}\n"],"mappings":";;;;;;;;AAoDA,MAAM,iBAAiB,MAAM,cAC3B,OACD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA8BD,SAAgB,aAAa;CAC3B,MAAM,UAAU,MAAM,WAAW,eAAe;AAChD,KAAI,CAAC,QACH,OAAM,IAAI,MAAM,mDAAmD;AAErE,QAAO;;AAOT,SAAgB,gBAAgB,EAAE,YAAkC;CAClE,MAAM,EACJ,WACA,MACA,YACA,cACA,SACA,eACA,oBACE,wBAAwB;CAC5B,MAAM,EACJ,SACA,YACA,iBACA,gBACA,oBACA,sBACA,MACA,SACA,eACA,gBACE,uBAAuB;EACzB;EACA;EACA;EACD,CAAC;CAEF,MAAM,QAAQ;EACZ;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD;AAED,QACE,oBAAC,eAAe;EAAgB;EAAQ;GAAmC"}
@@ -1,3 +1,4 @@
1
+ import "../../lib/client-tools.mjs";
1
2
  import React from "react";
2
3
 
3
4
  //#region src/components/provider/use-chatbot-frame-handler.d.ts
@@ -13,15 +14,23 @@ type InitProps = {
13
14
  sessionToken?: string;
14
15
  };
15
16
  type AppMessageAttachment = {
17
+ /** 添付ファイルの表示名です。 */
16
18
  fileName: string;
19
+ /** 添付ファイルの MIME type です。 */
17
20
  mediaType: string;
21
+ /** 添付ファイルを取得できる URL です。 */
18
22
  url: string;
19
23
  };
20
24
  type SendAppMessagePayload = {
25
+ /** チャット上に表示するタイトルです。 */
21
26
  title?: string;
27
+ /** ユーザーに表示するメッセージ本文です。 */
22
28
  message: string;
29
+ /** UI に表示せず、裏側のコンテキストとして渡すメッセージです。 */
23
30
  hiddenMessage?: string;
31
+ /** 新規チャットを開始してから送るかを指定します。 */
24
32
  startOnNewChat?: boolean;
33
+ /** メッセージと一緒に渡す添付ファイル一覧です。 */
25
34
  attachments?: AppMessageAttachment[];
26
35
  };
27
36
  //#endregion
@@ -1 +1 @@
1
- {"version":3,"file":"use-chatbot-frame-handler.d.mts","names":[],"sources":["../../../src/components/provider/use-chatbot-frame-handler.tsx"],"sourcesContent":[],"mappings":";;;;KAcY,GAAA;;;YAGA;;KAGA,SAAA;SACH;;;;KAKG,oBAAA;;;;;KAMA,qBAAA;;;;;gBAKI"}
1
+ {"version":3,"file":"use-chatbot-frame-handler.d.mts","names":[],"sources":["../../../src/components/provider/use-chatbot-frame-handler.tsx"],"sourcesContent":[],"mappings":";;;;;KAsBY,GAAA;;;YAGA;;KAGA,SAAA;SACH;;;;KAKG,oBAAA;;;;;;;;KASA,qBAAA;;;;;;;;;;gBAUI"}
@@ -4,11 +4,46 @@ import { _objectSpread2 } from "../../_virtual/_@oxc-project_runtime@0.103.0/hel
4
4
  import React from "react";
5
5
 
6
6
  //#region src/components/provider/use-chatbot-frame-handler.tsx
7
+ function serializeError(error) {
8
+ if (error instanceof Error) return error.message;
9
+ if (typeof error === "string") return error;
10
+ return "Unknown client tool error";
11
+ }
12
+ /**
13
+ * iframe と親アプリ間のメッセージ送受信を管理し、チャットボット制御 API をまとめます。
14
+ *
15
+ * @returns iframe 初期化、メッセージ送信、Client Tool 登録に使うハンドラ一式を返します。
16
+ *
17
+ * @example
18
+ * ```tsx
19
+ * export function FrameBridge() {
20
+ * const [open, setOpen] = React.useState(false);
21
+ * const [isExpanded, setIsExpanded] = React.useState(false);
22
+ * const [isFullScreen, setIsFullScreen] = React.useState(false);
23
+ * const { setIframeEl, init, sendAppMessage, isReady } =
24
+ * useChatbotFrameHandler({
25
+ * setOpen,
26
+ * setIsExpanded,
27
+ * setIsFullScreen,
28
+ * });
29
+ *
30
+ * React.useEffect(() => {
31
+ * if (isReady) {
32
+ * init({ appName: "EC Force" });
33
+ * sendAppMessage({ message: "在庫を確認したいです" });
34
+ * }
35
+ * }, [init, isReady, sendAppMessage]);
36
+ *
37
+ * return <iframe ref={setIframeEl} src="https://example.com/embed" />;
38
+ * }
39
+ * ```
40
+ */
7
41
  function useChatbotFrameHandler({ setOpen, setIsExpanded, setIsFullScreen }) {
8
42
  const [iframeEl, setIframeEl] = React.useState(null);
9
43
  const [isReady, setIsReady] = React.useState(false);
10
44
  const [isInitialized, setIsInitialized] = React.useState(false);
11
45
  const pendingAppMessageRef = React.useRef(null);
46
+ const clientToolRegistryRef = React.useRef(/* @__PURE__ */ new Map());
12
47
  const postMessage = React.useCallback((message) => {
13
48
  if (iframeEl === null || iframeEl === void 0 ? void 0 : iframeEl.contentWindow) {
14
49
  if (process.env.NODE_ENV === "development") console.log("sending message to iframe", message);
@@ -61,6 +96,42 @@ function useChatbotFrameHandler({ setOpen, setIsExpanded, setIsFullScreen }) {
61
96
  React.useEffect(() => {
62
97
  flushPendingAppMessage();
63
98
  }, [flushPendingAppMessage]);
99
+ const registerClientTool = React.useCallback((tool) => {
100
+ clientToolRegistryRef.current.set(tool.definition.name, tool);
101
+ if (!isReady || !isInitialized) return;
102
+ postMessage({
103
+ type: "registerClientTools",
104
+ tools: [tool.definition]
105
+ });
106
+ }, [
107
+ isInitialized,
108
+ isReady,
109
+ postMessage
110
+ ]);
111
+ const unregisterClientTool = React.useCallback((name) => {
112
+ if (!clientToolRegistryRef.current.delete(name) || !isReady || !isInitialized) return;
113
+ postMessage({
114
+ type: "unregisterClientTools",
115
+ names: [name]
116
+ });
117
+ }, [
118
+ isInitialized,
119
+ isReady,
120
+ postMessage
121
+ ]);
122
+ React.useEffect(() => {
123
+ if (!isReady || !isInitialized) return;
124
+ const tools = [...clientToolRegistryRef.current.values()].map((tool) => tool.definition);
125
+ if (tools.length === 0) return;
126
+ postMessage({
127
+ type: "registerClientTools",
128
+ tools
129
+ });
130
+ }, [
131
+ isInitialized,
132
+ isReady,
133
+ postMessage
134
+ ]);
64
135
  React.useEffect(() => {
65
136
  const iframe = iframeEl;
66
137
  if (!iframe) return;
@@ -93,6 +164,34 @@ function useChatbotFrameHandler({ setOpen, setIsExpanded, setIsFullScreen }) {
93
164
  case "RELOAD_CHATBOT":
94
165
  iframe.src = iframe.src;
95
166
  break;
167
+ case "EXECUTE_CLIENT_TOOL": {
168
+ const { callId, name, args } = event.data;
169
+ const tool = clientToolRegistryRef.current.get(name);
170
+ if (!tool) {
171
+ postMessage({
172
+ type: "clientToolResult",
173
+ callId,
174
+ error: `Client tool not found: ${name}`
175
+ });
176
+ break;
177
+ }
178
+ (async () => {
179
+ try {
180
+ postMessage({
181
+ type: "clientToolResult",
182
+ callId,
183
+ result: await tool.execute(args)
184
+ });
185
+ } catch (error) {
186
+ postMessage({
187
+ type: "clientToolResult",
188
+ callId,
189
+ error: serializeError(error)
190
+ });
191
+ }
192
+ })();
193
+ break;
194
+ }
96
195
  }
97
196
  };
98
197
  const handleIframeLoad = () => {
@@ -112,6 +211,8 @@ function useChatbotFrameHandler({ setOpen, setIsExpanded, setIsFullScreen }) {
112
211
  setAppName,
113
212
  setSessionToken,
114
213
  sendAppMessage,
214
+ registerClientTool,
215
+ unregisterClientTool,
115
216
  setIframeEl,
116
217
  isReady,
117
218
  isInitialized
@@ -1 +1 @@
1
- {"version":3,"file":"use-chatbot-frame-handler.mjs","names":[],"sources":["../../../src/components/provider/use-chatbot-frame-handler.tsx"],"sourcesContent":["\"use client\";\n\nimport React from \"react\";\n\nexport type IframeMessage =\n | { type: \"CHATBOT_READY\" }\n | { type: \"CHATBOT_INITIALIZED\" }\n | { type: \"CLOSE_CHATBOT\" }\n | { type: \"EXPAND_CHATBOT\" }\n | { type: \"SHRINK_CHATBOT\" }\n | { type: \"FULLSCREEN_CHATBOT\" }\n | { type: \"EXIT_FULLSCREEN_CHATBOT\" }\n | { type: \"RELOAD_CHATBOT\" };\n\nexport type MCP = {\n name: string;\n url: string;\n headers?: Record<string, string>;\n};\n\nexport type InitProps = {\n mcps?: MCP[];\n appName?: string;\n sessionToken?: string;\n};\n\nexport type AppMessageAttachment = {\n fileName: string;\n mediaType: string;\n url: string;\n};\n\nexport type SendAppMessagePayload = {\n title?: string;\n message: string;\n hiddenMessage?: string;\n startOnNewChat?: boolean;\n attachments?: AppMessageAttachment[];\n};\n\nexport type ParentMessage =\n | {\n type: \"mcps\";\n mcps: MCP[];\n }\n | {\n type: \"appName\";\n appName: string;\n }\n | {\n type: \"sessionToken\";\n sessionToken: string;\n }\n | {\n type: \"init\";\n mcps?: MCP[];\n appName?: string;\n sessionToken?: string;\n }\n | {\n type: \"appMessage\";\n payload: SendAppMessagePayload;\n };\n\nexport function useChatbotFrameHandler({\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n}: {\n setOpen: (open: boolean) => void;\n setIsExpanded: (expanded: boolean) => void;\n setIsFullScreen: (fullScreen: boolean) => void;\n}) {\n const [iframeEl, setIframeEl] = React.useState<HTMLIFrameElement | null>(\n null,\n );\n const [isReady, setIsReady] = React.useState(false);\n const [isInitialized, setIsInitialized] = React.useState(false);\n const pendingAppMessageRef = React.useRef<SendAppMessagePayload | null>(null);\n\n // helper to post message to the iframe\n const postMessage = React.useCallback(\n (message: ParentMessage) => {\n if (iframeEl?.contentWindow) {\n if (process.env.NODE_ENV === \"development\") {\n console.log(\"sending message to iframe\", message);\n }\n iframeEl.contentWindow.postMessage(message, \"*\");\n return true;\n }\n return false;\n },\n [iframeEl],\n );\n\n const init = React.useCallback(\n (props: InitProps) => {\n postMessage({\n type: \"init\",\n ...props,\n });\n },\n [postMessage],\n );\n\n const setMcps = React.useCallback(\n (mcps: MCP[]) => {\n postMessage({ type: \"mcps\", mcps });\n },\n [postMessage],\n );\n\n const setAppName = React.useCallback(\n (appName: string) => {\n postMessage({ type: \"appName\", appName });\n },\n [postMessage],\n );\n\n const setSessionToken = React.useCallback(\n (sessionToken: string) => {\n postMessage({ type: \"sessionToken\", sessionToken });\n },\n [postMessage],\n );\n\n const flushPendingAppMessage = React.useCallback(() => {\n const pendingPayload = pendingAppMessageRef.current;\n if (!pendingPayload || !isReady || !isInitialized) {\n return false;\n }\n\n const sent = postMessage({\n type: \"appMessage\",\n payload: pendingPayload,\n });\n\n if (sent) {\n pendingAppMessageRef.current = null;\n }\n\n return sent;\n }, [isInitialized, isReady, postMessage]);\n\n const sendAppMessage = React.useCallback(\n (payload: SendAppMessagePayload) => {\n pendingAppMessageRef.current = {\n ...payload,\n startOnNewChat: payload.startOnNewChat ?? true,\n };\n setOpen(true);\n flushPendingAppMessage();\n },\n [flushPendingAppMessage, setOpen],\n );\n\n React.useEffect(() => {\n flushPendingAppMessage();\n }, [flushPendingAppMessage]);\n\n // Initialize and setup listeners\n React.useEffect(() => {\n const iframe = iframeEl;\n if (!iframe) {\n return;\n }\n\n const handleMessage = (event: MessageEvent<IframeMessage>) => {\n if (event.source !== iframe.contentWindow) {\n return;\n }\n if (process.env.NODE_ENV === \"development\") {\n console.log(\"iframe message\", event.data);\n }\n // Handle iframe messages\n switch (event.data?.type) {\n case \"CHATBOT_READY\":\n setIsReady(true);\n break;\n case \"CHATBOT_INITIALIZED\":\n setIsInitialized(true);\n break;\n case \"CLOSE_CHATBOT\":\n setOpen(false);\n break;\n case \"EXPAND_CHATBOT\":\n setIsExpanded(true);\n break;\n case \"SHRINK_CHATBOT\":\n setIsExpanded(false);\n break;\n case \"FULLSCREEN_CHATBOT\":\n setIsFullScreen(true);\n break;\n case \"EXIT_FULLSCREEN_CHATBOT\":\n setIsFullScreen(false);\n break;\n case \"RELOAD_CHATBOT\":\n // reload iframe\n iframe.src = iframe.src;\n break;\n }\n };\n\n const handleIframeLoad = () => {\n setIsReady(false);\n setIsInitialized(false);\n };\n\n window.addEventListener(\"message\", handleMessage);\n iframe.addEventListener(\"load\", handleIframeLoad);\n\n return () => {\n window.removeEventListener(\"message\", handleMessage);\n iframe.removeEventListener(\"load\", handleIframeLoad);\n };\n }, [iframeEl]); // Run when iframe gets mounted\n\n return {\n init,\n setMcps,\n setAppName,\n setSessionToken,\n sendAppMessage,\n setIframeEl,\n isReady,\n isInitialized,\n };\n}\n"],"mappings":";;;;;;AAgEA,SAAgB,uBAAuB,EACrC,SACA,eACA,mBAKC;CACD,MAAM,CAAC,UAAU,eAAe,MAAM,SACpC,KACD;CACD,MAAM,CAAC,SAAS,cAAc,MAAM,SAAS,MAAM;CACnD,MAAM,CAAC,eAAe,oBAAoB,MAAM,SAAS,MAAM;CAC/D,MAAM,uBAAuB,MAAM,OAAqC,KAAK;CAG7E,MAAM,cAAc,MAAM,aACvB,YAA2B;AAC1B,0DAAI,SAAU,eAAe;AAC3B,OAAI,QAAQ,IAAI,aAAa,cAC3B,SAAQ,IAAI,6BAA6B,QAAQ;AAEnD,YAAS,cAAc,YAAY,SAAS,IAAI;AAChD,UAAO;;AAET,SAAO;IAET,CAAC,SAAS,CACX;CAED,MAAM,OAAO,MAAM,aAChB,UAAqB;AACpB,+BACE,MAAM,UACH,OACH;IAEJ,CAAC,YAAY,CACd;CAED,MAAM,UAAU,MAAM,aACnB,SAAgB;AACf,cAAY;GAAE,MAAM;GAAQ;GAAM,CAAC;IAErC,CAAC,YAAY,CACd;CAED,MAAM,aAAa,MAAM,aACtB,YAAoB;AACnB,cAAY;GAAE,MAAM;GAAW;GAAS,CAAC;IAE3C,CAAC,YAAY,CACd;CAED,MAAM,kBAAkB,MAAM,aAC3B,iBAAyB;AACxB,cAAY;GAAE,MAAM;GAAgB;GAAc,CAAC;IAErD,CAAC,YAAY,CACd;CAED,MAAM,yBAAyB,MAAM,kBAAkB;EACrD,MAAM,iBAAiB,qBAAqB;AAC5C,MAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,cAClC,QAAO;EAGT,MAAM,OAAO,YAAY;GACvB,MAAM;GACN,SAAS;GACV,CAAC;AAEF,MAAI,KACF,sBAAqB,UAAU;AAGjC,SAAO;IACN;EAAC;EAAe;EAAS;EAAY,CAAC;CAEzC,MAAM,iBAAiB,MAAM,aAC1B,YAAmC;;AAClC,uBAAqB,4CAChB,gBACH,yCAAgB,QAAQ,uFAAkB;AAE5C,UAAQ,KAAK;AACb,0BAAwB;IAE1B,CAAC,wBAAwB,QAAQ,CAClC;AAED,OAAM,gBAAgB;AACpB,0BAAwB;IACvB,CAAC,uBAAuB,CAAC;AAG5B,OAAM,gBAAgB;EACpB,MAAM,SAAS;AACf,MAAI,CAAC,OACH;EAGF,MAAM,iBAAiB,UAAuC;;AAC5D,OAAI,MAAM,WAAW,OAAO,cAC1B;AAEF,OAAI,QAAQ,IAAI,aAAa,cAC3B,SAAQ,IAAI,kBAAkB,MAAM,KAAK;AAG3C,0BAAQ,MAAM,gEAAM,MAApB;IACE,KAAK;AACH,gBAAW,KAAK;AAChB;IACF,KAAK;AACH,sBAAiB,KAAK;AACtB;IACF,KAAK;AACH,aAAQ,MAAM;AACd;IACF,KAAK;AACH,mBAAc,KAAK;AACnB;IACF,KAAK;AACH,mBAAc,MAAM;AACpB;IACF,KAAK;AACH,qBAAgB,KAAK;AACrB;IACF,KAAK;AACH,qBAAgB,MAAM;AACtB;IACF,KAAK;AAEH,YAAO,MAAM,OAAO;AACpB;;;EAIN,MAAM,yBAAyB;AAC7B,cAAW,MAAM;AACjB,oBAAiB,MAAM;;AAGzB,SAAO,iBAAiB,WAAW,cAAc;AACjD,SAAO,iBAAiB,QAAQ,iBAAiB;AAEjD,eAAa;AACX,UAAO,oBAAoB,WAAW,cAAc;AACpD,UAAO,oBAAoB,QAAQ,iBAAiB;;IAErD,CAAC,SAAS,CAAC;AAEd,QAAO;EACL;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD"}
1
+ {"version":3,"file":"use-chatbot-frame-handler.mjs","names":[],"sources":["../../../src/components/provider/use-chatbot-frame-handler.tsx"],"sourcesContent":["\"use client\";\n\nimport React from \"react\";\nimport type {\n ClientToolDefinition,\n ClientToolExecuteRequest,\n RegisteredClientTool,\n} from \"../../lib/client-tools\";\n\nexport type IframeMessage =\n | { type: \"CHATBOT_READY\" }\n | { type: \"CHATBOT_INITIALIZED\" }\n | { type: \"CLOSE_CHATBOT\" }\n | { type: \"EXPAND_CHATBOT\" }\n | { type: \"SHRINK_CHATBOT\" }\n | { type: \"FULLSCREEN_CHATBOT\" }\n | { type: \"EXIT_FULLSCREEN_CHATBOT\" }\n | { type: \"RELOAD_CHATBOT\" }\n | ({\n type: \"EXECUTE_CLIENT_TOOL\";\n } & ClientToolExecuteRequest);\n\nexport type MCP = {\n name: string;\n url: string;\n headers?: Record<string, string>;\n};\n\nexport type InitProps = {\n mcps?: MCP[];\n appName?: string;\n sessionToken?: string;\n};\n\nexport type AppMessageAttachment = {\n /** 添付ファイルの表示名です。 */\n fileName: string;\n /** 添付ファイルの MIME type です。 */\n mediaType: string;\n /** 添付ファイルを取得できる URL です。 */\n url: string;\n};\n\nexport type SendAppMessagePayload = {\n /** チャット上に表示するタイトルです。 */\n title?: string;\n /** ユーザーに表示するメッセージ本文です。 */\n message: string;\n /** UI に表示せず、裏側のコンテキストとして渡すメッセージです。 */\n hiddenMessage?: string;\n /** 新規チャットを開始してから送るかを指定します。 */\n startOnNewChat?: boolean;\n /** メッセージと一緒に渡す添付ファイル一覧です。 */\n attachments?: AppMessageAttachment[];\n};\n\nexport type ParentMessage =\n | {\n type: \"mcps\";\n mcps: MCP[];\n }\n | {\n type: \"appName\";\n appName: string;\n }\n | {\n type: \"sessionToken\";\n sessionToken: string;\n }\n | {\n type: \"init\";\n mcps?: MCP[];\n appName?: string;\n sessionToken?: string;\n }\n | {\n type: \"appMessage\";\n payload: SendAppMessagePayload;\n }\n | {\n type: \"registerClientTools\";\n tools: ClientToolDefinition[];\n }\n | {\n type: \"unregisterClientTools\";\n names: string[];\n }\n | {\n type: \"clientToolResult\";\n callId: string;\n result?: unknown;\n error?: string;\n };\n\nfunction serializeError(error: unknown) {\n if (error instanceof Error) {\n return error.message;\n }\n\n if (typeof error === \"string\") {\n return error;\n }\n\n return \"Unknown client tool error\";\n}\n\ntype UseChatbotFrameHandlerProps = {\n /** チャットボットの開閉状態を更新します。 */\n setOpen: (open: boolean) => void;\n /** 展開状態を更新します。 */\n setIsExpanded: (expanded: boolean) => void;\n /** フルスクリーン状態を更新します。 */\n setIsFullScreen: (fullScreen: boolean) => void;\n};\n\n/**\n * iframe と親アプリ間のメッセージ送受信を管理し、チャットボット制御 API をまとめます。\n *\n * @returns iframe 初期化、メッセージ送信、Client Tool 登録に使うハンドラ一式を返します。\n *\n * @example\n * ```tsx\n * export function FrameBridge() {\n * const [open, setOpen] = React.useState(false);\n * const [isExpanded, setIsExpanded] = React.useState(false);\n * const [isFullScreen, setIsFullScreen] = React.useState(false);\n * const { setIframeEl, init, sendAppMessage, isReady } =\n * useChatbotFrameHandler({\n * setOpen,\n * setIsExpanded,\n * setIsFullScreen,\n * });\n *\n * React.useEffect(() => {\n * if (isReady) {\n * init({ appName: \"EC Force\" });\n * sendAppMessage({ message: \"在庫を確認したいです\" });\n * }\n * }, [init, isReady, sendAppMessage]);\n *\n * return <iframe ref={setIframeEl} src=\"https://example.com/embed\" />;\n * }\n * ```\n */\nexport function useChatbotFrameHandler({\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n}: UseChatbotFrameHandlerProps) {\n const [iframeEl, setIframeEl] = React.useState<HTMLIFrameElement | null>(\n null,\n );\n const [isReady, setIsReady] = React.useState(false);\n const [isInitialized, setIsInitialized] = React.useState(false);\n const pendingAppMessageRef = React.useRef<SendAppMessagePayload | null>(null);\n const clientToolRegistryRef = React.useRef(\n new Map<string, RegisteredClientTool>(),\n );\n\n // helper to post message to the iframe\n const postMessage = React.useCallback(\n (message: ParentMessage) => {\n if (iframeEl?.contentWindow) {\n if (process.env.NODE_ENV === \"development\") {\n console.log(\"sending message to iframe\", message);\n }\n iframeEl.contentWindow.postMessage(message, \"*\");\n return true;\n }\n return false;\n },\n [iframeEl],\n );\n\n const init = React.useCallback(\n (props: InitProps) => {\n postMessage({\n type: \"init\",\n ...props,\n });\n },\n [postMessage],\n );\n\n const setMcps = React.useCallback(\n (mcps: MCP[]) => {\n postMessage({ type: \"mcps\", mcps });\n },\n [postMessage],\n );\n\n const setAppName = React.useCallback(\n (appName: string) => {\n postMessage({ type: \"appName\", appName });\n },\n [postMessage],\n );\n\n const setSessionToken = React.useCallback(\n (sessionToken: string) => {\n postMessage({ type: \"sessionToken\", sessionToken });\n },\n [postMessage],\n );\n\n const flushPendingAppMessage = React.useCallback(() => {\n const pendingPayload = pendingAppMessageRef.current;\n if (!pendingPayload || !isReady || !isInitialized) {\n return false;\n }\n\n const sent = postMessage({\n type: \"appMessage\",\n payload: pendingPayload,\n });\n\n if (sent) {\n pendingAppMessageRef.current = null;\n }\n\n return sent;\n }, [isInitialized, isReady, postMessage]);\n\n const sendAppMessage = React.useCallback(\n (payload: SendAppMessagePayload) => {\n pendingAppMessageRef.current = {\n ...payload,\n startOnNewChat: payload.startOnNewChat ?? true,\n };\n setOpen(true);\n flushPendingAppMessage();\n },\n [flushPendingAppMessage, setOpen],\n );\n\n React.useEffect(() => {\n flushPendingAppMessage();\n }, [flushPendingAppMessage]);\n\n const registerClientTool = React.useCallback(\n (tool: RegisteredClientTool) => {\n clientToolRegistryRef.current.set(tool.definition.name, tool);\n if (!isReady || !isInitialized) {\n return;\n }\n\n postMessage({\n type: \"registerClientTools\",\n tools: [tool.definition],\n });\n },\n [isInitialized, isReady, postMessage],\n );\n\n const unregisterClientTool = React.useCallback(\n (name: string) => {\n const deleted = clientToolRegistryRef.current.delete(name);\n if (!deleted || !isReady || !isInitialized) {\n return;\n }\n\n postMessage({\n type: \"unregisterClientTools\",\n names: [name],\n });\n },\n [isInitialized, isReady, postMessage],\n );\n\n React.useEffect(() => {\n if (!isReady || !isInitialized) {\n return;\n }\n\n const tools = [...clientToolRegistryRef.current.values()].map(\n (tool) => tool.definition,\n );\n if (tools.length === 0) {\n return;\n }\n\n postMessage({\n type: \"registerClientTools\",\n tools,\n });\n }, [isInitialized, isReady, postMessage]);\n\n // Initialize and setup listeners\n React.useEffect(() => {\n const iframe = iframeEl;\n if (!iframe) {\n return;\n }\n\n const handleMessage = (event: MessageEvent<IframeMessage>) => {\n if (event.source !== iframe.contentWindow) {\n return;\n }\n if (process.env.NODE_ENV === \"development\") {\n console.log(\"iframe message\", event.data);\n }\n // Handle iframe messages\n switch (event.data?.type) {\n case \"CHATBOT_READY\":\n setIsReady(true);\n break;\n case \"CHATBOT_INITIALIZED\":\n setIsInitialized(true);\n break;\n case \"CLOSE_CHATBOT\":\n setOpen(false);\n break;\n case \"EXPAND_CHATBOT\":\n setIsExpanded(true);\n break;\n case \"SHRINK_CHATBOT\":\n setIsExpanded(false);\n break;\n case \"FULLSCREEN_CHATBOT\":\n setIsFullScreen(true);\n break;\n case \"EXIT_FULLSCREEN_CHATBOT\":\n setIsFullScreen(false);\n break;\n case \"RELOAD_CHATBOT\":\n // reload iframe\n iframe.src = iframe.src;\n break;\n case \"EXECUTE_CLIENT_TOOL\": {\n const { callId, name, args } = event.data;\n const tool = clientToolRegistryRef.current.get(name);\n if (!tool) {\n postMessage({\n type: \"clientToolResult\",\n callId,\n error: `Client tool not found: ${name}`,\n });\n break;\n }\n\n void (async () => {\n try {\n const result = await tool.execute(args);\n postMessage({\n type: \"clientToolResult\",\n callId,\n result,\n });\n } catch (error) {\n postMessage({\n type: \"clientToolResult\",\n callId,\n error: serializeError(error),\n });\n }\n })();\n break;\n }\n }\n };\n\n const handleIframeLoad = () => {\n setIsReady(false);\n setIsInitialized(false);\n };\n\n window.addEventListener(\"message\", handleMessage);\n iframe.addEventListener(\"load\", handleIframeLoad);\n\n return () => {\n window.removeEventListener(\"message\", handleMessage);\n iframe.removeEventListener(\"load\", handleIframeLoad);\n };\n }, [iframeEl]); // Run when iframe gets mounted\n\n return {\n init,\n setMcps,\n setAppName,\n setSessionToken,\n sendAppMessage,\n registerClientTool,\n unregisterClientTool,\n setIframeEl,\n isReady,\n isInitialized,\n };\n}\n"],"mappings":"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"}
@@ -3,6 +3,31 @@
3
3
  import React from "react";
4
4
 
5
5
  //#region src/components/provider/use-chatbot-window-states.tsx
6
+ /**
7
+ * チャットボット UI の開閉・展開・フルスクリーン状態を管理します。
8
+ *
9
+ * @returns 初回オープン判定と、各表示状態、その更新関数を返します。
10
+ *
11
+ * @example
12
+ * ```tsx
13
+ * export function ChatbotWindowController() {
14
+ * const { open, setOpen, isExpanded, setIsExpanded } =
15
+ * useChatbotWindowStates();
16
+ *
17
+ * return (
18
+ * <button
19
+ * type="button"
20
+ * onClick={() => {
21
+ * setOpen(!open);
22
+ * setIsExpanded(!isExpanded);
23
+ * }}
24
+ * >
25
+ * Toggle window
26
+ * </button>
27
+ * );
28
+ * }
29
+ * ```
30
+ */
6
31
  function useChatbotWindowStates() {
7
32
  const [hasOpened, setHasOpened] = React.useState(false);
8
33
  const [open, setOpen] = React.useState(false);
@@ -1 +1 @@
1
- {"version":3,"file":"use-chatbot-window-states.mjs","names":[],"sources":["../../../src/components/provider/use-chatbot-window-states.tsx"],"sourcesContent":["\"use client\";\n\nimport React from \"react\";\n\nexport function useChatbotWindowStates() {\n const [hasOpened, setHasOpened] = React.useState(false);\n const [open, setOpen] = React.useState(false);\n const [isExpanded, setIsExpanded] = React.useState(false);\n const [isFullScreen, setIsFullScreen] = React.useState(false);\n\n React.useEffect(() => {\n if (open) {\n setHasOpened(true);\n }\n }, [open]);\n\n return {\n hasOpened,\n open,\n isExpanded,\n isFullScreen,\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n };\n}\n"],"mappings":";;;;;AAIA,SAAgB,yBAAyB;CACvC,MAAM,CAAC,WAAW,gBAAgB,MAAM,SAAS,MAAM;CACvD,MAAM,CAAC,MAAM,WAAW,MAAM,SAAS,MAAM;CAC7C,MAAM,CAAC,YAAY,iBAAiB,MAAM,SAAS,MAAM;CACzD,MAAM,CAAC,cAAc,mBAAmB,MAAM,SAAS,MAAM;AAE7D,OAAM,gBAAgB;AACpB,MAAI,KACF,cAAa,KAAK;IAEnB,CAAC,KAAK,CAAC;AAEV,QAAO;EACL;EACA;EACA;EACA;EACA;EACA;EACA;EACD"}
1
+ {"version":3,"file":"use-chatbot-window-states.mjs","names":[],"sources":["../../../src/components/provider/use-chatbot-window-states.tsx"],"sourcesContent":["\"use client\";\n\nimport React from \"react\";\n\n/**\n * チャットボット UI の開閉・展開・フルスクリーン状態を管理します。\n *\n * @returns 初回オープン判定と、各表示状態、その更新関数を返します。\n *\n * @example\n * ```tsx\n * export function ChatbotWindowController() {\n * const { open, setOpen, isExpanded, setIsExpanded } =\n * useChatbotWindowStates();\n *\n * return (\n * <button\n * type=\"button\"\n * onClick={() => {\n * setOpen(!open);\n * setIsExpanded(!isExpanded);\n * }}\n * >\n * Toggle window\n * </button>\n * );\n * }\n * ```\n */\nexport function useChatbotWindowStates() {\n const [hasOpened, setHasOpened] = React.useState(false);\n const [open, setOpen] = React.useState(false);\n const [isExpanded, setIsExpanded] = React.useState(false);\n const [isFullScreen, setIsFullScreen] = React.useState(false);\n\n React.useEffect(() => {\n if (open) {\n setHasOpened(true);\n }\n }, [open]);\n\n return {\n hasOpened,\n open,\n isExpanded,\n isFullScreen,\n setOpen,\n setIsExpanded,\n setIsFullScreen,\n };\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6BA,SAAgB,yBAAyB;CACvC,MAAM,CAAC,WAAW,gBAAgB,MAAM,SAAS,MAAM;CACvD,MAAM,CAAC,MAAM,WAAW,MAAM,SAAS,MAAM;CAC7C,MAAM,CAAC,YAAY,iBAAiB,MAAM,SAAS,MAAM;CACzD,MAAM,CAAC,cAAc,mBAAmB,MAAM,SAAS,MAAM;AAE7D,OAAM,gBAAgB;AACpB,MAAI,KACF,cAAa,KAAK;IAEnB,CAAC,KAAK,CAAC;AAEV,QAAO;EACL;EACA;EACA;EACA;EACA;EACA;EACA;EACD"}
@@ -1,11 +1,11 @@
1
1
  import * as React$1 from "react";
2
- import * as react_jsx_runtime3 from "react/jsx-runtime";
2
+ import * as react_jsx_runtime2 from "react/jsx-runtime";
3
3
  import * as SheetPrimitive from "@radix-ui/react-dialog";
4
4
 
5
5
  //#region src/components/sheet.d.ts
6
6
  declare function Sheet({
7
7
  ...props
8
- }: React$1.ComponentProps<typeof SheetPrimitive.Root>): react_jsx_runtime3.JSX.Element;
8
+ }: React$1.ComponentProps<typeof SheetPrimitive.Root>): react_jsx_runtime2.JSX.Element;
9
9
  declare const SheetTrigger: React$1.ForwardRefExoticComponent<Omit<SheetPrimitive.DialogTriggerProps & React$1.RefAttributes<HTMLButtonElement>, "ref"> & React$1.RefAttributes<HTMLButtonElement>>;
10
10
  declare const SheetClose: React$1.ForwardRefExoticComponent<Omit<SheetPrimitive.DialogCloseProps & React$1.RefAttributes<HTMLButtonElement>, "ref"> & React$1.RefAttributes<HTMLButtonElement>>;
11
11
  declare const SheetContent: React$1.ForwardRefExoticComponent<Omit<SheetPrimitive.DialogContentProps & React$1.RefAttributes<HTMLDivElement>, "ref"> & React$1.RefAttributes<HTMLDivElement>>;
@@ -1,4 +1,4 @@
1
- import * as react_jsx_runtime0 from "react/jsx-runtime";
1
+ import * as react_jsx_runtime3 from "react/jsx-runtime";
2
2
 
3
3
  //#region src/components/tooltip.d.ts
4
4
  type Side = "top" | "right" | "bottom" | "left";
@@ -32,7 +32,7 @@ declare function Tooltip({
32
32
  sideOffset,
33
33
  arrowClassName,
34
34
  keepTooltipOpen
35
- }: TooltipProps): react_jsx_runtime0.JSX.Element;
35
+ }: TooltipProps): react_jsx_runtime3.JSX.Element;
36
36
  //#endregion
37
37
  export { Tooltip };
38
38
  //# sourceMappingURL=tooltip.d.mts.map
package/dist/index.d.mts CHANGED
@@ -1,6 +1,7 @@
1
+ import { ClientToolDefinition, ClientToolExecuteRequest, ClientToolJsonSchema, RegisteredClientTool, UseClientToolOptions, useClientTool } from "./lib/client-tools.mjs";
1
2
  import { ChatbotProvider, useChatbot } from "./components/provider/chatbot-provider.mjs";
2
3
  import { ChatbotFrame, ChatbotFrameProps, SessionData, UseSessionProps, useSession } from "./components/chatbot-frame.mjs";
3
4
  import { ChatbotSheet } from "./components/chatbot-sheet.mjs";
4
5
  import { Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger } from "./components/sheet.mjs";
5
6
  import { Tooltip } from "./components/tooltip.mjs";
6
- export { ChatbotFrame, ChatbotFrameProps, ChatbotProvider, ChatbotSheet, SessionData, Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger, Tooltip, UseSessionProps, useChatbot, useSession };
7
+ export { ChatbotFrame, ChatbotFrameProps, ChatbotProvider, ChatbotSheet, ClientToolDefinition, ClientToolExecuteRequest, ClientToolJsonSchema, RegisteredClientTool, SessionData, Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger, Tooltip, UseClientToolOptions, UseSessionProps, useChatbot, useClientTool, useSession };
package/dist/index.mjs CHANGED
@@ -3,5 +3,6 @@ import { ChatbotFrame, useSession } from "./components/chatbot-frame.mjs";
3
3
  import { Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger } from "./components/sheet.mjs";
4
4
  import { Tooltip } from "./components/tooltip.mjs";
5
5
  import { ChatbotSheet } from "./components/chatbot-sheet.mjs";
6
+ import { useClientTool } from "./lib/client-tools.mjs";
6
7
 
7
- export { ChatbotFrame, ChatbotProvider, ChatbotSheet, Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger, Tooltip, useChatbot, useSession };
8
+ export { ChatbotFrame, ChatbotProvider, ChatbotSheet, Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger, Tooltip, useChatbot, useClientTool, useSession };
@@ -0,0 +1,75 @@
1
+ import { z } from "zod";
2
+
3
+ //#region src/lib/client-tools.d.ts
4
+ type ClientToolJsonSchema = Record<string, unknown>;
5
+ type ClientToolDefinition = {
6
+ name: string;
7
+ displayName: string;
8
+ description: string;
9
+ summary?: string;
10
+ requireConfirmation?: boolean;
11
+ parameters: ClientToolJsonSchema;
12
+ };
13
+ type ClientToolExecuteRequest = {
14
+ callId: string;
15
+ name: string;
16
+ args: unknown;
17
+ };
18
+ type RegisteredClientTool = {
19
+ definition: ClientToolDefinition;
20
+ execute: (args: unknown) => Promise<unknown> | unknown;
21
+ };
22
+ type UseClientToolOptions<TParameters extends z.ZodTypeAny> = {
23
+ /** ツール名です。`page_`のプレフィックスが自動で追加されます。 */
24
+ name: string;
25
+ /** UI上に表示するツール名です。 */
26
+ displayName: string;
27
+ /** LLMが見れるツールの詳細説明です。 */
28
+ description: string;
29
+ /** ユーザーに表示するツールの要約です。 */
30
+ summary?: string;
31
+ /** 実行前にユーザー確認を要求するかを指定します。 */
32
+ requireConfirmation?: boolean;
33
+ /** ツールの引数を定義するZodスキーマです。 */
34
+ parameters: TParameters;
35
+ /** 検証済み引数を受け取り、ツール本体の処理を実行します。 */
36
+ execute: (args: z.output<TParameters>) => Promise<unknown> | unknown;
37
+ };
38
+ /**
39
+ * チャットボットから呼び出せるクライアントツールを登録します。
40
+ *
41
+ * @example
42
+ * ```tsx
43
+ * import { z } from "zod";
44
+ * import { useClientTool } from "@super_studio/ecforce-ai-agent-react";
45
+ *
46
+ * export function InventoryToolRegistration() {
47
+ * useClientTool({
48
+ * name: "lookup_inventory",
49
+ * displayName: "在庫確認",
50
+ * description: "商品コードから在庫数を取得します。",
51
+ * summary: "商品在庫を返します。",
52
+ * parameters: z.object({
53
+ * sku: z.string().describe("商品コード"),
54
+ * }),
55
+ * execute: async ({ sku }) => {
56
+ * return { sku, available: true };
57
+ * },
58
+ * });
59
+ *
60
+ * return null;
61
+ * }
62
+ * ```
63
+ */
64
+ declare function useClientTool<TParameters extends z.ZodTypeAny>({
65
+ name,
66
+ displayName,
67
+ description,
68
+ summary,
69
+ requireConfirmation,
70
+ parameters,
71
+ execute
72
+ }: UseClientToolOptions<TParameters>): void;
73
+ //#endregion
74
+ export { ClientToolDefinition, ClientToolExecuteRequest, ClientToolJsonSchema, RegisteredClientTool, UseClientToolOptions, useClientTool };
75
+ //# sourceMappingURL=client-tools.d.mts.map