@lark-apaas/aily-web-sdk 0.0.2-alpha.2 → 0.0.2-alpha.20

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
@@ -30,6 +30,16 @@ const chatPanel = await initAilyChat(container, {
30
30
  });
31
31
  ```
32
32
 
33
+ ```typescript
34
+ import { initAgentChat } from '@lark-apaas/aily-web-sdk'
35
+
36
+ const container = document.querySelector('.container');
37
+
38
+ const sdk = await initAgentChat(container, {
39
+ channelToken: 'wsk_xxx',
40
+ });
41
+ ```
42
+
33
43
  ## API
34
44
 
35
45
  ### `initAilyChat(root, config)`
@@ -53,6 +63,18 @@ const chatPanel = await initAilyChat(container, {
53
63
 
54
64
  **返回:** `Promise<ChatPanel>`
55
65
 
66
+ ### `initAgentChat(root, config)`
67
+
68
+ 初始化 Agent Chat 面板。
69
+
70
+ **参数:**
71
+
72
+ - `root` (`HTMLElement`) - iframe 挂载的容器元素
73
+ - `config` (`AgentSDKInitConfig`) - 配置项
74
+ - `channelToken` (`string`) - agent 渠道 token
75
+
76
+ **返回:** `Promise<AgentSDKInstance>`
77
+
56
78
  ### `ChatPanel`
57
79
 
58
80
  | 方法 | 说明 |
@@ -65,6 +87,12 @@ const chatPanel = await initAilyChat(container, {
65
87
  | `updateConfig(config)` | 更新面板配置 |
66
88
  | `destroy()` | 销毁实例 |
67
89
 
90
+ ### `AgentSDKInstance`
91
+
92
+ | 方法 | 说明 |
93
+ | --- | --- |
94
+ | `destroy()` | 销毁 agent iframe 实例 |
95
+
68
96
  ## License
69
97
 
70
98
  MIT
package/dist/index.cjs CHANGED
@@ -3,6 +3,7 @@ var __defProp = Object.defineProperty;
3
3
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
5
5
  var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
6
7
  var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
7
8
  var __export = (target, all) => {
8
9
  for (var name in all)
@@ -17,67 +18,93 @@ var __copyProps = (to, from, except, desc) => {
17
18
  return to;
18
19
  };
19
20
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
21
+ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
20
22
 
21
23
  // src/index.ts
22
24
  var index_exports = {};
23
25
  __export(index_exports, {
26
+ AGENT_IFRAME_BASE_URL: () => AGENT_IFRAME_BASE_URL,
24
27
  DEFAULT_ANONYMOUS_CHANNEL_TYPE: () => DEFAULT_ANONYMOUS_CHANNEL_TYPE,
25
28
  DEFAULT_CHANNEL_TYPE: () => DEFAULT_CHANNEL_TYPE,
26
29
  IFRAME_BASE_URL: () => IFRAME_BASE_URL,
27
30
  MESSAGE_TIMEOUT: () => MESSAGE_TIMEOUT,
31
+ getAgentIframeBaseURL: () => getAgentIframeBaseURL,
28
32
  getIframeBaseURL: () => getIframeBaseURL,
33
+ initAgentChat: () => initAgentChat,
29
34
  initAilyChat: () => initAilyChat
30
35
  });
31
36
  module.exports = __toCommonJS(index_exports);
32
37
 
33
38
  // src/constants.ts
34
39
  var MESSAGE_BIZ_ID = "cui-sdk-message";
40
+ var IFRAME_ORIGIN = {
41
+ production: "https://aily.feishu.cn",
42
+ staging: "https://aily.feishu-pre.cn",
43
+ boe: "https://aily.feishu-boe.cn",
44
+ development: "http://localhost:8080"
45
+ };
46
+ var CHAT_IFRAME_PATH = "/cui";
47
+ var AGENT_IFRAME_PATH = "/custom_agent/cui";
35
48
  var IFRAME_BASE_URL = {
36
- production: "https://aily.feishu.cn/cui",
37
- staging: "https://aily.feishu-pre.cn/cui",
38
- boe: "https://aily.feishu-boe.cn/cui",
39
- development: "http://localhost:8080/cui"
49
+ production: `${IFRAME_ORIGIN.production}${CHAT_IFRAME_PATH}`,
50
+ staging: `${IFRAME_ORIGIN.staging}${CHAT_IFRAME_PATH}`,
51
+ boe: `${IFRAME_ORIGIN.boe}${CHAT_IFRAME_PATH}`,
52
+ development: `${IFRAME_ORIGIN.development}${CHAT_IFRAME_PATH}`
40
53
  };
41
- function getIframeBaseURL(env) {
42
- if (env && IFRAME_BASE_URL[env]) {
43
- return IFRAME_BASE_URL[env];
54
+ var AGENT_IFRAME_BASE_URL = {
55
+ production: `${IFRAME_ORIGIN.production}${AGENT_IFRAME_PATH}`,
56
+ staging: `${IFRAME_ORIGIN.staging}${AGENT_IFRAME_PATH}`,
57
+ boe: `${IFRAME_ORIGIN.boe}${AGENT_IFRAME_PATH}`,
58
+ development: `${IFRAME_ORIGIN.development}${AGENT_IFRAME_PATH}`
59
+ };
60
+ function resolveIframeEnv(env) {
61
+ if (env && env in IFRAME_BASE_URL) {
62
+ return env;
44
63
  }
45
64
  const hostname = typeof window !== "undefined" ? window.location.hostname : "";
46
65
  if (hostname.includes("aiforce-boe")) {
47
- return IFRAME_BASE_URL.boe;
66
+ return "boe";
48
67
  }
49
- if (hostname.includes("aiforce-pre")) {
50
- return IFRAME_BASE_URL.staging;
68
+ if (hostname.includes("aiforce-pre") || hostname.includes("force-pre.feishuapp.net") || hostname.includes("fsapp.kundou.cn")) {
69
+ return "staging";
51
70
  }
52
71
  if (hostname.includes("localhost") || hostname.includes("127.0.0.1")) {
53
- return IFRAME_BASE_URL.development;
72
+ return "development";
54
73
  }
55
- return IFRAME_BASE_URL.production;
74
+ return "production";
75
+ }
76
+ __name(resolveIframeEnv, "resolveIframeEnv");
77
+ function getIframeBaseURL(env) {
78
+ return IFRAME_BASE_URL[resolveIframeEnv(env)];
56
79
  }
57
80
  __name(getIframeBaseURL, "getIframeBaseURL");
81
+ function getAgentIframeBaseURL(env) {
82
+ return AGENT_IFRAME_BASE_URL[resolveIframeEnv(env)];
83
+ }
84
+ __name(getAgentIframeBaseURL, "getAgentIframeBaseURL");
58
85
  var MESSAGE_TIMEOUT = 30 * 1e3;
59
86
  var DEFAULT_CHANNEL_TYPE = "MIAODA_CUI_SDK";
60
87
  var DEFAULT_ANONYMOUS_CHANNEL_TYPE = "MIAODA_ANONYMOUS_CUI_SDK";
61
88
  var ALLOWED_ORIGINS = [
62
- "https://aily.feishu.cn",
63
- "https://aily.feishu-pre.cn",
64
- "https://aily.feishu-boe.cn",
89
+ IFRAME_ORIGIN.production,
90
+ IFRAME_ORIGIN.staging,
91
+ IFRAME_ORIGIN.boe,
65
92
  "http://localhost:3000",
66
93
  "http://localhost:5173",
67
- "http://localhost:8080"
94
+ IFRAME_ORIGIN.development
68
95
  ];
69
96
 
97
+ // src/iframe-manager.ts
98
+ var import_internal_slardar2 = require("@lark-apaas/internal-slardar");
99
+
70
100
  // src/device-detector.ts
71
101
  var MOBILE_BREAKPOINT = 768;
72
- var DeviceDetector = class {
73
- static {
74
- __name(this, "DeviceDetector");
75
- }
76
- mediaQuery;
77
- currentDevice;
78
- callback = null;
79
- handleChange = null;
102
+ var _DeviceDetector = class _DeviceDetector {
80
103
  constructor() {
104
+ __publicField(this, "mediaQuery");
105
+ __publicField(this, "currentDevice");
106
+ __publicField(this, "callback", null);
107
+ __publicField(this, "handleChange", null);
81
108
  this.mediaQuery = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);
82
109
  this.currentDevice = window.innerWidth < MOBILE_BREAKPOINT ? "mobile" : "pc";
83
110
  }
@@ -103,19 +130,19 @@ var DeviceDetector = class {
103
130
  this.callback = null;
104
131
  }
105
132
  };
133
+ __name(_DeviceDetector, "DeviceDetector");
134
+ var DeviceDetector = _DeviceDetector;
106
135
 
107
136
  // src/message-bridge.ts
108
- var MessageBridge = class {
109
- static {
110
- __name(this, "MessageBridge");
111
- }
112
- iframe = null;
113
- requestId = 0;
114
- pendingRequests = /* @__PURE__ */ new Map();
115
- eventHandlers = /* @__PURE__ */ new Map();
116
- targetOrigin;
137
+ var import_internal_slardar = require("@lark-apaas/internal-slardar");
138
+ var _MessageBridge = class _MessageBridge {
117
139
  constructor(targetOrigin) {
118
- this.targetOrigin = targetOrigin;
140
+ __publicField(this, "iframe", null);
141
+ __publicField(this, "requestId", 0);
142
+ __publicField(this, "pendingRequests", /* @__PURE__ */ new Map());
143
+ __publicField(this, "eventHandlers", /* @__PURE__ */ new Map());
144
+ __publicField(this, "targetOrigin");
145
+ this.targetOrigin = new URL(targetOrigin).origin;
119
146
  this.handleMessage = this.handleMessage.bind(this);
120
147
  }
121
148
  /**
@@ -157,7 +184,13 @@ var MessageBridge = class {
157
184
  };
158
185
  const timer = setTimeout(() => {
159
186
  this.pendingRequests.delete(id);
160
- reject(new Error(`Message timeout: ${type}`));
187
+ const error = new Error(`Message timeout: ${type}`);
188
+ import_internal_slardar.slardar.captureException(error, {
189
+ source: "aily-web-sdk",
190
+ module: "message-bridge",
191
+ messageType: type
192
+ });
193
+ reject(error);
161
194
  }, MESSAGE_TIMEOUT);
162
195
  this.pendingRequests.set(id, {
163
196
  resolve,
@@ -200,7 +233,13 @@ var MessageBridge = class {
200
233
  clearTimeout(pending.timer);
201
234
  this.pendingRequests.delete(data.id);
202
235
  if (data.error) {
203
- pending.reject(new Error(data.error.message));
236
+ const error = new Error(data.error.message);
237
+ import_internal_slardar.slardar.captureException(error, {
238
+ source: "aily-web-sdk",
239
+ module: "message-bridge",
240
+ errorCode: data.error.code
241
+ });
242
+ pending.reject(error);
204
243
  } else {
205
244
  pending.resolve(data.data);
206
245
  }
@@ -223,19 +262,18 @@ var MessageBridge = class {
223
262
  return data.type === "event";
224
263
  }
225
264
  };
265
+ __name(_MessageBridge, "MessageBridge");
266
+ var MessageBridge = _MessageBridge;
226
267
 
227
268
  // src/iframe-manager.ts
228
- var IframeManager = class {
229
- static {
230
- __name(this, "IframeManager");
231
- }
232
- iframe = null;
233
- messageBridge;
234
- config;
235
- channelType;
236
- root;
237
- deviceDetector = null;
269
+ var _IframeManager = class _IframeManager {
238
270
  constructor(root, config) {
271
+ __publicField(this, "iframe", null);
272
+ __publicField(this, "messageBridge");
273
+ __publicField(this, "config");
274
+ __publicField(this, "channelType");
275
+ __publicField(this, "root");
276
+ __publicField(this, "deviceDetector", null);
239
277
  this.root = root;
240
278
  this.config = config;
241
279
  const defaultChannel = config.anonymous ? DEFAULT_ANONYMOUS_CHANNEL_TYPE : DEFAULT_CHANNEL_TYPE;
@@ -317,7 +355,7 @@ var IframeManager = class {
317
355
  },
318
356
  editor: editorConfig
319
357
  };
320
- const configBase64 = btoa(JSON.stringify(iframeConfig));
358
+ const configBase64 = btoa(encodeURIComponent(JSON.stringify(iframeConfig)));
321
359
  const params = new URLSearchParams({
322
360
  appKey: this.config.appKey,
323
361
  config: encodeURIComponent(configBase64)
@@ -330,7 +368,12 @@ var IframeManager = class {
330
368
  waitForReady() {
331
369
  return new Promise((resolve, reject) => {
332
370
  const timeout = setTimeout(() => {
333
- reject(new Error("Iframe ready timeout"));
371
+ const error = new Error("Iframe ready timeout");
372
+ import_internal_slardar2.slardar.captureException(error, {
373
+ source: "aily-web-sdk",
374
+ module: "iframe-ready"
375
+ });
376
+ reject(error);
334
377
  }, 3e4);
335
378
  const checkMessage = /* @__PURE__ */ __name((event) => {
336
379
  if (event.data?.messageBizId === MESSAGE_BIZ_ID && event.data?.type === "event" && event.data?.eventName === "ready") {
@@ -363,24 +406,24 @@ var IframeManager = class {
363
406
  if (events.onInitedWithWelcome) {
364
407
  this.messageBridge.onEvent("initedWithWelcome", events.onInitedWithWelcome);
365
408
  }
409
+ if (events.onClose) {
410
+ this.messageBridge.onEvent("onClose", events.onClose);
411
+ }
366
412
  }
367
413
  reportInitMetrics(initStart) {
368
414
  try {
369
415
  const cost = performance.now() - initStart;
370
- const Slardar = window.__Slardar;
371
- if (Slardar?.sendEvent) {
372
- Slardar.sendEvent({
373
- name: "cui_init_total",
374
- metrics: {
375
- cost
376
- },
377
- categories: {
378
- arch: "iframe-static",
379
- appKey: this.config.appKey,
380
- channelType: this.channelType
381
- }
382
- });
383
- }
416
+ import_internal_slardar2.slardar.sendEvent({
417
+ name: "cui_init_total",
418
+ metrics: {
419
+ cost
420
+ },
421
+ categories: {
422
+ arch: "iframe-static",
423
+ appKey: this.config.appKey,
424
+ channelType: this.channelType
425
+ }
426
+ });
384
427
  } catch {
385
428
  }
386
429
  }
@@ -420,6 +463,76 @@ var IframeManager = class {
420
463
  };
421
464
  }
422
465
  };
466
+ __name(_IframeManager, "IframeManager");
467
+ var IframeManager = _IframeManager;
468
+
469
+ // src/agent-iframe-manager.ts
470
+ var _AgentIframeManager = class _AgentIframeManager {
471
+ constructor(root, config) {
472
+ __publicField(this, "iframe", null);
473
+ __publicField(this, "messageBridge");
474
+ __publicField(this, "config");
475
+ __publicField(this, "root");
476
+ this.root = root;
477
+ this.config = config;
478
+ this.messageBridge = new MessageBridge(getAgentIframeBaseURL());
479
+ }
480
+ async init() {
481
+ this.iframe = document.createElement("iframe");
482
+ this.iframe.src = this.buildIframeURL();
483
+ this.iframe.style.width = "100%";
484
+ this.iframe.style.height = "100%";
485
+ this.iframe.style.border = "none";
486
+ this.iframe.allow = "microphone; clipboard-write";
487
+ this.root.appendChild(this.iframe);
488
+ this.messageBridge.attach(this.iframe);
489
+ await this.waitForReady();
490
+ return this.createAgentInstance();
491
+ }
492
+ async destroy() {
493
+ try {
494
+ await this.messageBridge.send("destroy");
495
+ } catch {
496
+ }
497
+ this.messageBridge.detach();
498
+ if (this.iframe?.parentNode) {
499
+ this.iframe.parentNode.removeChild(this.iframe);
500
+ }
501
+ this.iframe = null;
502
+ }
503
+ buildIframeURL() {
504
+ const baseURL = getAgentIframeBaseURL();
505
+ const params = new URLSearchParams({
506
+ channelToken: this.config.channelToken
507
+ });
508
+ return `${baseURL}?${params.toString()}`;
509
+ }
510
+ waitForReady() {
511
+ return new Promise((resolve, reject) => {
512
+ const timeout = setTimeout(() => {
513
+ window.removeEventListener("message", checkMessage);
514
+ reject(new Error("Iframe ready timeout"));
515
+ }, 3e4);
516
+ const checkMessage = /* @__PURE__ */ __name((event) => {
517
+ if (event.data?.messageBizId === MESSAGE_BIZ_ID && event.data?.type === "event" && event.data?.eventName === "ready") {
518
+ clearTimeout(timeout);
519
+ window.removeEventListener("message", checkMessage);
520
+ resolve();
521
+ }
522
+ }, "checkMessage");
523
+ window.addEventListener("message", checkMessage);
524
+ });
525
+ }
526
+ createAgentInstance() {
527
+ return {
528
+ destroy: /* @__PURE__ */ __name(async () => {
529
+ await this.destroy();
530
+ }, "destroy")
531
+ };
532
+ }
533
+ };
534
+ __name(_AgentIframeManager, "AgentIframeManager");
535
+ var AgentIframeManager = _AgentIframeManager;
423
536
 
424
537
  // src/index.ts
425
538
  async function initAilyChat(root, config) {
@@ -427,13 +540,21 @@ async function initAilyChat(root, config) {
427
540
  return manager.init();
428
541
  }
429
542
  __name(initAilyChat, "initAilyChat");
543
+ async function initAgentChat(root, config) {
544
+ const manager = new AgentIframeManager(root, config);
545
+ return manager.init();
546
+ }
547
+ __name(initAgentChat, "initAgentChat");
430
548
  // Annotate the CommonJS export names for ESM import in node:
431
549
  0 && (module.exports = {
550
+ AGENT_IFRAME_BASE_URL,
432
551
  DEFAULT_ANONYMOUS_CHANNEL_TYPE,
433
552
  DEFAULT_CHANNEL_TYPE,
434
553
  IFRAME_BASE_URL,
435
554
  MESSAGE_TIMEOUT,
555
+ getAgentIframeBaseURL,
436
556
  getIframeBaseURL,
557
+ initAgentChat,
437
558
  initAilyChat
438
559
  });
439
560
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/constants.ts","../src/device-detector.ts","../src/message-bridge.ts","../src/iframe-manager.ts"],"sourcesContent":["/*\n * index.ts\n * Author: perterpon.wang<perterpon.wang@bytedance.com>\n * Create: Tue Feb 10 2026\n */\n\nimport { IframeManager } from \"./iframe-manager\";\n\nimport type { ChatPanel, WebSDKConfig } from \"./types/config\";\n\n/**\n * Initialize Aily Chat via iframe\n * This is the primary method for initializing the chat panel\n * @param root - Container element for the iframe\n * @param config - Configuration options\n * @returns ChatPanel instance for controlling the chat\n */\nexport async function initAilyChat(\n\troot: HTMLElement,\n\tconfig: WebSDKConfig,\n): Promise<ChatPanel> {\n\tconst manager = new IframeManager(root, config);\n\treturn manager.init();\n}\n\n// Export constants\nexport {\n\tDEFAULT_ANONYMOUS_CHANNEL_TYPE,\n\tDEFAULT_CHANNEL_TYPE,\n\tgetIframeBaseURL,\n\tIFRAME_BASE_URL,\n\tMESSAGE_TIMEOUT,\n} from \"./constants\";\n// Export types\nexport type {\n\tChatPanel,\n\tWebSDKCommonConfig,\n\tWebSDKConfig,\n\tWebSDKConversionConfig,\n\tWebSDKEditorConfig,\n\tWebSDKEvents,\n\tWebSDKSkillInfo,\n} from \"./types/config\";\n\n// Export message types\nexport type {\n\tIframeEventMessage,\n\tIframeMessage,\n\tIframeMessageType,\n\tIframeRequestMessage,\n\tIframeResponseMessage,\n} from \"./types/message\";\n","/**\n * Constants for Aily Web SDK\n */\n\n/** Message business ID for postMessage identification */\nexport const MESSAGE_BIZ_ID = \"cui-sdk-message\" as const;\n\n/** Iframe base URLs by environment */\nexport const IFRAME_BASE_URL: Record<string, string> = {\n\tproduction: \"https://aily.feishu.cn/cui\",\n\tstaging: \"https://aily.feishu-pre.cn/cui\",\n\tboe: \"https://aily.feishu-boe.cn/cui\",\n\tdevelopment: \"http://localhost:8080/cui\",\n};\n\n/**\n * Get iframe base URL based on environment\n */\nexport function getIframeBaseURL(env?: string): string {\n\tif (env && IFRAME_BASE_URL[env]) {\n\t\treturn IFRAME_BASE_URL[env];\n\t}\n\n\t// Auto-detect based on hostname\n\t// MiaoDa 宿主域名: miaoda.feishu[-env].cn, *.aiforce[-env][-preview].bytedance.net, *.aiforce.run, *.aiforce.cloud\n\tconst hostname =\n\t\ttypeof window !== \"undefined\" ? window.location.hostname : \"\";\n\n\tif (hostname.includes(\"aiforce-boe\")) {\n\t\treturn IFRAME_BASE_URL.boe;\n\t}\n\tif (hostname.includes(\"aiforce-pre\")) {\n\t\treturn IFRAME_BASE_URL.staging;\n\t}\n\tif (hostname.includes(\"localhost\") || hostname.includes(\"127.0.0.1\")) {\n\t\treturn IFRAME_BASE_URL.development;\n\t}\n\n\treturn IFRAME_BASE_URL.production;\n}\n\n/** Timeout for postMessage requests (30 seconds) */\nexport const MESSAGE_TIMEOUT = 30 * 1000;\n\n/** Miaoda 渠道标识默认值(用户可通过 config.common.channelType 覆盖) */\nexport const DEFAULT_CHANNEL_TYPE = \"MIAODA_CUI_SDK\";\nexport const DEFAULT_ANONYMOUS_CHANNEL_TYPE = \"MIAODA_ANONYMOUS_CUI_SDK\";\n\n/** Allowed origins for postMessage (cui-iframe 的 origin) */\nexport const ALLOWED_ORIGINS = [\n\t\"https://aily.feishu.cn\",\n\t\"https://aily.feishu-pre.cn\",\n\t\"https://aily.feishu-boe.cn\",\n\t\"http://localhost:3000\",\n\t\"http://localhost:5173\",\n\t\"http://localhost:8080\",\n];\n","const MOBILE_BREAKPOINT = 768;\n\ntype DeviceType = 'pc' | 'mobile';\ntype DeviceChangeCallback = (device: DeviceType) => void;\n\n/**\n * 设备类型自动检测器\n *\n * 使用 matchMedia + 768px 断点检测设备类型,与 miaoda useIsMobile 逻辑一致。\n * 检测在宿主页面(而非 iframe)中执行,结果通过 postMessage 推送给 iframe。\n */\nexport class DeviceDetector {\n private mediaQuery: MediaQueryList;\n private currentDevice: DeviceType;\n private callback: DeviceChangeCallback | null = null;\n private handleChange: (() => void) | null = null;\n\n constructor() {\n this.mediaQuery = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);\n this.currentDevice = window.innerWidth < MOBILE_BREAKPOINT ? 'mobile' : 'pc';\n }\n\n getDevice(): DeviceType {\n return this.currentDevice;\n }\n\n observe(callback: DeviceChangeCallback): void {\n this.callback = callback;\n this.handleChange = () => {\n const newDevice: DeviceType = window.innerWidth < MOBILE_BREAKPOINT ? 'mobile' : 'pc';\n if (newDevice !== this.currentDevice) {\n this.currentDevice = newDevice;\n this.callback?.(newDevice);\n }\n };\n this.mediaQuery.addEventListener('change', this.handleChange);\n }\n\n destroy(): void {\n if (this.handleChange) {\n this.mediaQuery.removeEventListener('change', this.handleChange);\n this.handleChange = null;\n }\n this.callback = null;\n }\n}\n","import { MESSAGE_TIMEOUT, ALLOWED_ORIGINS } from './constants';\nimport {\n MESSAGE_BIZ_ID,\n type IframeMessage,\n type IframeRequestMessage,\n type IframeResponseMessage,\n type IframeEventMessage,\n type IframeMessageType,\n} from './types/message';\n\ninterface PendingRequest {\n resolve: (value: unknown) => void;\n reject: (reason: Error) => void;\n timer: ReturnType<typeof setTimeout>;\n}\n\n/**\n * MessageBridge handles postMessage communication between parent and iframe\n */\nexport class MessageBridge {\n private iframe: HTMLIFrameElement | null = null;\n private requestId = 0;\n private pendingRequests = new Map<number, PendingRequest>();\n private eventHandlers = new Map<string, ((data: unknown) => void)[]>();\n private targetOrigin: string;\n\n constructor(targetOrigin: string) {\n this.targetOrigin = targetOrigin;\n this.handleMessage = this.handleMessage.bind(this);\n }\n\n /**\n * Attach to an iframe and start listening for messages\n */\n attach(iframe: HTMLIFrameElement): void {\n this.iframe = iframe;\n window.addEventListener('message', this.handleMessage);\n }\n\n /**\n * Detach from iframe and clean up\n */\n detach(): void {\n window.removeEventListener('message', this.handleMessage);\n this.iframe = null;\n // Clear all pending requests\n this.pendingRequests.forEach(({ reject }) => {\n reject(new Error('MessageBridge detached'));\n });\n this.pendingRequests.clear();\n }\n\n /**\n * Send a message to the iframe and wait for response\n */\n send<T = unknown>(type: IframeMessageType, data?: unknown): Promise<T> {\n return new Promise((resolve, reject) => {\n if (!this.iframe?.contentWindow) {\n reject(new Error('Iframe not attached'));\n return;\n }\n\n this.requestId += 1;\n const id = this.requestId;\n const message: IframeRequestMessage = {\n messageBizId: MESSAGE_BIZ_ID,\n id,\n type,\n data,\n timestamp: Date.now(),\n isRequest: true,\n };\n\n const timer = setTimeout(() => {\n this.pendingRequests.delete(id);\n reject(new Error(`Message timeout: ${type}`));\n }, MESSAGE_TIMEOUT);\n\n this.pendingRequests.set(id, { resolve: resolve as (value: unknown) => void, reject, timer });\n\n this.iframe.contentWindow.postMessage(message, this.targetOrigin);\n });\n }\n\n /**\n * Register an event handler\n */\n onEvent(eventName: string, handler: (data: unknown) => void): void {\n const handlers = this.eventHandlers.get(eventName) || [];\n handlers.push(handler);\n this.eventHandlers.set(eventName, handlers);\n }\n\n /**\n * Remove an event handler\n */\n offEvent(eventName: string, handler: (data: unknown) => void): void {\n const handlers = this.eventHandlers.get(eventName) || [];\n const index = handlers.indexOf(handler);\n if (index > -1) {\n handlers.splice(index, 1);\n this.eventHandlers.set(eventName, handlers);\n }\n }\n\n private handleMessage(event: MessageEvent<IframeMessage>): void {\n // Validate origin\n if (!ALLOWED_ORIGINS.includes(event.origin)) {\n return;\n }\n\n const { data } = event;\n\n // Validate message format\n if (data?.messageBizId !== MESSAGE_BIZ_ID) {\n return;\n }\n\n // Handle response messages\n if (this.isResponseMessage(data)) {\n const pending = this.pendingRequests.get(data.id);\n if (pending) {\n clearTimeout(pending.timer);\n this.pendingRequests.delete(data.id);\n\n if (data.error) {\n pending.reject(new Error(data.error.message));\n } else {\n pending.resolve(data.data);\n }\n }\n }\n\n // Handle event messages\n if (this.isEventMessage(data)) {\n const handlers = this.eventHandlers.get(data.eventName) || [];\n handlers.forEach((handler) => {\n try {\n handler(data.data);\n } catch {\n // Ignore handler errors\n }\n });\n }\n }\n\n private isResponseMessage(data: IframeMessage): data is IframeResponseMessage {\n return 'isResponse' in data && data.isResponse === true;\n }\n\n private isEventMessage(data: IframeMessage): data is IframeEventMessage {\n return data.type === 'event';\n }\n}\n","import {\n\tDEFAULT_ANONYMOUS_CHANNEL_TYPE,\n\tDEFAULT_CHANNEL_TYPE,\n\tgetIframeBaseURL,\n\tMESSAGE_BIZ_ID,\n} from \"./constants\";\nimport { DeviceDetector } from \"./device-detector\";\nimport { MessageBridge } from \"./message-bridge\";\n\nimport type { ChatPanel, WebSDKConfig } from \"./types/config\";\n\n/**\n * IframeManager handles iframe creation and lifecycle\n *\n * iframe 采用自初始化模式:配置通过 URL hash 传入,iframe 加载后自行解析并渲染,\n * 与旧版 copilot iframe 的行为保持一致。宿主通过 postMessage 进行后续操作。\n */\nexport class IframeManager {\n\tprivate iframe: HTMLIFrameElement | null = null;\n\tprivate messageBridge: MessageBridge;\n\tprivate config: WebSDKConfig;\n\tprivate channelType: string;\n\tprivate root: HTMLElement;\n\tprivate deviceDetector: DeviceDetector | null = null;\n\n\tconstructor(root: HTMLElement, config: WebSDKConfig) {\n\t\tthis.root = root;\n\t\tthis.config = config;\n\t\tconst defaultChannel = config.anonymous\n\t\t\t? DEFAULT_ANONYMOUS_CHANNEL_TYPE\n\t\t\t: DEFAULT_CHANNEL_TYPE;\n\t\tthis.channelType = config.common?.channelType || defaultChannel;\n\t\tthis.messageBridge = new MessageBridge(getIframeBaseURL());\n\t}\n\n\t/**\n\t * Initialize the iframe and establish communication\n\t * 配置通过 URL hash 编码传入 iframe,iframe 加载后自行初始化\n\t */\n\tasync init(): Promise<ChatPanel> {\n\t\tconst initStart = performance.now();\n\n\t\t// Auto-detect device if not explicitly provided\n\t\tif (!this.config.device) {\n\t\t\tthis.deviceDetector = new DeviceDetector();\n\t\t\tthis.config.device = this.deviceDetector.getDevice();\n\t\t}\n\n\t\t// Create iframe element\n\t\tthis.iframe = document.createElement(\"iframe\");\n\t\tthis.iframe.src = this.buildIframeURL();\n\t\tthis.iframe.style.width = \"100%\";\n\t\tthis.iframe.style.height = \"100%\";\n\t\tthis.iframe.style.border = \"none\";\n\t\tthis.iframe.allow = \"microphone; clipboard-write\";\n\n\t\t// Attach to DOM\n\t\tthis.root.appendChild(this.iframe);\n\n\t\t// Attach message bridge (开始监听消息)\n\t\tthis.messageBridge.attach(this.iframe);\n\n\t\t// Setup event handlers (在 iframe 初始化完成之前就开始监听,确保不丢失事件)\n\t\tthis.setupEventHandlers();\n\n\t\t// Listen for device changes and push updates to iframe\n\t\tif (this.deviceDetector) {\n\t\t\tthis.deviceDetector.observe((newDevice) => {\n\t\t\t\tthis.messageBridge.send(\"updateConfig\", { device: newDevice });\n\t\t\t});\n\t\t}\n\n\t\t// Wait for iframe to finish initialization (iframe 自行读取 URL 配置并渲染,完成后发送 ready 事件)\n\t\tawait this.waitForReady();\n\n\t\t// 上报 cui_init_total 耗时(iframe 创建 → init 完成)\n\t\tthis.reportInitMetrics(initStart);\n\n\t\t// Return ChatPanel interface\n\t\treturn this.createChatPanel();\n\t}\n\n\t/**\n\t * Destroy the iframe and cleanup\n\t */\n\tasync destroy(): Promise<void> {\n\t\ttry {\n\t\t\tawait this.messageBridge.send(\"destroy\");\n\t\t} catch {\n\t\t\t// Ignore errors during destroy\n\t\t}\n\n\t\tthis.messageBridge.detach();\n\n\t\tif (this.deviceDetector) {\n\t\t\tthis.deviceDetector.destroy();\n\t\t\tthis.deviceDetector = null;\n\t\t}\n\n\t\tif (this.iframe?.parentNode) {\n\t\t\tthis.iframe.parentNode.removeChild(this.iframe);\n\t\t}\n\n\t\tthis.iframe = null;\n\t}\n\n\t/**\n\t * 构建 iframe URL,将配置编码到 hash 中\n\t * 格式: {baseURL}#{params} 其中 config 为 base64(JSON.stringify(configWithoutEvents))\n\t *\n\t * 鉴权说明:SDK 不参与 token 获取/传递。iframe 自行处理飞书登录流程,\n\t * 允许用户在 iframe 中独立登录。匿名渠道跳过登录检查。\n\t */\n\tprivate buildIframeURL(): string {\n\t\tconst baseURL = getIframeBaseURL();\n\n\t\t// 移除 events(函数无法序列化)\n\t\tconst {\n\t\t\tevents: _events,\n\t\t\tanonymous: _anonymous,\n\t\t\t...configWithoutEvents\n\t\t} = this.config;\n\t\t// 注入 channelType/resourceType(SDK 内部控制,不暴露给消费者)\n\t\t// 补全 editor.skill 硬编码字段\n\t\tconst editorConfig = configWithoutEvents.editor?.skill\n\t\t\t? {\n\t\t\t\t\t...configWithoutEvents.editor,\n\t\t\t\t\tskill: {\n\t\t\t\t\t\ttype: \"custom\",\n\t\t\t\t\t\tbuiltinType: \"unspecified\",\n\t\t\t\t\t\t...configWithoutEvents.editor.skill,\n\t\t\t\t\t},\n\t\t\t\t}\n\t\t\t: configWithoutEvents.editor;\n\n\t\tconst iframeConfig = {\n\t\t\t...configWithoutEvents,\n\t\t\tcommon: {\n\t\t\t\t...configWithoutEvents.common,\n\t\t\t\tchannelType: this.channelType,\n\t\t\t\tresourceType:\n\t\t\t\t\tconfigWithoutEvents.common?.resourceType || this.channelType,\n\t\t\t},\n\t\t\teditor: editorConfig,\n\t\t};\n\t\tconst configBase64 = btoa(JSON.stringify(iframeConfig));\n\n\t\tconst params = new URLSearchParams({\n\t\t\tappKey: this.config.appKey,\n\t\t\tconfig: encodeURIComponent(configBase64),\n\t\t});\n\n\t\treturn `${baseURL}#${params.toString()}`;\n\t}\n\n\t/**\n\t * 等待 iframe 自行初始化完成后发送 ready 事件\n\t */\n\tprivate waitForReady(): Promise<void> {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tconst timeout = setTimeout(() => {\n\t\t\t\treject(new Error(\"Iframe ready timeout\"));\n\t\t\t}, 30000);\n\n\t\t\tconst checkMessage = (event: MessageEvent) => {\n\t\t\t\tif (\n\t\t\t\t\tevent.data?.messageBizId === MESSAGE_BIZ_ID &&\n\t\t\t\t\tevent.data?.type === \"event\" &&\n\t\t\t\t\tevent.data?.eventName === \"ready\"\n\t\t\t\t) {\n\t\t\t\t\tclearTimeout(timeout);\n\t\t\t\t\twindow.removeEventListener(\"message\", checkMessage);\n\t\t\t\t\tresolve();\n\t\t\t\t}\n\t\t\t};\n\n\t\t\twindow.addEventListener(\"message\", checkMessage);\n\t\t});\n\t}\n\n\tprivate setupEventHandlers(): void {\n\t\tconst { events } = this.config;\n\t\tif (!events) return;\n\n\t\tif (events.onReady) {\n\t\t\tthis.messageBridge.onEvent(\"ready\", events.onReady);\n\t\t}\n\n\t\tif (events.onError) {\n\t\t\tthis.messageBridge.onEvent(\"error\", (data: unknown) => {\n\t\t\t\tconst msg = (data as { message?: string })?.message || \"Unknown error\";\n\t\t\t\tevents.onError?.(new Error(msg));\n\t\t\t});\n\t\t}\n\n\t\tif (events.onMessage) {\n\t\t\tthis.messageBridge.onEvent(\"onMessage\", events.onMessage);\n\t\t}\n\n\t\tif (events.onInited) {\n\t\t\tthis.messageBridge.onEvent(\"inited\", events.onInited);\n\t\t}\n\n\t\tif (events.onInitedWithWelcome) {\n\t\t\tthis.messageBridge.onEvent(\n\t\t\t\t\"initedWithWelcome\",\n\t\t\t\tevents.onInitedWithWelcome,\n\t\t\t);\n\t\t}\n\t}\n\n\tprivate reportInitMetrics(initStart: number): void {\n\t\ttry {\n\t\t\tconst cost = performance.now() - initStart;\n\t\t\tconst Slardar = (window as unknown as Record<string, unknown>)\n\t\t\t\t.__Slardar as\n\t\t\t\t| { sendEvent?: (event: Record<string, unknown>) => void }\n\t\t\t\t| undefined;\n\t\t\tif (Slardar?.sendEvent) {\n\t\t\t\tSlardar.sendEvent({\n\t\t\t\t\tname: \"cui_init_total\",\n\t\t\t\t\tmetrics: { cost },\n\t\t\t\t\tcategories: {\n\t\t\t\t\t\tarch: \"iframe-static\",\n\t\t\t\t\t\tappKey: this.config.appKey,\n\t\t\t\t\t\tchannelType: this.channelType,\n\t\t\t\t\t},\n\t\t\t\t});\n\t\t\t}\n\t\t} catch {\n\t\t\t// 性能上报失败不影响功能\n\t\t}\n\t}\n\n\tprivate createChatPanel(): ChatPanel {\n\t\treturn {\n\t\t\tsendMessage: async (data) => {\n\t\t\t\tawait this.messageBridge.send(\"sendMessage\", data);\n\t\t\t},\n\t\t\tclear: async () => {\n\t\t\t\tawait this.messageBridge.send(\"clear\");\n\t\t\t},\n\t\t\tcancelMessage: async (messageItem) => {\n\t\t\t\tawait this.messageBridge.send(\"cancelMessage\", messageItem);\n\t\t\t},\n\t\t\tclearAndStop: async () => {\n\t\t\t\tawait this.messageBridge.send(\"clearAndStop\");\n\t\t\t},\n\t\t\tupdateWelcomeMessage: async () => {\n\t\t\t\tawait this.messageBridge.send(\"updateWelcomeMessage\");\n\t\t\t},\n\t\t\tupdateConfig: async (config) => {\n\t\t\t\tawait this.messageBridge.send(\"updateConfig\", config);\n\t\t\t},\n\t\t\tobserveSkill: async (skillId, name, skillType) => {\n\t\t\t\tawait this.messageBridge.send(\"observeSkill\", {\n\t\t\t\t\tskillId,\n\t\t\t\t\tname,\n\t\t\t\t\tskillType,\n\t\t\t\t});\n\t\t\t},\n\t\t\tsetCurrentSkill: async (skill) => {\n\t\t\t\tawait this.messageBridge.send(\"setCurrentSkill\", skill);\n\t\t\t},\n\t\t\tdestroy: async () => {\n\t\t\t\tawait this.destroy();\n\t\t\t},\n\t\t};\n\t}\n}\n"],"mappings":"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","names":["MESSAGE_BIZ_ID","IFRAME_BASE_URL","production","staging","boe","development","getIframeBaseURL","env","hostname","window","location","includes","MESSAGE_TIMEOUT","DEFAULT_CHANNEL_TYPE","DEFAULT_ANONYMOUS_CHANNEL_TYPE","ALLOWED_ORIGINS","MOBILE_BREAKPOINT","DeviceDetector","mediaQuery","currentDevice","callback","handleChange","window","matchMedia","innerWidth","getDevice","observe","newDevice","addEventListener","destroy","removeEventListener","MessageBridge","iframe","requestId","pendingRequests","Map","eventHandlers","targetOrigin","handleMessage","bind","attach","window","addEventListener","detach","removeEventListener","forEach","reject","Error","clear","send","type","data","Promise","resolve","contentWindow","id","message","messageBizId","MESSAGE_BIZ_ID","timestamp","Date","now","isRequest","timer","setTimeout","delete","MESSAGE_TIMEOUT","set","postMessage","onEvent","eventName","handler","handlers","get","push","offEvent","index","indexOf","splice","event","ALLOWED_ORIGINS","includes","origin","isResponseMessage","pending","clearTimeout","error","isEventMessage","isResponse","IframeManager","iframe","messageBridge","config","channelType","root","deviceDetector","defaultChannel","anonymous","DEFAULT_ANONYMOUS_CHANNEL_TYPE","DEFAULT_CHANNEL_TYPE","common","MessageBridge","getIframeBaseURL","init","initStart","performance","now","device","DeviceDetector","getDevice","document","createElement","src","buildIframeURL","style","width","height","border","allow","appendChild","attach","setupEventHandlers","observe","newDevice","send","waitForReady","reportInitMetrics","createChatPanel","destroy","detach","parentNode","removeChild","baseURL","events","_events","_anonymous","configWithoutEvents","editorConfig","editor","skill","type","builtinType","iframeConfig","resourceType","configBase64","btoa","JSON","stringify","params","URLSearchParams","appKey","encodeURIComponent","toString","Promise","resolve","reject","timeout","setTimeout","Error","checkMessage","event","data","messageBizId","MESSAGE_BIZ_ID","eventName","clearTimeout","window","removeEventListener","addEventListener","onReady","onEvent","onError","msg","message","onMessage","onInited","onInitedWithWelcome","cost","Slardar","__Slardar","sendEvent","name","metrics","categories","arch","sendMessage","clear","cancelMessage","messageItem","clearAndStop","updateWelcomeMessage","updateConfig","observeSkill","skillId","skillType","setCurrentSkill","initAilyChat","root","config","manager","IframeManager","init"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/constants.ts","../src/iframe-manager.ts","../src/device-detector.ts","../src/message-bridge.ts","../src/agent-iframe-manager.ts"],"sourcesContent":["/*\n * index.ts\n * Author: perterpon.wang<perterpon.wang@bytedance.com>\n * Create: Tue Feb 10 2026\n */\n\nimport { IframeManager } from \"./iframe-manager\";\nimport { AgentIframeManager } from \"./agent-iframe-manager\";\n\nimport type {\n\tAgentSDKInitConfig,\n\tAgentSDKInstance,\n\tChatPanel,\n\tWebSDKConfig,\n} from \"./types/config\";\n\n/**\n * Initialize Aily Chat via iframe\n * This is the primary method for initializing the chat panel\n * @param root - Container element for the iframe\n * @param config - Configuration options\n * @returns ChatPanel instance for controlling the chat\n */\nexport async function initAilyChat(\n\troot: HTMLElement,\n\tconfig: WebSDKConfig,\n): Promise<ChatPanel> {\n\tconst manager = new IframeManager(root, config);\n\treturn manager.init();\n}\n\n/**\n * Initialize Agent Chat via iframe\n * @param root - Container element for the iframe\n * @param config - Agent SDK configuration\n * @returns Agent SDK instance for controlling the agent page\n */\nexport async function initAgentChat(\n\troot: HTMLElement,\n\tconfig: AgentSDKInitConfig,\n): Promise<AgentSDKInstance> {\n\tconst manager = new AgentIframeManager(root, config);\n\treturn manager.init();\n}\n\n// Export constants\nexport {\n\tAGENT_IFRAME_BASE_URL,\n\tDEFAULT_ANONYMOUS_CHANNEL_TYPE,\n\tDEFAULT_CHANNEL_TYPE,\n\tgetAgentIframeBaseURL,\n\tgetIframeBaseURL,\n\tIFRAME_BASE_URL,\n\tMESSAGE_TIMEOUT,\n} from \"./constants\";\n// Export types\nexport type {\n\tAgentSDKInitConfig,\n\tAgentSDKInstance,\n\tChatPanel,\n\tWebSDKCommonConfig,\n\tWebSDKConfig,\n\tWebSDKConversionConfig,\n\tWebSDKEditorConfig,\n\tWebSDKEvents,\n\tWebSDKSkillInfo,\n} from \"./types/config\";\n\n// Export message types\nexport type {\n\tIframeEventMessage,\n\tIframeMessage,\n\tIframeMessageType,\n\tIframeRequestMessage,\n\tIframeResponseMessage,\n} from \"./types/message\";\n","/**\n * Constants for Aily Web SDK\n */\n\n/** Message business ID for postMessage identification */\nexport const MESSAGE_BIZ_ID = 'cui-sdk-message' as const;\n\n/** Iframe origins by environment */\nexport const IFRAME_ORIGIN: Record<string, string> = {\n production: 'https://aily.feishu.cn',\n staging: 'https://aily.feishu-pre.cn',\n boe: 'https://aily.feishu-boe.cn',\n development: 'http://localhost:8080',\n};\n\n/** Page paths for iframe apps */\nexport const CHAT_IFRAME_PATH = '/cui' as const;\nexport const AGENT_IFRAME_PATH = '/custom_agent/cui' as const;\n\n/** Chat iframe base URLs by environment */\nexport const IFRAME_BASE_URL: Record<string, string> = {\n production: `${IFRAME_ORIGIN.production}${CHAT_IFRAME_PATH}`,\n staging: `${IFRAME_ORIGIN.staging}${CHAT_IFRAME_PATH}`,\n boe: `${IFRAME_ORIGIN.boe}${CHAT_IFRAME_PATH}`,\n development: `${IFRAME_ORIGIN.development}${CHAT_IFRAME_PATH}`,\n};\n\n/** Agent iframe base URLs by environment */\nexport const AGENT_IFRAME_BASE_URL: Record<string, string> = {\n production: `${IFRAME_ORIGIN.production}${AGENT_IFRAME_PATH}`,\n staging: `${IFRAME_ORIGIN.staging}${AGENT_IFRAME_PATH}`,\n boe: `${IFRAME_ORIGIN.boe}${AGENT_IFRAME_PATH}`,\n development: `${IFRAME_ORIGIN.development}${AGENT_IFRAME_PATH}`,\n};\n\n/**\n * Get iframe base URL based on environment\n */\nfunction resolveIframeEnv(env?: string): keyof typeof IFRAME_BASE_URL {\n if (env && env in IFRAME_BASE_URL) {\n return env as keyof typeof IFRAME_BASE_URL;\n }\n\n // Auto-detect based on hostname\n // MiaoDa 宿主域名: miaoda.feishu[-env].cn, *.aiforce[-env][-preview].bytedance.net, *.force-pre.feishuapp.net, *.fsapp.kundou.cn, *.aiforce.run, *.aiforce.cloud\n const hostname =\n typeof window !== 'undefined' ? window.location.hostname : '';\n\n if (hostname.includes('aiforce-boe')) {\n return 'boe';\n }\n if (\n hostname.includes('aiforce-pre') ||\n hostname.includes('force-pre.feishuapp.net') ||\n hostname.includes('fsapp.kundou.cn')\n ) {\n return 'staging';\n }\n if (hostname.includes('localhost') || hostname.includes('127.0.0.1')) {\n return 'development';\n }\n\n return 'production';\n}\n\nexport function getIframeBaseURL(env?: string): string {\n return IFRAME_BASE_URL[resolveIframeEnv(env)];\n}\n\n/**\n * Get agent iframe base URL based on environment\n */\nexport function getAgentIframeBaseURL(env?: string): string {\n return AGENT_IFRAME_BASE_URL[resolveIframeEnv(env)];\n}\n\n/** Timeout for postMessage requests (30 seconds) */\nexport const MESSAGE_TIMEOUT = 30 * 1000;\n\n/** Miaoda 渠道标识默认值(用户可通过 config.common.channelType 覆盖) */\nexport const DEFAULT_CHANNEL_TYPE = 'MIAODA_CUI_SDK';\nexport const DEFAULT_ANONYMOUS_CHANNEL_TYPE = 'MIAODA_ANONYMOUS_CUI_SDK';\n\n/** Allowed origins for postMessage (cui-iframe 的 origin) */\nexport const ALLOWED_ORIGINS = [\n IFRAME_ORIGIN.production,\n IFRAME_ORIGIN.staging,\n IFRAME_ORIGIN.boe,\n 'http://localhost:3000',\n 'http://localhost:5173',\n IFRAME_ORIGIN.development,\n];\n","import {\n\tDEFAULT_ANONYMOUS_CHANNEL_TYPE,\n\tDEFAULT_CHANNEL_TYPE,\n\tgetIframeBaseURL,\n\tMESSAGE_BIZ_ID,\n} from \"./constants\";\nimport { slardar } from \"@lark-apaas/internal-slardar\";\nimport { DeviceDetector } from \"./device-detector\";\nimport { MessageBridge } from \"./message-bridge\";\n\nimport type { ChatPanel, WebSDKConfig } from \"./types/config\";\n\n/**\n * IframeManager handles iframe creation and lifecycle\n *\n * iframe 采用自初始化模式:配置通过 URL hash 传入,iframe 加载后自行解析并渲染,\n * 与旧版 copilot iframe 的行为保持一致。宿主通过 postMessage 进行后续操作。\n */\nexport class IframeManager {\n\tprivate iframe: HTMLIFrameElement | null = null;\n\tprivate messageBridge: MessageBridge;\n\tprivate config: WebSDKConfig;\n\tprivate channelType: string;\n\tprivate root: HTMLElement;\n\tprivate deviceDetector: DeviceDetector | null = null;\n\n\tconstructor(root: HTMLElement, config: WebSDKConfig) {\n\t\tthis.root = root;\n\t\tthis.config = config;\n\t\tconst defaultChannel = config.anonymous\n\t\t\t? DEFAULT_ANONYMOUS_CHANNEL_TYPE\n\t\t\t: DEFAULT_CHANNEL_TYPE;\n\t\tthis.channelType = config.common?.channelType || defaultChannel;\n\t\tthis.messageBridge = new MessageBridge(getIframeBaseURL());\n\t}\n\n\t/**\n\t * Initialize the iframe and establish communication\n\t * 配置通过 URL hash 编码传入 iframe,iframe 加载后自行初始化\n\t */\n\tasync init(): Promise<ChatPanel> {\n\t\tconst initStart = performance.now();\n\n\t\t// Auto-detect device if not explicitly provided\n\t\tif (!this.config.device) {\n\t\t\tthis.deviceDetector = new DeviceDetector();\n\t\t\tthis.config.device = this.deviceDetector.getDevice();\n\t\t}\n\n\t\t// Create iframe element\n\t\tthis.iframe = document.createElement(\"iframe\");\n\t\tthis.iframe.src = this.buildIframeURL();\n\t\tthis.iframe.style.width = \"100%\";\n\t\tthis.iframe.style.height = \"100%\";\n\t\tthis.iframe.style.border = \"none\";\n\t\tthis.iframe.allow = \"microphone; clipboard-write\";\n\n\t\t// Attach to DOM\n\t\tthis.root.appendChild(this.iframe);\n\n\t\t// Attach message bridge (开始监听消息)\n\t\tthis.messageBridge.attach(this.iframe);\n\n\t\t// Setup event handlers (在 iframe 初始化完成之前就开始监听,确保不丢失事件)\n\t\tthis.setupEventHandlers();\n\n\t\t// Listen for device changes and push updates to iframe\n\t\tif (this.deviceDetector) {\n\t\t\tthis.deviceDetector.observe((newDevice) => {\n\t\t\t\tthis.messageBridge.send(\"updateConfig\", { device: newDevice });\n\t\t\t});\n\t\t}\n\n\t\t// Wait for iframe to finish initialization (iframe 自行读取 URL 配置并渲染,完成后发送 ready 事件)\n\t\tawait this.waitForReady();\n\n\t\t// 上报 cui_init_total 耗时(iframe 创建 → init 完成)\n\t\tthis.reportInitMetrics(initStart);\n\n\t\t// Return ChatPanel interface\n\t\treturn this.createChatPanel();\n\t}\n\n\t/**\n\t * Destroy the iframe and cleanup\n\t */\n\tasync destroy(): Promise<void> {\n\t\ttry {\n\t\t\tawait this.messageBridge.send(\"destroy\");\n\t\t} catch {\n\t\t\t// Ignore errors during destroy\n\t\t}\n\n\t\tthis.messageBridge.detach();\n\n\t\tif (this.deviceDetector) {\n\t\t\tthis.deviceDetector.destroy();\n\t\t\tthis.deviceDetector = null;\n\t\t}\n\n\t\tif (this.iframe?.parentNode) {\n\t\t\tthis.iframe.parentNode.removeChild(this.iframe);\n\t\t}\n\n\t\tthis.iframe = null;\n\t}\n\n\t/**\n\t * 构建 iframe URL,将配置编码到 hash 中\n\t * 格式: {baseURL}#{params} 其中 config 为 base64(JSON.stringify(configWithoutEvents))\n\t *\n\t * 鉴权说明:SDK 不参与 token 获取/传递。iframe 自行处理飞书登录流程,\n\t * 允许用户在 iframe 中独立登录。匿名渠道跳过登录检查。\n\t */\n\tprivate buildIframeURL(): string {\n\t\tconst baseURL = getIframeBaseURL();\n\n\t\t// 移除 events(函数无法序列化)\n\t\tconst {\n\t\t\tevents: _events,\n\t\t\tanonymous: _anonymous,\n\t\t\t...configWithoutEvents\n\t\t} = this.config;\n\t\t// 注入 channelType/resourceType(SDK 内部控制,不暴露给消费者)\n\t\t// 补全 editor.skill 硬编码字段\n\t\tconst editorConfig = configWithoutEvents.editor?.skill\n\t\t\t? {\n\t\t\t\t\t...configWithoutEvents.editor,\n\t\t\t\t\tskill: {\n\t\t\t\t\t\ttype: \"custom\",\n\t\t\t\t\t\tbuiltinType: \"unspecified\",\n\t\t\t\t\t\t...configWithoutEvents.editor.skill,\n\t\t\t\t\t},\n\t\t\t\t}\n\t\t\t: configWithoutEvents.editor;\n\n\t\tconst iframeConfig = {\n\t\t\t...configWithoutEvents,\n\t\t\tcommon: {\n\t\t\t\t...configWithoutEvents.common,\n\t\t\t\tchannelType: this.channelType,\n\t\t\t\tresourceType:\n\t\t\t\t\tconfigWithoutEvents.common?.resourceType || this.channelType,\n\t\t\t},\n\t\t\teditor: editorConfig,\n\t\t};\n\t\tconst configBase64 = btoa(encodeURIComponent(JSON.stringify(iframeConfig)));\n\n\t\tconst params = new URLSearchParams({\n\t\t\tappKey: this.config.appKey,\n\t\t\tconfig: encodeURIComponent(configBase64),\n\t\t});\n\n\t\treturn `${baseURL}#${params.toString()}`;\n\t}\n\n\t/**\n\t * 等待 iframe 自行初始化完成后发送 ready 事件\n\t */\n\tprivate waitForReady(): Promise<void> {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tconst timeout = setTimeout(() => {\n\t\t\t\tconst error = new Error(\"Iframe ready timeout\");\n\t\t\t\tslardar.captureException(error, { source: 'aily-web-sdk', module: 'iframe-ready' });\n\t\t\t\treject(error);\n\t\t\t}, 30000);\n\n\t\t\tconst checkMessage = (event: MessageEvent) => {\n\t\t\t\tif (\n\t\t\t\t\tevent.data?.messageBizId === MESSAGE_BIZ_ID &&\n\t\t\t\t\tevent.data?.type === \"event\" &&\n\t\t\t\t\tevent.data?.eventName === \"ready\"\n\t\t\t\t) {\n\t\t\t\t\tclearTimeout(timeout);\n\t\t\t\t\twindow.removeEventListener(\"message\", checkMessage);\n\t\t\t\t\tresolve();\n\t\t\t\t}\n\t\t\t};\n\n\t\t\twindow.addEventListener(\"message\", checkMessage);\n\t\t});\n\t}\n\n\tprivate setupEventHandlers(): void {\n\t\tconst { events } = this.config;\n\t\tif (!events) return;\n\n\t\tif (events.onReady) {\n\t\t\tthis.messageBridge.onEvent(\"ready\", events.onReady);\n\t\t}\n\n\t\tif (events.onError) {\n\t\t\tthis.messageBridge.onEvent(\"error\", (data: unknown) => {\n\t\t\t\tconst msg = (data as { message?: string })?.message || \"Unknown error\";\n\t\t\t\tevents.onError?.(new Error(msg));\n\t\t\t});\n\t\t}\n\n\t\tif (events.onMessage) {\n\t\t\tthis.messageBridge.onEvent(\"onMessage\", events.onMessage);\n\t\t}\n\n\t\tif (events.onInited) {\n\t\t\tthis.messageBridge.onEvent(\n\t\t\t\t\"inited\",\n\t\t\t\tevents.onInited as (data: unknown) => void,\n\t\t\t);\n\t\t}\n\n\t\tif (events.onInitedWithWelcome) {\n\t\t\tthis.messageBridge.onEvent(\n\t\t\t\t\"initedWithWelcome\",\n\t\t\t\tevents.onInitedWithWelcome,\n\t\t\t);\n\t\t}\n\n\t\tif (events.onClose) {\n\t\t\tthis.messageBridge.onEvent(\"onClose\", events.onClose);\n\t\t}\n\t}\n\n\tprivate reportInitMetrics(initStart: number): void {\n\t\ttry {\n\t\t\tconst cost = performance.now() - initStart;\n\t\t\tslardar.sendEvent({\n\t\t\t\tname: \"cui_init_total\",\n\t\t\t\tmetrics: { cost },\n\t\t\t\tcategories: {\n\t\t\t\t\tarch: \"iframe-static\",\n\t\t\t\t\tappKey: this.config.appKey,\n\t\t\t\t\tchannelType: this.channelType,\n\t\t\t\t},\n\t\t\t});\n\t\t} catch {\n\t\t\t// 性能上报失败不影响功能\n\t\t}\n\t}\n\n\tprivate createChatPanel(): ChatPanel {\n\t\treturn {\n\t\t\tsendMessage: async (data) => {\n\t\t\t\tawait this.messageBridge.send(\"sendMessage\", data);\n\t\t\t},\n\t\t\tclear: async () => {\n\t\t\t\tawait this.messageBridge.send(\"clear\");\n\t\t\t},\n\t\t\tcancelMessage: async (messageItem) => {\n\t\t\t\tawait this.messageBridge.send(\"cancelMessage\", messageItem);\n\t\t\t},\n\t\t\tclearAndStop: async () => {\n\t\t\t\tawait this.messageBridge.send(\"clearAndStop\");\n\t\t\t},\n\t\t\tupdateWelcomeMessage: async () => {\n\t\t\t\tawait this.messageBridge.send(\"updateWelcomeMessage\");\n\t\t\t},\n\t\t\tupdateConfig: async (config) => {\n\t\t\t\tawait this.messageBridge.send(\"updateConfig\", config);\n\t\t\t},\n\t\t\tobserveSkill: async (skillId, name, skillType) => {\n\t\t\t\tawait this.messageBridge.send(\"observeSkill\", {\n\t\t\t\t\tskillId,\n\t\t\t\t\tname,\n\t\t\t\t\tskillType,\n\t\t\t\t});\n\t\t\t},\n\t\t\tsetCurrentSkill: async (skill) => {\n\t\t\t\tawait this.messageBridge.send(\"setCurrentSkill\", skill);\n\t\t\t},\n\t\t\tdestroy: async () => {\n\t\t\t\tawait this.destroy();\n\t\t\t},\n\t\t};\n\t}\n}\n","const MOBILE_BREAKPOINT = 768;\n\ntype DeviceType = 'pc' | 'mobile';\ntype DeviceChangeCallback = (device: DeviceType) => void;\n\n/**\n * 设备类型自动检测器\n *\n * 使用 matchMedia + 768px 断点检测设备类型,与 miaoda useIsMobile 逻辑一致。\n * 检测在宿主页面(而非 iframe)中执行,结果通过 postMessage 推送给 iframe。\n */\nexport class DeviceDetector {\n private mediaQuery: MediaQueryList;\n private currentDevice: DeviceType;\n private callback: DeviceChangeCallback | null = null;\n private handleChange: (() => void) | null = null;\n\n constructor() {\n this.mediaQuery = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`);\n this.currentDevice = window.innerWidth < MOBILE_BREAKPOINT ? 'mobile' : 'pc';\n }\n\n getDevice(): DeviceType {\n return this.currentDevice;\n }\n\n observe(callback: DeviceChangeCallback): void {\n this.callback = callback;\n this.handleChange = () => {\n const newDevice: DeviceType = window.innerWidth < MOBILE_BREAKPOINT ? 'mobile' : 'pc';\n if (newDevice !== this.currentDevice) {\n this.currentDevice = newDevice;\n this.callback?.(newDevice);\n }\n };\n this.mediaQuery.addEventListener('change', this.handleChange);\n }\n\n destroy(): void {\n if (this.handleChange) {\n this.mediaQuery.removeEventListener('change', this.handleChange);\n this.handleChange = null;\n }\n this.callback = null;\n }\n}\n","import { MESSAGE_TIMEOUT, ALLOWED_ORIGINS } from './constants';\nimport {\n MESSAGE_BIZ_ID,\n type IframeMessage,\n type IframeRequestMessage,\n type IframeResponseMessage,\n type IframeEventMessage,\n type IframeMessageType,\n} from './types/message';\nimport { slardar } from '@lark-apaas/internal-slardar';\n\ninterface PendingRequest {\n resolve: (value: unknown) => void;\n reject: (reason: Error) => void;\n timer: ReturnType<typeof setTimeout>;\n}\n\n/**\n * MessageBridge handles postMessage communication between parent and iframe\n */\nexport class MessageBridge {\n private iframe: HTMLIFrameElement | null = null;\n private requestId = 0;\n private pendingRequests = new Map<number, PendingRequest>();\n private eventHandlers = new Map<string, ((data: unknown) => void)[]>();\n private targetOrigin: string;\n\n constructor(targetOrigin: string) {\n this.targetOrigin = new URL(targetOrigin).origin;\n this.handleMessage = this.handleMessage.bind(this);\n }\n\n /**\n * Attach to an iframe and start listening for messages\n */\n attach(iframe: HTMLIFrameElement): void {\n this.iframe = iframe;\n window.addEventListener('message', this.handleMessage);\n }\n\n /**\n * Detach from iframe and clean up\n */\n detach(): void {\n window.removeEventListener('message', this.handleMessage);\n this.iframe = null;\n // Clear all pending requests\n this.pendingRequests.forEach(({ reject }) => {\n reject(new Error('MessageBridge detached'));\n });\n this.pendingRequests.clear();\n }\n\n /**\n * Send a message to the iframe and wait for response\n */\n send<T = unknown>(type: IframeMessageType, data?: unknown): Promise<T> {\n return new Promise((resolve, reject) => {\n if (!this.iframe?.contentWindow) {\n reject(new Error('Iframe not attached'));\n return;\n }\n\n this.requestId += 1;\n const id = this.requestId;\n const message: IframeRequestMessage = {\n messageBizId: MESSAGE_BIZ_ID,\n id,\n type,\n data,\n timestamp: Date.now(),\n isRequest: true,\n };\n\n const timer = setTimeout(() => {\n this.pendingRequests.delete(id);\n const error = new Error(`Message timeout: ${type}`);\n slardar.captureException(error, { source: 'aily-web-sdk', module: 'message-bridge', messageType: type });\n reject(error);\n }, MESSAGE_TIMEOUT);\n\n this.pendingRequests.set(id, { resolve: resolve as (value: unknown) => void, reject, timer });\n\n this.iframe.contentWindow.postMessage(message, this.targetOrigin);\n });\n }\n\n /**\n * Register an event handler\n */\n onEvent(eventName: string, handler: (data: unknown) => void): void {\n const handlers = this.eventHandlers.get(eventName) || [];\n handlers.push(handler);\n this.eventHandlers.set(eventName, handlers);\n }\n\n /**\n * Remove an event handler\n */\n offEvent(eventName: string, handler: (data: unknown) => void): void {\n const handlers = this.eventHandlers.get(eventName) || [];\n const index = handlers.indexOf(handler);\n if (index > -1) {\n handlers.splice(index, 1);\n this.eventHandlers.set(eventName, handlers);\n }\n }\n\n private handleMessage(event: MessageEvent<IframeMessage>): void {\n // Validate origin\n if (!ALLOWED_ORIGINS.includes(event.origin)) {\n return;\n }\n\n const { data } = event;\n\n // Validate message format\n if (data?.messageBizId !== MESSAGE_BIZ_ID) {\n return;\n }\n\n // Handle response messages\n if (this.isResponseMessage(data)) {\n const pending = this.pendingRequests.get(data.id);\n if (pending) {\n clearTimeout(pending.timer);\n this.pendingRequests.delete(data.id);\n\n if (data.error) {\n const error = new Error(data.error.message);\n slardar.captureException(error, { source: 'aily-web-sdk', module: 'message-bridge', errorCode: data.error.code });\n pending.reject(error);\n } else {\n pending.resolve(data.data);\n }\n }\n }\n\n // Handle event messages\n if (this.isEventMessage(data)) {\n const handlers = this.eventHandlers.get(data.eventName) || [];\n handlers.forEach((handler) => {\n try {\n handler(data.data);\n } catch {\n // Ignore handler errors\n }\n });\n }\n }\n\n private isResponseMessage(data: IframeMessage): data is IframeResponseMessage {\n return 'isResponse' in data && data.isResponse === true;\n }\n\n private isEventMessage(data: IframeMessage): data is IframeEventMessage {\n return data.type === 'event';\n }\n}\n","import { MESSAGE_BIZ_ID, getAgentIframeBaseURL } from './constants';\nimport { MessageBridge } from './message-bridge';\n\nimport type { AgentSDKInitConfig, AgentSDKInstance } from './types/config';\n\n/**\n * AgentIframeManager handles agent iframe creation and lifecycle.\n */\nexport class AgentIframeManager {\n private iframe: HTMLIFrameElement | null = null;\n private readonly messageBridge: MessageBridge;\n private readonly config: AgentSDKInitConfig;\n private readonly root: HTMLElement;\n\n constructor(root: HTMLElement, config: AgentSDKInitConfig) {\n this.root = root;\n this.config = config;\n this.messageBridge = new MessageBridge(getAgentIframeBaseURL());\n }\n\n async init(): Promise<AgentSDKInstance> {\n this.iframe = document.createElement('iframe');\n this.iframe.src = this.buildIframeURL();\n this.iframe.style.width = '100%';\n this.iframe.style.height = '100%';\n this.iframe.style.border = 'none';\n this.iframe.allow = 'microphone; clipboard-write';\n\n this.root.appendChild(this.iframe);\n this.messageBridge.attach(this.iframe);\n\n await this.waitForReady();\n\n return this.createAgentInstance();\n }\n\n async destroy(): Promise<void> {\n try {\n await this.messageBridge.send('destroy');\n } catch {\n // Ignore errors during destroy.\n }\n\n this.messageBridge.detach();\n\n if (this.iframe?.parentNode) {\n this.iframe.parentNode.removeChild(this.iframe);\n }\n\n this.iframe = null;\n }\n\n private buildIframeURL(): string {\n const baseURL = getAgentIframeBaseURL();\n const params = new URLSearchParams({\n channelToken: this.config.channelToken,\n });\n\n return `${baseURL}?${params.toString()}`;\n }\n\n private waitForReady(): Promise<void> {\n return new Promise((resolve, reject) => {\n const timeout = setTimeout(() => {\n window.removeEventListener('message', checkMessage);\n reject(new Error('Iframe ready timeout'));\n }, 30000);\n\n const checkMessage = (event: MessageEvent) => {\n if (\n event.data?.messageBizId === MESSAGE_BIZ_ID &&\n event.data?.type === 'event' &&\n event.data?.eventName === 'ready'\n ) {\n clearTimeout(timeout);\n window.removeEventListener('message', checkMessage);\n resolve();\n }\n };\n\n window.addEventListener('message', checkMessage);\n });\n }\n\n private createAgentInstance(): AgentSDKInstance {\n return {\n destroy: async () => {\n await this.destroy();\n },\n };\n }\n}\n"],"mappings":"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","names":["MESSAGE_BIZ_ID","IFRAME_ORIGIN","production","staging","boe","development","CHAT_IFRAME_PATH","AGENT_IFRAME_PATH","IFRAME_BASE_URL","AGENT_IFRAME_BASE_URL","resolveIframeEnv","env","hostname","window","location","includes","getIframeBaseURL","getAgentIframeBaseURL","MESSAGE_TIMEOUT","DEFAULT_CHANNEL_TYPE","DEFAULT_ANONYMOUS_CHANNEL_TYPE","ALLOWED_ORIGINS","import_internal_slardar","MOBILE_BREAKPOINT","DeviceDetector","mediaQuery","currentDevice","callback","handleChange","window","matchMedia","innerWidth","getDevice","observe","newDevice","addEventListener","destroy","removeEventListener","MessageBridge","targetOrigin","iframe","requestId","pendingRequests","Map","eventHandlers","URL","origin","handleMessage","bind","attach","window","addEventListener","detach","removeEventListener","forEach","reject","Error","clear","send","type","data","Promise","resolve","contentWindow","id","message","messageBizId","MESSAGE_BIZ_ID","timestamp","Date","now","isRequest","timer","setTimeout","delete","error","slardar","captureException","source","module","messageType","MESSAGE_TIMEOUT","set","postMessage","onEvent","eventName","handler","handlers","get","push","offEvent","index","indexOf","splice","event","ALLOWED_ORIGINS","includes","isResponseMessage","pending","clearTimeout","errorCode","code","isEventMessage","isResponse","IframeManager","root","config","iframe","messageBridge","channelType","deviceDetector","defaultChannel","anonymous","DEFAULT_ANONYMOUS_CHANNEL_TYPE","DEFAULT_CHANNEL_TYPE","common","MessageBridge","getIframeBaseURL","init","initStart","performance","now","device","DeviceDetector","getDevice","document","createElement","src","buildIframeURL","style","width","height","border","allow","appendChild","attach","setupEventHandlers","observe","newDevice","send","waitForReady","reportInitMetrics","createChatPanel","destroy","detach","parentNode","removeChild","baseURL","events","_events","_anonymous","configWithoutEvents","editorConfig","editor","skill","type","builtinType","iframeConfig","resourceType","configBase64","btoa","encodeURIComponent","JSON","stringify","params","URLSearchParams","appKey","toString","Promise","resolve","reject","timeout","setTimeout","error","Error","slardar","captureException","source","module","checkMessage","event","data","messageBizId","MESSAGE_BIZ_ID","eventName","clearTimeout","window","removeEventListener","addEventListener","onReady","onEvent","onError","msg","message","onMessage","onInited","onInitedWithWelcome","onClose","cost","sendEvent","name","metrics","categories","arch","sendMessage","clear","cancelMessage","messageItem","clearAndStop","updateWelcomeMessage","updateConfig","observeSkill","skillId","skillType","setCurrentSkill","AgentIframeManager","root","config","iframe","messageBridge","MessageBridge","getAgentIframeBaseURL","init","document","createElement","src","buildIframeURL","style","width","height","border","allow","appendChild","attach","waitForReady","createAgentInstance","destroy","send","detach","parentNode","removeChild","baseURL","params","URLSearchParams","channelToken","toString","Promise","resolve","reject","timeout","setTimeout","window","removeEventListener","checkMessage","Error","event","data","messageBizId","MESSAGE_BIZ_ID","type","eventName","clearTimeout","addEventListener","initAilyChat","root","config","manager","IframeManager","init","initAgentChat","AgentIframeManager"]}