@assistant-ui/react-devtools 0.1.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.
Files changed (63) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +65 -0
  3. package/dist/DevToolsFrame.d.ts +8 -0
  4. package/dist/DevToolsFrame.d.ts.map +1 -0
  5. package/dist/DevToolsFrame.js +49 -0
  6. package/dist/DevToolsFrame.js.map +1 -0
  7. package/dist/DevToolsHost.d.ts +35 -0
  8. package/dist/DevToolsHost.d.ts.map +1 -0
  9. package/dist/DevToolsHost.js +104 -0
  10. package/dist/DevToolsHost.js.map +1 -0
  11. package/dist/DevToolsModal.d.ts +2 -0
  12. package/dist/DevToolsModal.d.ts.map +1 -0
  13. package/dist/DevToolsModal.js +170 -0
  14. package/dist/DevToolsModal.js.map +1 -0
  15. package/dist/ExtensionHost.d.ts +7 -0
  16. package/dist/ExtensionHost.d.ts.map +1 -0
  17. package/dist/ExtensionHost.js +50 -0
  18. package/dist/ExtensionHost.js.map +1 -0
  19. package/dist/FrameClient.d.ts +35 -0
  20. package/dist/FrameClient.d.ts.map +1 -0
  21. package/dist/FrameClient.js +63 -0
  22. package/dist/FrameClient.js.map +1 -0
  23. package/dist/FrameHost.d.ts +8 -0
  24. package/dist/FrameHost.d.ts.map +1 -0
  25. package/dist/FrameHost.js +28 -0
  26. package/dist/FrameHost.js.map +1 -0
  27. package/dist/constants.d.ts +3 -0
  28. package/dist/constants.d.ts.map +1 -0
  29. package/dist/constants.js +8 -0
  30. package/dist/constants.js.map +1 -0
  31. package/dist/index.d.ts +11 -0
  32. package/dist/index.d.ts.map +1 -0
  33. package/dist/index.js +27 -0
  34. package/dist/index.js.map +1 -0
  35. package/dist/styles/DevToolsModal.styles.d.ts +14 -0
  36. package/dist/styles/DevToolsModal.styles.d.ts.map +1 -0
  37. package/dist/styles/DevToolsModal.styles.js +121 -0
  38. package/dist/styles/DevToolsModal.styles.js.map +1 -0
  39. package/dist/types.d.ts +10 -0
  40. package/dist/types.d.ts.map +1 -0
  41. package/dist/types.js +1 -0
  42. package/dist/types.js.map +1 -0
  43. package/dist/utils/serialization.d.ts +5 -0
  44. package/dist/utils/serialization.d.ts.map +1 -0
  45. package/dist/utils/serialization.js +77 -0
  46. package/dist/utils/serialization.js.map +1 -0
  47. package/dist/utils/toolNormalization.d.ts +9 -0
  48. package/dist/utils/toolNormalization.d.ts.map +1 -0
  49. package/dist/utils/toolNormalization.js +58 -0
  50. package/dist/utils/toolNormalization.js.map +1 -0
  51. package/package.json +76 -0
  52. package/src/DevToolsFrame.tsx +55 -0
  53. package/src/DevToolsHost.ts +150 -0
  54. package/src/DevToolsModal.tsx +178 -0
  55. package/src/ExtensionHost.ts +57 -0
  56. package/src/FrameClient.ts +98 -0
  57. package/src/FrameHost.ts +31 -0
  58. package/src/constants.ts +2 -0
  59. package/src/index.ts +17 -0
  60. package/src/styles/DevToolsModal.styles.ts +137 -0
  61. package/src/types.ts +13 -0
  62. package/src/utils/serialization.ts +97 -0
  63. package/src/utils/toolNormalization.ts +83 -0
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2025 AgentbaseAI Inc.
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,65 @@
1
+ # @assistant-ui/react-devtools
2
+
3
+ React-first development tools for assistant-ui components. This package ships the reusable React helpers, runtime adapters, and UI necessary to embed the DevTools experience in any host application.
4
+
5
+ ## Features
6
+
7
+ - **Component Library**: React components for debugging assistant-ui experiences
8
+ - **Event Logging**: Track and inspect assistant-ui events and state changes
9
+ - **Context Viewer**: View Assistant API context and state in real-time
10
+ - **Embeddable Host**: Frame bridges that power custom hosts, including the Chrome extension
11
+
12
+ ## Installation
13
+
14
+ ### As React Component Library
15
+
16
+ ```bash
17
+ npm install @assistant-ui/react-devtools
18
+ ```
19
+
20
+ ### As Chrome Extension
21
+
22
+ See `apps/devtools-extension` for the standalone Chrome extension source and build scripts (`pnpm --filter @assistant-ui/devtools-extension run build`).
23
+
24
+ ## Usage
25
+
26
+ ### React Components
27
+
28
+ ```tsx
29
+ import { DevToolsUI, DevToolsModal } from '@assistant-ui/react-devtools';
30
+
31
+ // Use the full DevTools UI
32
+ <DevToolsUI />
33
+
34
+ // Or use as a modal overlay
35
+ <DevToolsModal />
36
+ ```
37
+
38
+ ### Chrome Extension
39
+
40
+ The Chrome extension now lives under `apps/devtools-extension` as a separate workspace app. It consumes this package at build time to reuse the shared runtime and UI.
41
+
42
+ ## Build Scripts
43
+
44
+ - `npm run build` - Build the React component library
45
+ - `npm run build:lib` - Alias for `build`
46
+ - `npm run dev` - Development build with watch mode
47
+
48
+ ## Package Structure
49
+
50
+ ```
51
+ packages/react-devtools/
52
+ ├── src/ # React component library source
53
+ │ ├── DevToolsHooks.ts # Core devtools functionality
54
+ │ ├── DevToolsModal.tsx # Modal wrapper component (iframe host)
55
+ │ └── index.ts # Main exports
56
+ └── scripts/ # Build scripts
57
+ ```
58
+
59
+ ## Development
60
+
61
+ The devtools package builds with `@assistant-ui/x-buildutils` to transpile the TypeScript sources that power the React helpers and adapters consumed across the workspace.
62
+
63
+ ## License
64
+
65
+ MIT
@@ -0,0 +1,8 @@
1
+ export interface DevToolsFrameProps {
2
+ frameUrl?: string;
3
+ className?: string;
4
+ style?: React.CSSProperties;
5
+ title?: string;
6
+ }
7
+ export declare const DevToolsFrame: React.FC<DevToolsFrameProps>;
8
+ //# sourceMappingURL=DevToolsFrame.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DevToolsFrame.d.ts","sourceRoot":"","sources":["../src/DevToolsFrame.tsx"],"names":[],"mappings":"AAMA,MAAM,WAAW,kBAAkB;IACjC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,KAAK,CAAC,EAAE,KAAK,CAAC,aAAa,CAAC;IAC5B,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,eAAO,MAAM,aAAa,EAAE,KAAK,CAAC,EAAE,CAAC,kBAAkB,CAyCtD,CAAC"}
@@ -0,0 +1,49 @@
1
+ "use client";
2
+
3
+ // src/DevToolsFrame.tsx
4
+ import { useCallback, useEffect, useMemo, useRef } from "react";
5
+ import { FrameHost } from "./FrameHost.js";
6
+ import { DEFAULT_FRAME_URL } from "./constants.js";
7
+ import { jsx } from "react/jsx-runtime";
8
+ var DevToolsFrame = ({
9
+ frameUrl = DEFAULT_FRAME_URL,
10
+ className,
11
+ style,
12
+ title = "assistant-ui DevTools"
13
+ }) => {
14
+ const iframeRef = useRef(null);
15
+ const frameHostRef = useRef(null);
16
+ const resolvedFrameUrl = useMemo(() => frameUrl, [frameUrl]);
17
+ const handleFrameLoad = useCallback(() => {
18
+ if (frameHostRef.current) {
19
+ frameHostRef.current.destroy();
20
+ frameHostRef.current = null;
21
+ }
22
+ if (iframeRef.current) {
23
+ frameHostRef.current = new FrameHost(iframeRef.current);
24
+ }
25
+ }, []);
26
+ useEffect(() => {
27
+ return () => {
28
+ if (frameHostRef.current) {
29
+ frameHostRef.current.destroy();
30
+ frameHostRef.current = null;
31
+ }
32
+ };
33
+ }, []);
34
+ return /* @__PURE__ */ jsx(
35
+ "iframe",
36
+ {
37
+ ref: iframeRef,
38
+ src: resolvedFrameUrl,
39
+ onLoad: handleFrameLoad,
40
+ className,
41
+ style,
42
+ title
43
+ }
44
+ );
45
+ };
46
+ export {
47
+ DevToolsFrame
48
+ };
49
+ //# sourceMappingURL=DevToolsFrame.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/DevToolsFrame.tsx"],"sourcesContent":["\"use client\";\n\nimport { useCallback, useEffect, useMemo, useRef } from \"react\";\nimport { FrameHost } from \"./FrameHost\";\nimport { DEFAULT_FRAME_URL } from \"./constants\";\n\nexport interface DevToolsFrameProps {\n frameUrl?: string;\n className?: string;\n style?: React.CSSProperties;\n title?: string;\n}\n\nexport const DevToolsFrame: React.FC<DevToolsFrameProps> = ({\n frameUrl = DEFAULT_FRAME_URL,\n className,\n style,\n title = \"assistant-ui DevTools\",\n}) => {\n const iframeRef = useRef<HTMLIFrameElement | null>(null);\n const frameHostRef = useRef<FrameHost | null>(null);\n\n const resolvedFrameUrl = useMemo(() => frameUrl, [frameUrl]);\n\n const handleFrameLoad = useCallback(() => {\n if (frameHostRef.current) {\n frameHostRef.current.destroy();\n frameHostRef.current = null;\n }\n\n if (iframeRef.current) {\n frameHostRef.current = new FrameHost(iframeRef.current);\n }\n }, []);\n\n useEffect(() => {\n return () => {\n if (frameHostRef.current) {\n frameHostRef.current.destroy();\n frameHostRef.current = null;\n }\n };\n }, []);\n\n return (\n <iframe\n ref={iframeRef}\n src={resolvedFrameUrl}\n onLoad={handleFrameLoad}\n className={className}\n style={style}\n title={title}\n />\n );\n};\n"],"mappings":";;;AAEA,SAAS,aAAa,WAAW,SAAS,cAAc;AACxD,SAAS,iBAAiB;AAC1B,SAAS,yBAAyB;AAyC9B;AAhCG,IAAM,gBAA8C,CAAC;AAAA,EAC1D,WAAW;AAAA,EACX;AAAA,EACA;AAAA,EACA,QAAQ;AACV,MAAM;AACJ,QAAM,YAAY,OAAiC,IAAI;AACvD,QAAM,eAAe,OAAyB,IAAI;AAElD,QAAM,mBAAmB,QAAQ,MAAM,UAAU,CAAC,QAAQ,CAAC;AAE3D,QAAM,kBAAkB,YAAY,MAAM;AACxC,QAAI,aAAa,SAAS;AACxB,mBAAa,QAAQ,QAAQ;AAC7B,mBAAa,UAAU;AAAA,IACzB;AAEA,QAAI,UAAU,SAAS;AACrB,mBAAa,UAAU,IAAI,UAAU,UAAU,OAAO;AAAA,IACxD;AAAA,EACF,GAAG,CAAC,CAAC;AAEL,YAAU,MAAM;AACd,WAAO,MAAM;AACX,UAAI,aAAa,SAAS;AACxB,qBAAa,QAAQ,QAAQ;AAC7B,qBAAa,UAAU;AAAA,MACzB;AAAA,IACF;AAAA,EACF,GAAG,CAAC,CAAC;AAEL,SACE;AAAA,IAAC;AAAA;AAAA,MACC,KAAK;AAAA,MACL,KAAK;AAAA,MACL,QAAQ;AAAA,MACR;AAAA,MACA;AAAA,MACA;AAAA;AAAA,EACF;AAEJ;","names":[]}
@@ -0,0 +1,35 @@
1
+ interface FrameToHostMessage {
2
+ type: "subscription" | "clearEvents";
3
+ data: {
4
+ apiList?: boolean;
5
+ apis?: number[];
6
+ apiId?: number;
7
+ };
8
+ }
9
+ interface HostToFrameMessage {
10
+ type: "update";
11
+ data: {
12
+ apiList?: Array<{
13
+ apiId: number;
14
+ }>;
15
+ apis?: Array<{
16
+ apiId: number;
17
+ state: any;
18
+ events: any[];
19
+ modelContext?: any;
20
+ }>;
21
+ };
22
+ }
23
+ export declare class DevToolsHost {
24
+ private subscription;
25
+ private unsubscribe?;
26
+ private onSendMessage;
27
+ constructor(onSendMessage: (message: HostToFrameMessage) => void);
28
+ onReceiveMessage(message: FrameToHostMessage): void;
29
+ private handleSubscription;
30
+ private subscribeToDevTools;
31
+ private sendUpdate;
32
+ destroy(): void;
33
+ }
34
+ export {};
35
+ //# sourceMappingURL=DevToolsHost.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DevToolsHost.d.ts","sourceRoot":"","sources":["../src/DevToolsHost.ts"],"names":[],"mappings":"AAMA,UAAU,kBAAkB;IAC1B,IAAI,EAAE,cAAc,GAAG,aAAa,CAAC;IACrC,IAAI,EAAE;QACJ,OAAO,CAAC,EAAE,OAAO,CAAC;QAClB,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;QAChB,KAAK,CAAC,EAAE,MAAM,CAAC;KAChB,CAAC;CACH;AAED,UAAU,kBAAkB;IAC1B,IAAI,EAAE,QAAQ,CAAC;IACf,IAAI,EAAE;QACJ,OAAO,CAAC,EAAE,KAAK,CAAC;YAAE,KAAK,EAAE,MAAM,CAAA;SAAE,CAAC,CAAC;QACnC,IAAI,CAAC,EAAE,KAAK,CAAC;YACX,KAAK,EAAE,MAAM,CAAC;YACd,KAAK,EAAE,GAAG,CAAC;YACX,MAAM,EAAE,GAAG,EAAE,CAAC;YACd,YAAY,CAAC,EAAE,GAAG,CAAC;SACpB,CAAC,CAAC;KACJ,CAAC;CACH;AAED,qBAAa,YAAY;IACvB,OAAO,CAAC,YAAY,CAMlB;IACF,OAAO,CAAC,WAAW,CAAC,CAAa;IACjC,OAAO,CAAC,aAAa,CAAwC;gBAEjD,aAAa,EAAE,CAAC,OAAO,EAAE,kBAAkB,KAAK,IAAI;IAKhE,gBAAgB,CAAC,OAAO,EAAE,kBAAkB;IAc5C,OAAO,CAAC,kBAAkB;IAiB1B,OAAO,CAAC,mBAAmB;IAM3B,OAAO,CAAC,UAAU;IA+DlB,OAAO;CAKR"}
@@ -0,0 +1,104 @@
1
+ // src/DevToolsHost.ts
2
+ import { DevToolsHooks } from "@assistant-ui/react";
3
+ import {
4
+ sanitizeForMessage,
5
+ serializeModelContext
6
+ } from "./utils/serialization.js";
7
+ var DevToolsHost = class {
8
+ subscription = {
9
+ apiList: false,
10
+ apis: /* @__PURE__ */ new Set()
11
+ };
12
+ unsubscribe;
13
+ onSendMessage;
14
+ constructor(onSendMessage) {
15
+ this.onSendMessage = onSendMessage;
16
+ this.subscribeToDevTools();
17
+ }
18
+ onReceiveMessage(message) {
19
+ switch (message.type) {
20
+ case "subscription":
21
+ this.handleSubscription(message.data);
22
+ break;
23
+ case "clearEvents":
24
+ if (typeof message.data.apiId === "number") {
25
+ DevToolsHooks.clearEventLogs(message.data.apiId);
26
+ }
27
+ break;
28
+ }
29
+ }
30
+ handleSubscription(data) {
31
+ const prevApiList = this.subscription.apiList;
32
+ const prevApis = new Set(this.subscription.apis);
33
+ this.subscription.apiList = data.apiList || false;
34
+ this.subscription.apis = new Set(data.apis);
35
+ const apisChanged = prevApis.size !== this.subscription.apis.size || [...this.subscription.apis].some((id) => !prevApis.has(id));
36
+ if (prevApiList !== this.subscription.apiList || apisChanged) {
37
+ this.sendUpdate();
38
+ }
39
+ }
40
+ subscribeToDevTools() {
41
+ this.unsubscribe = DevToolsHooks.subscribe(() => {
42
+ this.sendUpdate();
43
+ });
44
+ }
45
+ sendUpdate() {
46
+ const update = {
47
+ type: "update",
48
+ data: {}
49
+ };
50
+ const allApis = DevToolsHooks.getApis();
51
+ for (const subscriptionApiId of this.subscription.apis) {
52
+ if (!allApis.has(subscriptionApiId)) {
53
+ this.subscription.apis.delete(subscriptionApiId);
54
+ }
55
+ }
56
+ if (this.subscription.apiList) {
57
+ update.data.apiList = [...allApis.keys()].map((apiId) => ({ apiId }));
58
+ if (this.subscription.apis.size === 0 && allApis.size > 0) {
59
+ this.subscription.apis = /* @__PURE__ */ new Set([allApis.keys().next().value]);
60
+ }
61
+ }
62
+ if (this.subscription.apis.size > 0) {
63
+ update.data.apis = [];
64
+ for (const apiId of this.subscription.apis) {
65
+ const apiEntry = allApis.get(apiId);
66
+ if (apiEntry) {
67
+ const state = {};
68
+ if (apiEntry.api) {
69
+ for (const [name, scope] of Object.entries(apiEntry.api)) {
70
+ if (typeof scope === "function" && "source" in scope) {
71
+ if (scope.source === "root") {
72
+ const scopeValue = scope();
73
+ state[name] = scopeValue?.getState?.() ?? scopeValue;
74
+ }
75
+ }
76
+ }
77
+ }
78
+ const modelContext = serializeModelContext(
79
+ apiEntry.api?.thread?.().getModelContext()
80
+ );
81
+ update.data.apis.push({
82
+ apiId,
83
+ state: sanitizeForMessage(state),
84
+ events: sanitizeForMessage(apiEntry.logs),
85
+ modelContext
86
+ });
87
+ }
88
+ }
89
+ }
90
+ if (Object.keys(update.data).length === 0) {
91
+ return;
92
+ }
93
+ this.onSendMessage(update);
94
+ }
95
+ destroy() {
96
+ if (this.unsubscribe) {
97
+ this.unsubscribe();
98
+ }
99
+ }
100
+ };
101
+ export {
102
+ DevToolsHost
103
+ };
104
+ //# sourceMappingURL=DevToolsHost.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/DevToolsHost.ts"],"sourcesContent":["import { DevToolsHooks } from \"@assistant-ui/react\";\nimport {\n sanitizeForMessage,\n serializeModelContext,\n} from \"./utils/serialization\";\n\ninterface FrameToHostMessage {\n type: \"subscription\" | \"clearEvents\";\n data: {\n apiList?: boolean;\n apis?: number[];\n apiId?: number;\n };\n}\n\ninterface HostToFrameMessage {\n type: \"update\";\n data: {\n apiList?: Array<{ apiId: number }>;\n apis?: Array<{\n apiId: number;\n state: any;\n events: any[];\n modelContext?: any;\n }>;\n };\n}\n\nexport class DevToolsHost {\n private subscription: {\n apiList: boolean;\n apis: Set<number>;\n } = {\n apiList: false,\n apis: new Set(),\n };\n private unsubscribe?: () => void;\n private onSendMessage: (message: HostToFrameMessage) => void;\n\n constructor(onSendMessage: (message: HostToFrameMessage) => void) {\n this.onSendMessage = onSendMessage;\n this.subscribeToDevTools();\n }\n\n onReceiveMessage(message: FrameToHostMessage) {\n switch (message.type) {\n case \"subscription\":\n this.handleSubscription(message.data);\n break;\n case \"clearEvents\":\n if (typeof message.data.apiId === \"number\") {\n DevToolsHooks.clearEventLogs(message.data.apiId);\n // The subscription will automatically trigger an update\n }\n break;\n }\n }\n\n private handleSubscription(data: FrameToHostMessage[\"data\"]) {\n const prevApiList = this.subscription.apiList;\n const prevApis = new Set(this.subscription.apis);\n\n this.subscription.apiList = data.apiList || false;\n this.subscription.apis = new Set(data.apis);\n\n // Only send update if subscription actually changed\n const apisChanged =\n prevApis.size !== this.subscription.apis.size ||\n [...this.subscription.apis].some((id) => !prevApis.has(id));\n\n if (prevApiList !== this.subscription.apiList || apisChanged) {\n this.sendUpdate();\n }\n }\n\n private subscribeToDevTools() {\n this.unsubscribe = DevToolsHooks.subscribe(() => {\n this.sendUpdate();\n });\n }\n\n private sendUpdate() {\n const update: HostToFrameMessage = {\n type: \"update\",\n data: {},\n };\n\n const allApis = DevToolsHooks.getApis();\n for (const subscriptionApiId of this.subscription.apis) {\n if (!allApis.has(subscriptionApiId)) {\n this.subscription.apis.delete(subscriptionApiId);\n }\n }\n\n if (this.subscription.apiList) {\n update.data.apiList = [...allApis.keys()].map((apiId) => ({ apiId }));\n\n if (this.subscription.apis.size === 0 && allApis.size > 0) {\n this.subscription.apis = new Set([allApis.keys().next().value!]);\n }\n }\n\n if (this.subscription.apis.size > 0) {\n update.data.apis = [];\n\n for (const apiId of this.subscription.apis) {\n const apiEntry = allApis.get(apiId);\n if (apiEntry) {\n // Collect state from api scopes (only root source)\n const state: Record<string, unknown> = {};\n if (apiEntry.api) {\n for (const [name, scope] of Object.entries(apiEntry.api)) {\n if (typeof scope === \"function\" && \"source\" in scope) {\n // Only forward scopes with source === \"root\"\n if (scope.source === \"root\") {\n const scopeValue = scope();\n state[name] = scopeValue?.getState?.() ?? scopeValue;\n }\n }\n }\n }\n\n // Extract model context from thread runtime\n const modelContext = serializeModelContext(\n apiEntry.api?.thread?.().getModelContext(),\n );\n\n update.data.apis.push({\n apiId,\n state: sanitizeForMessage(state),\n events: sanitizeForMessage(apiEntry.logs) as unknown[],\n modelContext: modelContext,\n });\n }\n }\n }\n\n if (Object.keys(update.data).length === 0) {\n return;\n }\n\n this.onSendMessage(update);\n }\n\n destroy() {\n if (this.unsubscribe) {\n this.unsubscribe();\n }\n }\n}\n"],"mappings":"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","names":[]}
@@ -0,0 +1,2 @@
1
+ export declare const DevToolsModal: () => import("react/jsx-runtime").JSX.Element | null;
2
+ //# sourceMappingURL=DevToolsModal.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DevToolsModal.d.ts","sourceRoot":"","sources":["../src/DevToolsModal.tsx"],"names":[],"mappings":"AAsKA,eAAO,MAAM,aAAa,sDAWzB,CAAC"}
@@ -0,0 +1,170 @@
1
+ "use client";
2
+
3
+ // src/DevToolsModal.tsx
4
+ import { useEffect, useMemo, useState } from "react";
5
+ import { DevToolsFrame } from "./DevToolsFrame.js";
6
+ import { getStyles, ANIMATION_STYLES } from "./styles/DevToolsModal.styles.js";
7
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
8
+ var isDarkMode = () => {
9
+ if (typeof document === "undefined") return false;
10
+ return document.documentElement.classList.contains("dark") || document.body.classList.contains("dark");
11
+ };
12
+ var DevToolsModalImpl = () => {
13
+ const [isOpen, setIsOpen] = useState(false);
14
+ const [darkMode, setDarkMode] = useState(false);
15
+ const [buttonHover, setButtonHover] = useState(false);
16
+ const [closeHover, setCloseHover] = useState(false);
17
+ const styles = useMemo(() => getStyles(darkMode), [darkMode]);
18
+ useEffect(() => {
19
+ if (typeof document === "undefined") return;
20
+ const styleId = "devtools-modal-animations";
21
+ if (!document.getElementById(styleId)) {
22
+ const style = document.createElement("style");
23
+ style.id = styleId;
24
+ style.textContent = ANIMATION_STYLES;
25
+ document.head.appendChild(style);
26
+ }
27
+ return () => {
28
+ const style = document.getElementById(styleId);
29
+ if (style && !document.querySelector("[data-devtools-modal]")) {
30
+ style.remove();
31
+ }
32
+ };
33
+ }, []);
34
+ useEffect(() => {
35
+ if (typeof MutationObserver === "undefined") return;
36
+ const checkDarkMode = () => setDarkMode(isDarkMode());
37
+ const observer = new MutationObserver(checkDarkMode);
38
+ observer.observe(document.documentElement, {
39
+ attributes: true,
40
+ attributeFilter: ["class"]
41
+ });
42
+ if (document.body !== document.documentElement) {
43
+ observer.observe(document.body, {
44
+ attributes: true,
45
+ attributeFilter: ["class"]
46
+ });
47
+ }
48
+ return () => observer.disconnect();
49
+ }, []);
50
+ useEffect(() => {
51
+ if (!isOpen) return;
52
+ const handleEscape = (event) => {
53
+ if (event.key === "Escape") {
54
+ setIsOpen(false);
55
+ }
56
+ };
57
+ document.addEventListener("keydown", handleEscape);
58
+ return () => document.removeEventListener("keydown", handleEscape);
59
+ }, [isOpen]);
60
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
61
+ /* @__PURE__ */ jsx("div", { style: styles.floatingContainer, children: /* @__PURE__ */ jsx(
62
+ "button",
63
+ {
64
+ onClick: () => setIsOpen(true),
65
+ onMouseEnter: () => setButtonHover(true),
66
+ onMouseLeave: () => setButtonHover(false),
67
+ style: {
68
+ ...styles.floatingButton,
69
+ ...buttonHover ? styles.floatingButtonHover : {}
70
+ },
71
+ "aria-label": "Open assistant-ui DevTools",
72
+ title: "Open assistant-ui DevTools",
73
+ children: /* @__PURE__ */ jsx(
74
+ "svg",
75
+ {
76
+ width: "20",
77
+ height: "20",
78
+ viewBox: "0 0 24 24",
79
+ fill: "none",
80
+ xmlns: "http://www.w3.org/2000/svg",
81
+ style: { width: "20px", height: "20px" },
82
+ children: /* @__PURE__ */ jsx(
83
+ "path",
84
+ {
85
+ d: "M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z",
86
+ stroke: "currentColor",
87
+ strokeWidth: "1.8",
88
+ strokeLinecap: "round",
89
+ strokeLinejoin: "round",
90
+ fill: "none"
91
+ }
92
+ )
93
+ }
94
+ )
95
+ }
96
+ ) }),
97
+ isOpen && /* @__PURE__ */ jsxs(Fragment, { children: [
98
+ /* @__PURE__ */ jsx("div", { style: styles.backdrop, onClick: () => setIsOpen(false) }),
99
+ /* @__PURE__ */ jsxs("div", { style: styles.modal, "data-devtools-modal": true, children: [
100
+ /* @__PURE__ */ jsx(
101
+ "button",
102
+ {
103
+ onClick: () => setIsOpen(false),
104
+ onMouseEnter: () => setCloseHover(true),
105
+ onMouseLeave: () => setCloseHover(false),
106
+ style: {
107
+ ...styles.dismissButton,
108
+ ...closeHover ? styles.dismissButtonHover : {}
109
+ },
110
+ "aria-label": "Close DevTools",
111
+ children: /* @__PURE__ */ jsxs(
112
+ "svg",
113
+ {
114
+ width: "14",
115
+ height: "14",
116
+ viewBox: "0 0 24 24",
117
+ fill: "none",
118
+ xmlns: "http://www.w3.org/2000/svg",
119
+ children: [
120
+ /* @__PURE__ */ jsx(
121
+ "path",
122
+ {
123
+ d: "M18 6L6 18",
124
+ stroke: "currentColor",
125
+ strokeWidth: "1.6",
126
+ strokeLinecap: "round",
127
+ strokeLinejoin: "round"
128
+ }
129
+ ),
130
+ /* @__PURE__ */ jsx(
131
+ "path",
132
+ {
133
+ d: "M6 6L18 18",
134
+ stroke: "currentColor",
135
+ strokeWidth: "1.6",
136
+ strokeLinecap: "round",
137
+ strokeLinejoin: "round"
138
+ }
139
+ )
140
+ ]
141
+ }
142
+ )
143
+ }
144
+ ),
145
+ /* @__PURE__ */ jsx("div", { style: styles.modalContent, children: /* @__PURE__ */ jsx(
146
+ DevToolsFrame,
147
+ {
148
+ style: {
149
+ width: "100%",
150
+ height: "100%",
151
+ border: "none",
152
+ borderRadius: "12px",
153
+ backgroundColor: "transparent"
154
+ }
155
+ }
156
+ ) })
157
+ ] })
158
+ ] })
159
+ ] });
160
+ };
161
+ var DevToolsModal = () => {
162
+ if (typeof process !== "undefined" && process.env?.["NODE_ENV"] === "production") {
163
+ return null;
164
+ }
165
+ return /* @__PURE__ */ jsx(DevToolsModalImpl, {});
166
+ };
167
+ export {
168
+ DevToolsModal
169
+ };
170
+ //# sourceMappingURL=DevToolsModal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/DevToolsModal.tsx"],"sourcesContent":["\"use client\";\n\nimport { useEffect, useMemo, useState } from \"react\";\nimport { DevToolsFrame } from \"./DevToolsFrame\";\nimport { getStyles, ANIMATION_STYLES } from \"./styles/DevToolsModal.styles\";\n\nconst isDarkMode = (): boolean => {\n if (typeof document === \"undefined\") return false;\n return (\n document.documentElement.classList.contains(\"dark\") ||\n document.body.classList.contains(\"dark\")\n );\n};\n\nconst DevToolsModalImpl = () => {\n const [isOpen, setIsOpen] = useState(false);\n const [darkMode, setDarkMode] = useState(false);\n const [buttonHover, setButtonHover] = useState(false);\n const [closeHover, setCloseHover] = useState(false);\n\n const styles = useMemo(() => getStyles(darkMode), [darkMode]);\n\n useEffect(() => {\n if (typeof document === \"undefined\") return;\n\n const styleId = \"devtools-modal-animations\";\n if (!document.getElementById(styleId)) {\n const style = document.createElement(\"style\");\n style.id = styleId;\n style.textContent = ANIMATION_STYLES;\n document.head.appendChild(style);\n }\n\n return () => {\n const style = document.getElementById(styleId);\n if (style && !document.querySelector(\"[data-devtools-modal]\")) {\n style.remove();\n }\n };\n }, []);\n\n useEffect(() => {\n if (typeof MutationObserver === \"undefined\") return;\n\n const checkDarkMode = () => setDarkMode(isDarkMode());\n const observer = new MutationObserver(checkDarkMode);\n\n observer.observe(document.documentElement, {\n attributes: true,\n attributeFilter: [\"class\"],\n });\n if (document.body !== document.documentElement) {\n observer.observe(document.body, {\n attributes: true,\n attributeFilter: [\"class\"],\n });\n }\n\n return () => observer.disconnect();\n }, []);\n\n useEffect(() => {\n if (!isOpen) return;\n\n const handleEscape = (event: KeyboardEvent) => {\n if (event.key === \"Escape\") {\n setIsOpen(false);\n }\n };\n\n document.addEventListener(\"keydown\", handleEscape);\n return () => document.removeEventListener(\"keydown\", handleEscape);\n }, [isOpen]);\n\n return (\n <>\n <div style={styles.floatingContainer}>\n <button\n onClick={() => setIsOpen(true)}\n onMouseEnter={() => setButtonHover(true)}\n onMouseLeave={() => setButtonHover(false)}\n style={{\n ...styles.floatingButton,\n ...(buttonHover ? styles.floatingButtonHover : {}),\n }}\n aria-label=\"Open assistant-ui DevTools\"\n title=\"Open assistant-ui DevTools\"\n >\n <svg\n width=\"20\"\n height=\"20\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n style={{ width: \"20px\", height: \"20px\" }}\n >\n <path\n d=\"M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z\"\n stroke=\"currentColor\"\n strokeWidth=\"1.8\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n fill=\"none\"\n />\n </svg>\n </button>\n </div>\n\n {isOpen && (\n <>\n <div style={styles.backdrop} onClick={() => setIsOpen(false)} />\n\n <div style={styles.modal} data-devtools-modal>\n <button\n onClick={() => setIsOpen(false)}\n onMouseEnter={() => setCloseHover(true)}\n onMouseLeave={() => setCloseHover(false)}\n style={{\n ...styles.dismissButton,\n ...(closeHover ? styles.dismissButtonHover : {}),\n }}\n aria-label=\"Close DevTools\"\n >\n <svg\n width=\"14\"\n height=\"14\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <path\n d=\"M18 6L6 18\"\n stroke=\"currentColor\"\n strokeWidth=\"1.6\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n <path\n d=\"M6 6L18 18\"\n stroke=\"currentColor\"\n strokeWidth=\"1.6\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n </svg>\n </button>\n\n <div style={styles.modalContent}>\n <DevToolsFrame\n style={{\n width: \"100%\",\n height: \"100%\",\n border: \"none\",\n borderRadius: \"12px\",\n backgroundColor: \"transparent\",\n }}\n />\n </div>\n </div>\n </>\n )}\n </>\n );\n};\n\n// Export a component that only renders in development\nexport const DevToolsModal = () => {\n // Check if we're in production - most bundlers will replace process.env.NODE_ENV\n // This allows the entire component to be eliminated via dead code elimination\n if (\n typeof process !== \"undefined\" &&\n process.env?.[\"NODE_ENV\"] === \"production\"\n ) {\n return null;\n }\n\n return <DevToolsModalImpl />;\n};\n"],"mappings":"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","names":[]}
@@ -0,0 +1,7 @@
1
+ export declare class ExtensionHost {
2
+ private devToolsHost;
3
+ private messageListener;
4
+ constructor();
5
+ destroy(): void;
6
+ }
7
+ //# sourceMappingURL=ExtensionHost.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ExtensionHost.d.ts","sourceRoot":"","sources":["../src/ExtensionHost.ts"],"names":[],"mappings":"AAEA,qBAAa,aAAa;IACxB,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,eAAe,CAAgC;;IAgDvD,OAAO;CAIR"}
@@ -0,0 +1,50 @@
1
+ // src/ExtensionHost.ts
2
+ import { DevToolsHost } from "./DevToolsHost.js";
3
+ var ExtensionHost = class {
4
+ devToolsHost;
5
+ messageListener;
6
+ constructor() {
7
+ this.devToolsHost = new DevToolsHost((message) => {
8
+ console.log("[ExtensionHost] Sending message to iframe:", message);
9
+ window.postMessage(
10
+ {
11
+ source: "assistant-ui-devtools-page",
12
+ payload: message
13
+ },
14
+ "*"
15
+ );
16
+ });
17
+ this.messageListener = (event) => {
18
+ if (event.source !== window) return;
19
+ if (event.data.source === "assistant-ui-devtools-iframe") {
20
+ console.log("[ExtensionHost] Received message from iframe:", {
21
+ source: event.data.source,
22
+ payload: event.data.payload,
23
+ fullData: event.data
24
+ });
25
+ this.devToolsHost.onReceiveMessage(event.data.payload);
26
+ }
27
+ };
28
+ window.addEventListener("message", this.messageListener);
29
+ setTimeout(() => {
30
+ console.log("[ExtensionHost] Announcing connection to iframe");
31
+ window.postMessage(
32
+ {
33
+ source: "assistant-ui-devtools-page",
34
+ payload: {
35
+ type: "host-connected"
36
+ }
37
+ },
38
+ "*"
39
+ );
40
+ }, 100);
41
+ }
42
+ destroy() {
43
+ window.removeEventListener("message", this.messageListener);
44
+ this.devToolsHost.destroy();
45
+ }
46
+ };
47
+ export {
48
+ ExtensionHost
49
+ };
50
+ //# sourceMappingURL=ExtensionHost.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/ExtensionHost.ts"],"sourcesContent":["import { DevToolsHost } from \"./DevToolsHost\";\n\nexport class ExtensionHost {\n private devToolsHost: DevToolsHost;\n private messageListener: (event: MessageEvent) => void;\n\n constructor() {\n // Create DevToolsHost with callback to send messages through extension\n this.devToolsHost = new DevToolsHost((message) => {\n console.log(\"[ExtensionHost] Sending message to iframe:\", message);\n window.postMessage(\n {\n source: \"assistant-ui-devtools-page\",\n payload: message,\n },\n \"*\",\n );\n });\n\n // Setup listener to forward messages from extension to DevToolsHost\n this.messageListener = (event: MessageEvent) => {\n if (event.source !== window) return;\n\n // Log ALL messages to see what's coming through\n if (event.data.source === \"assistant-ui-devtools-iframe\") {\n console.log(\"[ExtensionHost] Received message from iframe:\", {\n source: event.data.source,\n payload: event.data.payload,\n fullData: event.data,\n });\n this.devToolsHost.onReceiveMessage(event.data.payload);\n }\n };\n\n window.addEventListener(\"message\", this.messageListener);\n\n // Announce that a new host has connected\n // This tells the iframe to re-send its subscription\n setTimeout(() => {\n console.log(\"[ExtensionHost] Announcing connection to iframe\");\n window.postMessage(\n {\n source: \"assistant-ui-devtools-page\",\n payload: {\n type: \"host-connected\",\n },\n },\n \"*\",\n );\n }, 100);\n }\n\n destroy() {\n window.removeEventListener(\"message\", this.messageListener);\n this.devToolsHost.destroy();\n }\n}\n"],"mappings":";AAAA,SAAS,oBAAoB;AAEtB,IAAM,gBAAN,MAAoB;AAAA,EACjB;AAAA,EACA;AAAA,EAER,cAAc;AAEZ,SAAK,eAAe,IAAI,aAAa,CAAC,YAAY;AAChD,cAAQ,IAAI,8CAA8C,OAAO;AACjE,aAAO;AAAA,QACL;AAAA,UACE,QAAQ;AAAA,UACR,SAAS;AAAA,QACX;AAAA,QACA;AAAA,MACF;AAAA,IACF,CAAC;AAGD,SAAK,kBAAkB,CAAC,UAAwB;AAC9C,UAAI,MAAM,WAAW,OAAQ;AAG7B,UAAI,MAAM,KAAK,WAAW,gCAAgC;AACxD,gBAAQ,IAAI,iDAAiD;AAAA,UAC3D,QAAQ,MAAM,KAAK;AAAA,UACnB,SAAS,MAAM,KAAK;AAAA,UACpB,UAAU,MAAM;AAAA,QAClB,CAAC;AACD,aAAK,aAAa,iBAAiB,MAAM,KAAK,OAAO;AAAA,MACvD;AAAA,IACF;AAEA,WAAO,iBAAiB,WAAW,KAAK,eAAe;AAIvD,eAAW,MAAM;AACf,cAAQ,IAAI,iDAAiD;AAC7D,aAAO;AAAA,QACL;AAAA,UACE,QAAQ;AAAA,UACR,SAAS;AAAA,YACP,MAAM;AAAA,UACR;AAAA,QACF;AAAA,QACA;AAAA,MACF;AAAA,IACF,GAAG,GAAG;AAAA,EACR;AAAA,EAEA,UAAU;AACR,WAAO,oBAAoB,WAAW,KAAK,eAAe;AAC1D,SAAK,aAAa,QAAQ;AAAA,EAC5B;AACF;","names":[]}
@@ -0,0 +1,35 @@
1
+ interface ApiData {
2
+ apiId: number;
3
+ state: any;
4
+ events: any[];
5
+ context?: any;
6
+ }
7
+ type UpdateListener = (data: {
8
+ apiList?: Array<{
9
+ apiId: number;
10
+ }>;
11
+ apis?: ApiData[];
12
+ }) => void;
13
+ export declare class FrameClient {
14
+ private listeners;
15
+ private connectionListeners;
16
+ private lastUpdate;
17
+ constructor();
18
+ private setupMessageListener;
19
+ onHostConnected(listener: () => void): () => void;
20
+ subscribe(listener: UpdateListener): () => void;
21
+ setSubscription(options: {
22
+ apiList?: boolean;
23
+ apis?: number[];
24
+ }): void;
25
+ clearEvents(apiId: number): void;
26
+ private notifyListeners;
27
+ getLastUpdate(): {
28
+ apiList?: Array<{
29
+ apiId: number;
30
+ }>;
31
+ apis?: ApiData[];
32
+ };
33
+ }
34
+ export {};
35
+ //# sourceMappingURL=FrameClient.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FrameClient.d.ts","sourceRoot":"","sources":["../src/FrameClient.ts"],"names":[],"mappings":"AAAA,UAAU,OAAO;IACf,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,GAAG,CAAC;IACX,MAAM,EAAE,GAAG,EAAE,CAAC;IACd,OAAO,CAAC,EAAE,GAAG,CAAC;CACf;AAcD,KAAK,cAAc,GAAG,CAAC,IAAI,EAAE;IAC3B,OAAO,CAAC,EAAE,KAAK,CAAC;QAAE,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACnC,IAAI,CAAC,EAAE,OAAO,EAAE,CAAC;CAClB,KAAK,IAAI,CAAC;AAEX,qBAAa,WAAW;IACtB,OAAO,CAAC,SAAS,CAA6B;IAC9C,OAAO,CAAC,mBAAmB,CAAyB;IACpD,OAAO,CAAC,UAAU,CAGX;;IAMP,OAAO,CAAC,oBAAoB;IAc5B,eAAe,CAAC,QAAQ,EAAE,MAAM,IAAI,GAAG,MAAM,IAAI;IAOjD,SAAS,CAAC,QAAQ,EAAE,cAAc,GAAG,MAAM,IAAI;IAa/C,eAAe,CAAC,OAAO,EAAE;QAAE,OAAO,CAAC,EAAE,OAAO,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAA;KAAE;IAU/D,WAAW,CAAC,KAAK,EAAE,MAAM;IAUzB,OAAO,CAAC,eAAe;IAIvB,aAAa;kBAlED,KAAK,CAAC;YAAE,KAAK,EAAE,MAAM,CAAA;SAAE,CAAC;eAC3B,OAAO,EAAE;;CAoEnB"}