vite-plugin-opencode-assistant 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/client.js ADDED
@@ -0,0 +1,1549 @@
1
+ "use strict";
2
+ /**
3
+ * @fileoverview OpenCode 挂件客户端脚本
4
+ * @description 用于在浏览器中显示 OpenCode AI 助手挂件
5
+ */
6
+ (function () {
7
+ 'use strict';
8
+ /** @type {string} 初始化标记 */
9
+ const INIT_MARKER = '__OPENCODE_INITIALIZED__';
10
+ /** @type {string} 选中元素存储键 */
11
+ const SELECTED_ELEMENTS_KEY = '__opencode_selected_elements__';
12
+ /** @type {number} 服务器同步间隔(毫秒) */
13
+ const SERVER_SYNC_INTERVAL = 2000;
14
+ /** @type {number} 检查 Vue Inspector 间隔(毫秒) */
15
+ const INSPECTOR_CHECK_INTERVAL = 500;
16
+ /** @type {number} 自动打开延迟(毫秒) */
17
+ const AUTO_OPEN_DELAY = 1000;
18
+ /** @type {number} 通知显示时间(毫秒) */
19
+ const NOTIFICATION_DURATION = 3000;
20
+ /**
21
+ * @typedef {Object} HotkeyConfig
22
+ * @property {boolean} ctrl - 是否需要 Ctrl/Meta 键
23
+ * @property {boolean} shift - 是否需要 Shift 键
24
+ * @property {boolean} alt - 是否需要 Alt 键
25
+ * @property {string} key - 主键
26
+ */
27
+ /**
28
+ * @typedef {Object} SelectedElement
29
+ * @property {string|null} filePath - 文件路径
30
+ * @property {number|null} line - 行号
31
+ * @property {number|null} column - 列号
32
+ * @property {string} innerText - 元素内部文本
33
+ * @property {string} description - 元素描述(标签名+选择器)
34
+ */
35
+ /**
36
+ * @typedef {Object} WidgetConfig
37
+ * @property {string} webUrl - Web 服务 URL
38
+ * @property {string} position - 挂件位置
39
+ * @property {string} theme - 主题模式
40
+ * @property {boolean} open - 是否自动打开
41
+ * @property {string} sessionUrl - 会话 URL
42
+ * @property {boolean} lazy - 是否懒加载
43
+ * @property {string} hotkey - 快捷键配置
44
+ */
45
+ /**
46
+ * 初始化 OpenCode 挂件
47
+ * @param {WidgetConfig} config - 挂件配置
48
+ */
49
+ function initOpenCodeWidget(config) {
50
+ if (window[INIT_MARKER])
51
+ return;
52
+ window[INIT_MARKER] = true;
53
+ const { webUrl, position, theme, open, sessionUrl, lazy, hotkey, cwd } = config;
54
+ /** @type {string} 当前页面 URL */
55
+ let currentPageUrl = '';
56
+ /** @type {string} 当前页面标题 */
57
+ let currentPageTitle = '';
58
+ /** @type {boolean} 服务是否已启动 */
59
+ let servicesStarted = !lazy;
60
+ /** @type {boolean} 挂件是否打开 */
61
+ let isOpen = false;
62
+ /**
63
+ * 解析快捷键字符串
64
+ * @param {string} hotkeyStr - 快捷键字符串,如 'ctrl+k'
65
+ * @returns {HotkeyConfig} 快捷键配置
66
+ */
67
+ function parseHotkey(hotkeyStr) {
68
+ if (!hotkeyStr)
69
+ return { ctrl: true, shift: false, alt: false, key: 'k' };
70
+ const parts = hotkeyStr.toLowerCase().split('+');
71
+ const key = parts.pop();
72
+ return {
73
+ ctrl: parts.includes('ctrl') || parts.includes('cmd') || parts.includes('meta'),
74
+ shift: parts.includes('shift'),
75
+ alt: parts.includes('alt'),
76
+ key: key || 'k'
77
+ };
78
+ }
79
+ /** @type {HotkeyConfig} 主快捷键配置 */
80
+ const mainHotkey = parseHotkey(hotkey);
81
+ /**
82
+ * 检查键盘事件是否匹配快捷键
83
+ * @param {KeyboardEvent} e - 键盘事件
84
+ * @param {HotkeyConfig} hotkeyConfig - 快捷键配置
85
+ * @returns {boolean} 是否匹配
86
+ */
87
+ function matchHotkey(e, hotkeyConfig) {
88
+ const ctrlMatch = hotkeyConfig.ctrl ? (e.ctrlKey || e.metaKey) : !(e.ctrlKey || e.metaKey);
89
+ const shiftMatch = hotkeyConfig.shift ? e.shiftKey : !e.shiftKey;
90
+ const altMatch = hotkeyConfig.alt ? e.altKey : !e.altKey;
91
+ const keyMatch = e.key.toLowerCase() === hotkeyConfig.key.toLowerCase();
92
+ return ctrlMatch && shiftMatch && altMatch && keyMatch;
93
+ }
94
+ /**
95
+ * 从 sessionStorage 加载选中的元素
96
+ * @returns {SelectedElement[]} 选中的元素列表
97
+ */
98
+ function loadSelectedElements() {
99
+ try {
100
+ const stored = sessionStorage.getItem(SELECTED_ELEMENTS_KEY);
101
+ if (stored) {
102
+ return JSON.parse(stored);
103
+ }
104
+ }
105
+ catch (e) {
106
+ // 忽略错误
107
+ }
108
+ return [];
109
+ }
110
+ /**
111
+ * 保存选中的元素到 sessionStorage
112
+ * @param {SelectedElement[]} elements - 选中的元素列表
113
+ */
114
+ function saveSelectedElements(elements) {
115
+ try {
116
+ sessionStorage.setItem(SELECTED_ELEMENTS_KEY, JSON.stringify(elements));
117
+ }
118
+ catch (e) {
119
+ // 忽略错误
120
+ }
121
+ }
122
+ /** @type {SelectedElement[]} 选中的元素列表 */
123
+ let selectedElements = loadSelectedElements();
124
+ /**
125
+ * 确保服务已启动
126
+ * @returns {Promise<boolean>} 是否成功启动
127
+ */
128
+ async function ensureServicesStarted() {
129
+ if (servicesStarted)
130
+ return true;
131
+ try {
132
+ const res = await fetch('/__opencode_start__');
133
+ const data = await res.json();
134
+ if (data.success) {
135
+ servicesStarted = true;
136
+ if (data.sessionUrl && iframe) {
137
+ iframe.src = data.sessionUrl;
138
+ }
139
+ return true;
140
+ }
141
+ }
142
+ catch (e) {
143
+ console.error('[OpenCode Widget] Failed to start services:', e);
144
+ }
145
+ return false;
146
+ }
147
+ /**
148
+ * 更新页面上下文
149
+ * @param {boolean} [force=false] - 是否强制更新
150
+ */
151
+ function updateContext(force = false) {
152
+ if (!servicesStarted)
153
+ return;
154
+ const newUrl = window.location.href;
155
+ const newTitle = document.title;
156
+ if (force || newUrl !== currentPageUrl || newTitle !== currentPageTitle) {
157
+ currentPageUrl = newUrl;
158
+ currentPageTitle = newTitle;
159
+ fetch('/__opencode_context__', {
160
+ method: 'POST',
161
+ headers: { 'Content-Type': 'application/json' },
162
+ body: JSON.stringify({ url: newUrl, title: newTitle, selectedElements })
163
+ }).catch(() => { });
164
+ }
165
+ }
166
+ // 监听路由变化
167
+ const originalPushState = history.pushState;
168
+ const originalReplaceState = history.replaceState;
169
+ history.pushState = function (...args) {
170
+ originalPushState.apply(this, args);
171
+ setTimeout(updateContext, 0);
172
+ };
173
+ history.replaceState = function (...args) {
174
+ originalReplaceState.apply(this, args);
175
+ setTimeout(updateContext, 0);
176
+ };
177
+ window.addEventListener('popstate', () => setTimeout(updateContext, 0));
178
+ window.addEventListener('hashchange', () => setTimeout(updateContext, 0));
179
+ // 监听标题变化
180
+ const titleObserver = new MutationObserver(() => {
181
+ if (document.title !== currentPageTitle) {
182
+ updateContext();
183
+ }
184
+ });
185
+ if (document.head) {
186
+ titleObserver.observe(document.head, { childList: true, subtree: true });
187
+ }
188
+ if (servicesStarted) {
189
+ updateContext(true);
190
+ }
191
+ /** @type {string} iframe URL */
192
+ const iframeUrl = sessionUrl || webUrl;
193
+ // 创建样式
194
+ const style = document.createElement('style');
195
+ style.textContent = buildWidgetStyles();
196
+ document.head.appendChild(style);
197
+ // 创建容器
198
+ const container = document.createElement('div');
199
+ container.className = `opencode-widget ${position}`;
200
+ // 创建按钮
201
+ const button = document.createElement('button');
202
+ button.className = 'opencode-button';
203
+ button.innerHTML = `
204
+ <svg t="1775402599580" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns=" http://www.w3.org/2000/svg " p-id="5390" xmlns:xlink=" http://www.w3.org/1999/xlink " width="200" height="200"><path d="M512 981.33H85.34c-15.85 0-30.38-8.77-37.77-22.81a42.624 42.624 0 0 1 2.6-44.02L135 791.08C75.25 710.5 42.67 612.6 42.67 512 42.67 253.21 253.21 42.67 512 42.67S981.34 253.21 981.34 512 770.8 981.33 512 981.33zM166.44 896H512c211.73 0 384-172.27 384-384S723.73 128 512 128 128 300.27 128 512c0 91.29 32.83 179.9 92.46 249.46 12.58 14.69 13.73 36 2.77 51.94L166.44 896z" fill="white" p-id="5391"></path><path d="M384 448m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z" fill="white" p-id="5392"></path><path d="M640 448m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z" fill="white" p-id="5393"></path></svg>
205
+ `;
206
+ button.setAttribute('aria-label', 'Open AI Assistant');
207
+ // 创建聊天面板
208
+ const chat = document.createElement('div');
209
+ chat.className = 'opencode-chat';
210
+ // 创建会话列表
211
+ const sessionList = document.createElement('div');
212
+ sessionList.className = 'opencode-session-list collapsed';
213
+ // 创建折叠按钮
214
+ const toggleBtn = document.createElement('button');
215
+ toggleBtn.className = 'opencode-session-toggle';
216
+ toggleBtn.innerHTML = `
217
+ <svg viewBox="0 0 24 24" width="16" height="16" xmlns="http://www.w3.org/2000/svg">
218
+ <path d="M4 6h16M4 12h16M4 18h16" stroke="currentColor" stroke-width="2" stroke-linecap="round"/>
219
+ </svg>
220
+ `;
221
+ toggleBtn.title = '展开会话列表';
222
+ // 创建会话列表头部
223
+ const sessionListHeader = document.createElement('div');
224
+ sessionListHeader.className = 'opencode-session-list-header';
225
+ sessionListHeader.innerHTML = `
226
+ <span>会话列表</span>
227
+ <button class="opencode-new-session-btn" title="新建会话">+</button>
228
+ `;
229
+ // 创建会话列表内容
230
+ const sessionListContent = document.createElement('div');
231
+ sessionListContent.className = 'opencode-session-list-content';
232
+ sessionList.appendChild(toggleBtn);
233
+ sessionList.appendChild(sessionListHeader);
234
+ sessionList.appendChild(sessionListContent);
235
+ // 折叠/展开会话列表
236
+ let isSessionListCollapsed = true;
237
+ function toggleSessionList() {
238
+ isSessionListCollapsed = !isSessionListCollapsed;
239
+ sessionList.classList.toggle('collapsed', isSessionListCollapsed);
240
+ toggleBtn.title = isSessionListCollapsed ? '展开会话列表' : '折叠会话列表';
241
+ }
242
+ toggleBtn.addEventListener('click', toggleSessionList);
243
+ // 创建 iframe 容器
244
+ const iframeContainer = document.createElement('div');
245
+ iframeContainer.className = 'opencode-iframe-container';
246
+ // 创建加载指示器
247
+ const loadingOverlay = document.createElement('div');
248
+ loadingOverlay.className = 'opencode-loading-overlay';
249
+ loadingOverlay.innerHTML = `
250
+ <div class="opencode-loading-spinner"></div>
251
+ <div class="opencode-loading-text">加载中...</div>
252
+ `;
253
+ // 创建 iframe
254
+ const iframe = document.createElement('iframe');
255
+ iframe.className = 'opencode-iframe';
256
+ iframe.src = servicesStarted ? iframeUrl : 'about:blank';
257
+ iframe.allow = 'clipboard-write; clipboard-read';
258
+ iframe.referrerPolicy = 'origin';
259
+ iframe.onload = function () {
260
+ if (servicesStarted) {
261
+ updateContext();
262
+ loadSessions();
263
+ }
264
+ hideLoading();
265
+ };
266
+ iframeContainer.appendChild(loadingOverlay);
267
+ iframeContainer.appendChild(iframe);
268
+ // 创建右侧工具栏
269
+ const rightToolbar = document.createElement('div');
270
+ rightToolbar.className = `opencode-right-toolbar${selectedElements.length === 0 ? ' collapsed' : ''}`;
271
+ // 创建元素选择按钮
272
+ const selectButton = document.createElement('button');
273
+ selectButton.className = 'opencode-select-btn';
274
+ selectButton.innerHTML = `
275
+ <svg viewBox="0 0 1024 1024" width="18" height="18" xmlns="http://www.w3.org/2000/svg">
276
+ <path fill="currentColor" d="M512 896a384 384 0 1 0 0-768 384 384 0 0 0 0 768m0 64a448 448 0 1 1 0-896 448 448 0 0 1 0 896"></path><path fill="currentColor" d="M512 96a32 32 0 0 1 32 32v192a32 32 0 0 1-64 0V128a32 32 0 0 1 32-32m0 576a32 32 0 0 1 32 32v192a32 32 0 1 1-64 0V704a32 32 0 0 1 32-32M96 512a32 32 0 0 1 32-32h192a32 32 0 0 1 0 64H128a32 32 0 0 1-32-32m576 0a32 32 0 0 1 32-32h192a32 32 0 1 1 0 64H704a32 32 0 0 1-32-32"></path>
277
+ </svg>
278
+ `;
279
+ selectButton.title = '选择页面元素';
280
+ // 创建已选节点容器
281
+ const selectedNodesContainer = document.createElement('div');
282
+ selectedNodesContainer.className = 'opencode-selected-nodes';
283
+ rightToolbar.appendChild(selectButton);
284
+ rightToolbar.appendChild(selectedNodesContainer);
285
+ // 创建选择模式常驻提示(固定到页面顶部)
286
+ const selectModeHint = document.createElement('div');
287
+ selectModeHint.className = 'opencode-select-mode-hint';
288
+ selectModeHint.innerHTML = `
289
+ <span>🎯 选择模式已开启 - 点击元素进行选择</span>
290
+ <span class="opencode-hint-shortcut">按 ESC 退出</span>
291
+ `;
292
+ // 创建已选节点气泡容器(气泡按钮上方)
293
+ const selectedBubbles = document.createElement('div');
294
+ selectedBubbles.className = 'opencode-selected-bubbles';
295
+ chat.appendChild(sessionList);
296
+ chat.appendChild(iframeContainer);
297
+ chat.appendChild(rightToolbar);
298
+ container.appendChild(button);
299
+ container.appendChild(selectedBubbles);
300
+ container.appendChild(chat);
301
+ document.body.appendChild(container);
302
+ document.body.appendChild(selectModeHint);
303
+ if (selectedElements.length > 0) {
304
+ renderSelectedNodes();
305
+ }
306
+ /** @type {Array} 会话列表 */
307
+ let sessions = [];
308
+ /** @type {string|null} 当前会话 ID */
309
+ let currentSessionId = null;
310
+ /**
311
+ * 从 URL 中提取会话 ID
312
+ */
313
+ function extractSessionId(url) {
314
+ if (!url)
315
+ return null;
316
+ const match = url.match(/\/session\/([^/?]+)/);
317
+ return match ? match[1] : null;
318
+ }
319
+ // 从初始 URL 中提取会话 ID
320
+ currentSessionId = extractSessionId(sessionUrl);
321
+ /**
322
+ * 显示加载状态
323
+ */
324
+ function showLoading() {
325
+ loadingOverlay.classList.add('visible');
326
+ }
327
+ /**
328
+ * 隐藏加载状态
329
+ */
330
+ function hideLoading() {
331
+ loadingOverlay.classList.remove('visible');
332
+ }
333
+ /**
334
+ * 加载会话列表
335
+ */
336
+ async function loadSessions() {
337
+ try {
338
+ const response = await fetch('/__opencode_sessions__');
339
+ sessions = await response.json();
340
+ renderSessionList();
341
+ }
342
+ catch (e) {
343
+ console.error('Failed to load sessions:', e);
344
+ }
345
+ }
346
+ /**
347
+ * 渲染会话列表
348
+ */
349
+ function renderSessionList() {
350
+ sessionListContent.innerHTML = '';
351
+ const currentProjectSessions = sessions.filter(session => session.directory === cwd);
352
+ currentProjectSessions.forEach(session => {
353
+ const item = document.createElement('div');
354
+ item.className = 'opencode-session-item';
355
+ if (session.id === currentSessionId) {
356
+ item.classList.add('active');
357
+ }
358
+ const header = document.createElement('div');
359
+ header.className = 'opencode-session-header';
360
+ const title = document.createElement('div');
361
+ title.className = 'opencode-session-title';
362
+ title.textContent = session.title || '新会话';
363
+ const deleteBtn = document.createElement('button');
364
+ deleteBtn.className = 'opencode-session-delete-btn';
365
+ deleteBtn.innerHTML = '×';
366
+ deleteBtn.title = '删除会话';
367
+ deleteBtn.addEventListener('click', (e) => {
368
+ e.stopPropagation();
369
+ confirmDeleteSession(session);
370
+ });
371
+ header.appendChild(title);
372
+ header.appendChild(deleteBtn);
373
+ const meta = document.createElement('div');
374
+ meta.className = 'opencode-session-meta';
375
+ const date = new Date(session.time.updated);
376
+ meta.textContent = date.toLocaleDateString() + ' ' + date.toLocaleTimeString();
377
+ item.appendChild(header);
378
+ item.appendChild(meta);
379
+ item.addEventListener('click', () => {
380
+ switchSession(session);
381
+ });
382
+ sessionListContent.appendChild(item);
383
+ });
384
+ }
385
+ /**
386
+ * 切换会话
387
+ */
388
+ function switchSession(session) {
389
+ if (session.id === currentSessionId) {
390
+ return;
391
+ }
392
+ currentSessionId = session.id;
393
+ const encodedDir = btoa(cwd);
394
+ const baseUrl = iframeUrl.split('/').slice(0, 3).join('/');
395
+ showLoading();
396
+ iframe.src = `${baseUrl}/${encodedDir}/session/${session.id}`;
397
+ renderSessionList();
398
+ }
399
+ /**
400
+ * 创建新会话
401
+ */
402
+ async function createNewSession() {
403
+ try {
404
+ const response = await fetch('/__opencode_sessions__', {
405
+ method: 'POST'
406
+ });
407
+ const newSession = await response.json();
408
+ sessions.unshift(newSession);
409
+ switchSession(newSession);
410
+ }
411
+ catch (e) {
412
+ console.error('Failed to create session:', e);
413
+ showNotification('创建会话失败');
414
+ }
415
+ }
416
+ /**
417
+ * 确认删除会话
418
+ */
419
+ function confirmDeleteSession(session) {
420
+ const confirmed = confirm(`确定要删除会话 "${session.title || '新会话'}" 吗?`);
421
+ if (confirmed) {
422
+ deleteSession(session);
423
+ }
424
+ }
425
+ /**
426
+ * 删除会话
427
+ */
428
+ async function deleteSession(session) {
429
+ try {
430
+ const response = await fetch(`/__opencode_sessions__?id=${session.id}`, {
431
+ method: 'DELETE'
432
+ });
433
+ if (!response.ok) {
434
+ throw new Error('Delete failed');
435
+ }
436
+ sessions = sessions.filter(s => s.id !== session.id);
437
+ if (session.id === currentSessionId) {
438
+ const remainingSessions = sessions.filter(s => s.directory === cwd);
439
+ if (remainingSessions.length > 0) {
440
+ switchSession(remainingSessions[0]);
441
+ }
442
+ else {
443
+ currentSessionId = null;
444
+ iframe.src = 'about:blank';
445
+ }
446
+ }
447
+ renderSessionList();
448
+ showNotification('会话已删除');
449
+ }
450
+ catch (e) {
451
+ console.error('Failed to delete session:', e);
452
+ showNotification('删除会话失败');
453
+ }
454
+ }
455
+ // 绑定新建会话按钮
456
+ sessionListHeader.querySelector('.opencode-new-session-btn').addEventListener('click', createNewSession);
457
+ /**
458
+ * 应用主题
459
+ */
460
+ function applyTheme() {
461
+ if (theme === 'auto') {
462
+ const prefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
463
+ container.classList.toggle('opencode-dark', prefersDark);
464
+ }
465
+ else {
466
+ container.classList.toggle('opencode-dark', theme === 'dark');
467
+ }
468
+ }
469
+ applyTheme();
470
+ if (theme === 'auto') {
471
+ window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', applyTheme);
472
+ }
473
+ /**
474
+ * 切换挂件显示状态
475
+ */
476
+ async function toggle() {
477
+ if (lazy && !servicesStarted) {
478
+ button.classList.add('loading');
479
+ const started = await ensureServicesStarted();
480
+ button.classList.remove('loading');
481
+ if (!started) {
482
+ showNotification('服务启动失败,请检查控制台');
483
+ return;
484
+ }
485
+ }
486
+ isOpen = !isOpen;
487
+ chat.classList.toggle('open', isOpen);
488
+ button.classList.toggle('active', isOpen);
489
+ if (isOpen) {
490
+ updateContext();
491
+ }
492
+ }
493
+ button.addEventListener('click', toggle);
494
+ document.addEventListener('keydown', (e) => {
495
+ if (matchHotkey(e, mainHotkey)) {
496
+ e.preventDefault();
497
+ toggle();
498
+ }
499
+ });
500
+ if (open && servicesStarted) {
501
+ setTimeout(() => {
502
+ toggle();
503
+ }, AUTO_OPEN_DELAY);
504
+ }
505
+ /**
506
+ * 显示通知
507
+ * @param {string} message - 通知消息
508
+ */
509
+ function showNotification(message) {
510
+ const notification = document.createElement('div');
511
+ notification.className = 'opencode-notification';
512
+ notification.textContent = message;
513
+ chat.appendChild(notification);
514
+ setTimeout(() => {
515
+ notification.remove();
516
+ }, NOTIFICATION_DURATION);
517
+ }
518
+ /**
519
+ * 添加选中元素
520
+ * @param {SelectedElement} elementInfo - 元素信息
521
+ */
522
+ function addElement(elementInfo) {
523
+ const key = elementInfo.filePath && elementInfo.line
524
+ ? `${elementInfo.filePath}:${elementInfo.line}`
525
+ : null;
526
+ const exists = key && selectedElements.some(el => {
527
+ const elKey = el.filePath && el.line ? `${el.filePath}:${el.line}` : null;
528
+ return elKey === key;
529
+ });
530
+ if (!exists) {
531
+ selectedElements.push(elementInfo);
532
+ saveSelectedElements(selectedElements);
533
+ renderSelectedNodes();
534
+ if (isSelectMode) {
535
+ renderSelectedBubbles();
536
+ }
537
+ showNotification(`已选中元素 (${selectedElements.length}个)`);
538
+ }
539
+ else {
540
+ showNotification('该元素已选中');
541
+ }
542
+ }
543
+ /**
544
+ * 移除选中元素
545
+ * @param {number} index - 元素索引
546
+ */
547
+ function removeElement(index) {
548
+ selectedElements.splice(index, 1);
549
+ saveSelectedElements(selectedElements);
550
+ renderSelectedNodes();
551
+ updateToolbarState();
552
+ sendSelectedElements();
553
+ }
554
+ /**
555
+ * 清除所有选中元素
556
+ */
557
+ function clearAllElements() {
558
+ selectedElements = [];
559
+ saveSelectedElements(selectedElements);
560
+ renderSelectedNodes();
561
+ updateToolbarState();
562
+ sendSelectedElements();
563
+ showNotification('已清除所有选中元素');
564
+ }
565
+ /**
566
+ * 更新工具栏状态
567
+ */
568
+ function updateToolbarState() {
569
+ if (selectedElements.length > 0) {
570
+ rightToolbar.classList.remove('collapsed');
571
+ }
572
+ else {
573
+ rightToolbar.classList.add('collapsed');
574
+ }
575
+ }
576
+ /** @type {boolean} 是否处于元素选择模式 */
577
+ let isSelectMode = false;
578
+ /**
579
+ * 切换元素选择模式
580
+ */
581
+ function toggleSelectMode() {
582
+ const inspector = window.__VUE_INSPECTOR__;
583
+ if (!inspector) {
584
+ showNotification('Vue Inspector 未加载,无法使用元素选择功能');
585
+ return;
586
+ }
587
+ isSelectMode = !isSelectMode;
588
+ selectButton.classList.toggle('active', isSelectMode);
589
+ selectModeHint.classList.toggle('visible', isSelectMode);
590
+ selectedBubbles.classList.toggle('visible', isSelectMode);
591
+ if (isSelectMode) {
592
+ chat.style.display = 'none';
593
+ inspector.enable();
594
+ renderSelectedBubbles();
595
+ }
596
+ else {
597
+ chat.style.display = '';
598
+ inspector.disable();
599
+ }
600
+ }
601
+ /**
602
+ * 退出选择模式
603
+ */
604
+ function exitSelectMode() {
605
+ if (!isSelectMode)
606
+ return;
607
+ const inspector = window.__VUE_INSPECTOR__;
608
+ if (inspector) {
609
+ inspector.disable();
610
+ }
611
+ isSelectMode = false;
612
+ selectButton.classList.remove('active');
613
+ selectModeHint.classList.remove('visible');
614
+ selectedBubbles.classList.remove('visible');
615
+ chat.style.display = '';
616
+ }
617
+ // ESC 键退出选择模式
618
+ document.addEventListener('keydown', (e) => {
619
+ if (e.key === 'Escape' && isSelectMode) {
620
+ exitSelectMode();
621
+ }
622
+ });
623
+ /**
624
+ * 渲染已选节点气泡
625
+ */
626
+ function renderSelectedBubbles() {
627
+ selectedBubbles.innerHTML = '';
628
+ if (selectedElements.length === 0) {
629
+ selectedBubbles.innerHTML = '<div class="opencode-bubble-empty">暂无选中元素</div>';
630
+ return;
631
+ }
632
+ selectedElements.forEach((element, index) => {
633
+ const bubble = document.createElement('div');
634
+ bubble.className = 'opencode-selected-bubble';
635
+ const description = element.description || '未知元素';
636
+ const fileName = element.filePath ? element.filePath.split('/').pop() : '';
637
+ const lineInfo = element.line ? `:${element.line}` : '';
638
+ bubble.innerHTML = `
639
+ <span class="opencode-bubble-text">${description}</span>
640
+ ${fileName ? `<span class="opencode-bubble-file">${fileName}${lineInfo}</span>` : ''}
641
+ <button class="opencode-bubble-remove" data-index="${index}">×</button>
642
+ `;
643
+ bubble.querySelector('.opencode-bubble-remove').addEventListener('click', (e) => {
644
+ e.stopPropagation();
645
+ removeElement(index);
646
+ renderSelectedBubbles();
647
+ });
648
+ selectedBubbles.appendChild(bubble);
649
+ });
650
+ }
651
+ function renderSelectedNodes() {
652
+ selectedNodesContainer.innerHTML = '';
653
+ if (selectedElements.length === 0) {
654
+ rightToolbar.classList.add('collapsed');
655
+ }
656
+ else {
657
+ rightToolbar.classList.remove('collapsed');
658
+ }
659
+ selectedElements.forEach((element, index) => {
660
+ const node = document.createElement('div');
661
+ node.className = 'opencode-selected-node';
662
+ const description = element.description || '未知元素';
663
+ const textPreview = element.innerText ? element.innerText.substring(0, 30) : '';
664
+ const fileName = element.filePath ? element.filePath.split('/').pop() : '未知文件';
665
+ const lineInfo = element.line ? `:${element.line}` : '';
666
+ node.innerHTML = `
667
+ <div class="opencode-node-content">
668
+ <span class="opencode-node-text">${description}</span>
669
+ <span class="opencode-node-file">${textPreview ? textPreview + ' · ' : ''}${fileName}${lineInfo}</span>
670
+ </div>
671
+ <button class="opencode-node-remove" data-index="${index}">×</button>
672
+ `;
673
+ node.querySelector('.opencode-node-remove').addEventListener('click', (e) => {
674
+ e.stopPropagation();
675
+ removeElement(index);
676
+ });
677
+ selectedNodesContainer.appendChild(node);
678
+ });
679
+ }
680
+ // 绑定选择按钮点击事件
681
+ selectButton.addEventListener('click', toggleSelectMode);
682
+ /**
683
+ * 发送选中元素到服务器
684
+ */
685
+ function sendSelectedElements() {
686
+ if (!servicesStarted)
687
+ return;
688
+ fetch('/__opencode_context__', {
689
+ method: 'POST',
690
+ headers: { 'Content-Type': 'application/json' },
691
+ body: JSON.stringify({
692
+ url: currentPageUrl,
693
+ title: currentPageTitle,
694
+ selectedElements: selectedElements
695
+ })
696
+ }).catch(() => { });
697
+ }
698
+ /**
699
+ * 与服务器同步选中元素
700
+ */
701
+ function syncWithServer() {
702
+ if (!servicesStarted)
703
+ return;
704
+ fetch('/__opencode_context__')
705
+ .then(res => res.json())
706
+ .then(data => {
707
+ if (data.selectedElements && data.selectedElements.length === 0 && selectedElements.length > 0) {
708
+ selectedElements = [];
709
+ saveSelectedElements(selectedElements);
710
+ renderSelectedNodes();
711
+ }
712
+ })
713
+ .catch(() => { });
714
+ }
715
+ setInterval(syncWithServer, SERVER_SYNC_INTERVAL);
716
+ /**
717
+ * 截断字符串
718
+ * @param {string} str - 原字符串
719
+ * @param {number} maxLength - 最大长度
720
+ * @returns {string} 截断后的字符串
721
+ */
722
+ function truncate(str, maxLength) {
723
+ if (!str)
724
+ return '';
725
+ return str.length > maxLength ? str.substring(0, maxLength) + '...' : str;
726
+ }
727
+ /**
728
+ * 获取元素的直接文本内容
729
+ * @param {Element} element - DOM 元素
730
+ * @returns {string} 直接文本内容
731
+ */
732
+ function getDirectText(element) {
733
+ let text = '';
734
+ for (const child of element.childNodes) {
735
+ if (child.nodeType === Node.TEXT_NODE) {
736
+ text += child.textContent || '';
737
+ }
738
+ }
739
+ return text.trim();
740
+ }
741
+ /**
742
+ * 获取元素描述信息
743
+ * @param {Element} element - DOM 元素
744
+ * @returns {string} 元素描述
745
+ */
746
+ function getElementDescription(element) {
747
+ const tag = element.tagName.toLowerCase();
748
+ const parts = [tag];
749
+ const id = element.id;
750
+ if (id)
751
+ parts.push(`#${id}`);
752
+ const className = element.className && typeof element.className === 'string'
753
+ ? element.className.trim().split(/\s+/).filter(Boolean).slice(0, 2).join('.')
754
+ : '';
755
+ if (className)
756
+ parts.push(`.${className}`);
757
+ const name = element.getAttribute('name');
758
+ if (name)
759
+ parts.push(`[name="${name}"]`);
760
+ const placeholder = element.getAttribute('placeholder');
761
+ if (placeholder)
762
+ parts.push(`[placeholder="${placeholder.substring(0, 20)}"]`);
763
+ const src = element.getAttribute('src');
764
+ if (src)
765
+ parts.push(`[src]`);
766
+ const href = element.getAttribute('href');
767
+ if (href && href !== '#')
768
+ parts.push(`[href]`);
769
+ return parts.join('');
770
+ }
771
+ /**
772
+ * 设置 Vue Inspector 钩子
773
+ */
774
+ function setupInspectorHook() {
775
+ if (window.__VUE_INSPECTOR__) {
776
+ const inspector = window.__VUE_INSPECTOR__;
777
+ const originalHandleClick = inspector.handleClick.bind(inspector);
778
+ inspector.handleClick = function (e) {
779
+ if (isSelectMode) {
780
+ const { targetNode, params } = inspector.getTargetNode(e);
781
+ if (targetNode && params) {
782
+ const innerText = getDirectText(targetNode);
783
+ const description = getElementDescription(targetNode);
784
+ const elementInfo = {
785
+ filePath: params.file,
786
+ line: params.line,
787
+ column: params.column,
788
+ innerText: truncate(innerText, 200),
789
+ description
790
+ };
791
+ addElement(elementInfo);
792
+ sendSelectedElements();
793
+ }
794
+ return;
795
+ }
796
+ return originalHandleClick.call(inspector, e);
797
+ };
798
+ }
799
+ }
800
+ if (window.__VUE_INSPECTOR__) {
801
+ setupInspectorHook();
802
+ }
803
+ else {
804
+ const checkInspector = setInterval(() => {
805
+ if (window.__VUE_INSPECTOR__) {
806
+ setupInspectorHook();
807
+ clearInterval(checkInspector);
808
+ }
809
+ }, INSPECTOR_CHECK_INTERVAL);
810
+ }
811
+ // 导出全局 API
812
+ window.OpenCodeWidget = {
813
+ open: () => { if (!isOpen)
814
+ toggle(); },
815
+ close: () => { if (isOpen)
816
+ toggle(); },
817
+ toggle,
818
+ showNotification,
819
+ updateContext,
820
+ };
821
+ }
822
+ /**
823
+ * 构建挂件样式
824
+ * @returns {string} CSS 样式字符串
825
+ */
826
+ function buildWidgetStyles() {
827
+ return `
828
+ .opencode-widget {
829
+ position: fixed;
830
+ z-index: 999999;
831
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
832
+ }
833
+
834
+ .opencode-widget.bottom-right {
835
+ bottom: 20px;
836
+ right: 20px;
837
+ }
838
+
839
+ .opencode-widget.bottom-left {
840
+ bottom: 20px;
841
+ left: 20px;
842
+ }
843
+
844
+ .opencode-widget.top-right {
845
+ top: 20px;
846
+ right: 20px;
847
+ }
848
+
849
+ .opencode-widget.top-left {
850
+ top: 20px;
851
+ left: 20px;
852
+ }
853
+
854
+ .opencode-button {
855
+ width: 60px;
856
+ height: 60px;
857
+ border-radius: 50%;
858
+ background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
859
+ border: none;
860
+ cursor: pointer;
861
+ box-shadow: 0 4px 15px rgba(102, 126, 234, 0.4);
862
+ transition: all 0.3s ease;
863
+ display: flex;
864
+ align-items: center;
865
+ justify-content: center;
866
+ color: white;
867
+ padding: 0;
868
+ }
869
+
870
+ .opencode-button:hover {
871
+ transform: scale(1.1);
872
+ box-shadow: 0 6px 20px rgba(102, 126, 234, 0.6);
873
+ background: linear-gradient(135deg, #764ba2 0%, #667eea 100%);
874
+ }
875
+
876
+ .opencode-button.active {
877
+ background: linear-gradient(135deg, #f093fb 0%, #f5576c 100%);
878
+ box-shadow: 0 6px 20px rgba(240, 147, 251, 0.4);
879
+ transform: rotate(180deg);
880
+ }
881
+
882
+ .opencode-button.loading {
883
+ animation: pulse 1s infinite;
884
+ }
885
+
886
+ @keyframes pulse {
887
+ 0%, 100% { opacity: 1; }
888
+ 50% { opacity: 0.5; }
889
+ }
890
+
891
+ .opencode-chat {
892
+ position: absolute;
893
+ width: 700px;
894
+ height: 80vh;
895
+ background: white;
896
+ border-radius: 16px;
897
+ box-shadow: 0 8px 32px rgba(0, 0, 0, 0.12);
898
+ overflow: hidden;
899
+ opacity: 0;
900
+ visibility: hidden;
901
+ transform: translateY(20px) scale(0.95);
902
+ transition: all 0.3s ease;
903
+ display: flex;
904
+ }
905
+
906
+ .opencode-widget.bottom-right .opencode-chat {
907
+ bottom: 80px;
908
+ right: 0;
909
+ }
910
+
911
+ .opencode-widget.bottom-left .opencode-chat {
912
+ bottom: 80px;
913
+ left: 0;
914
+ }
915
+
916
+ .opencode-widget.top-right .opencode-chat {
917
+ top: 80px;
918
+ right: 0;
919
+ }
920
+
921
+ .opencode-widget.top-left .opencode-chat {
922
+ top: 80px;
923
+ left: 0;
924
+ }
925
+
926
+ .opencode-chat.open {
927
+ opacity: 1;
928
+ visibility: visible;
929
+ transform: translateY(0) scale(1);
930
+ }
931
+
932
+ .opencode-session-list {
933
+ width: 240px;
934
+ background: #f8f9fa;
935
+ border-right: 1px solid #e5e7eb;
936
+ display: flex;
937
+ flex-direction: column;
938
+ flex-shrink: 0;
939
+ transition: width 0.2s ease;
940
+ }
941
+
942
+ .opencode-session-list.collapsed {
943
+ width: 40px;
944
+ }
945
+
946
+ .opencode-session-toggle {
947
+ width: 100%;
948
+ height: 40px;
949
+ border: none;
950
+ background: transparent;
951
+ color: #6b7280;
952
+ cursor: pointer;
953
+ display: flex;
954
+ align-items: center;
955
+ justify-content: center;
956
+ transition: all 0.2s;
957
+ border-bottom: 1px solid #e5e7eb;
958
+ }
959
+
960
+ .opencode-session-toggle:hover {
961
+ background: #e5e7eb;
962
+ color: #374151;
963
+ }
964
+
965
+ .opencode-session-list.collapsed .opencode-session-list-header,
966
+ .opencode-session-list.collapsed .opencode-session-list-content {
967
+ display: none;
968
+ }
969
+
970
+ .opencode-session-list-header {
971
+ padding: 16px;
972
+ border-bottom: 1px solid #e5e7eb;
973
+ display: flex;
974
+ justify-content: space-between;
975
+ align-items: center;
976
+ font-weight: 600;
977
+ font-size: 14px;
978
+ color: #374151;
979
+ }
980
+
981
+ .opencode-new-session-btn {
982
+ width: 28px;
983
+ height: 28px;
984
+ border-radius: 6px;
985
+ border: none;
986
+ background: #3b82f6;
987
+ color: white;
988
+ font-size: 18px;
989
+ cursor: pointer;
990
+ display: flex;
991
+ align-items: center;
992
+ justify-content: center;
993
+ transition: all 0.2s;
994
+ }
995
+
996
+ .opencode-new-session-btn:hover {
997
+ background: #2563eb;
998
+ transform: scale(1.05);
999
+ }
1000
+
1001
+ .opencode-session-list-content {
1002
+ flex: 1;
1003
+ overflow-y: auto;
1004
+ padding: 8px;
1005
+ }
1006
+
1007
+ .opencode-session-item {
1008
+ padding: 12px;
1009
+ border-radius: 8px;
1010
+ cursor: pointer;
1011
+ transition: all 0.2s;
1012
+ margin-bottom: 4px;
1013
+ }
1014
+
1015
+ .opencode-session-item:hover {
1016
+ background: #e5e7eb;
1017
+ }
1018
+
1019
+ .opencode-session-item.active {
1020
+ background: #3b82f6;
1021
+ color: white;
1022
+ }
1023
+
1024
+ .opencode-session-title {
1025
+ font-size: 14px;
1026
+ font-weight: 500;
1027
+ margin-bottom: 4px;
1028
+ overflow: hidden;
1029
+ text-overflow: ellipsis;
1030
+ white-space: nowrap;
1031
+ }
1032
+
1033
+ .opencode-session-meta {
1034
+ font-size: 12px;
1035
+ opacity: 0.6;
1036
+ }
1037
+
1038
+ .opencode-session-header {
1039
+ display: flex;
1040
+ justify-content: space-between;
1041
+ align-items: center;
1042
+ margin-bottom: 4px;
1043
+ }
1044
+
1045
+ .opencode-session-delete-btn {
1046
+ width: 20px;
1047
+ height: 20px;
1048
+ border-radius: 4px;
1049
+ border: none;
1050
+ background: transparent;
1051
+ color: #6b7280;
1052
+ font-size: 16px;
1053
+ cursor: pointer;
1054
+ display: flex;
1055
+ align-items: center;
1056
+ justify-content: center;
1057
+ transition: all 0.2s;
1058
+ opacity: 0;
1059
+ flex-shrink: 0;
1060
+ }
1061
+
1062
+ .opencode-session-item:hover .opencode-session-delete-btn {
1063
+ opacity: 1;
1064
+ }
1065
+
1066
+ .opencode-session-delete-btn:hover {
1067
+ background: #ef4444;
1068
+ color: white;
1069
+ }
1070
+
1071
+ .opencode-session-item.active .opencode-session-delete-btn {
1072
+ color: rgba(255, 255, 255, 0.7);
1073
+ }
1074
+
1075
+ .opencode-session-item.active .opencode-session-delete-btn:hover {
1076
+ background: rgba(255, 255, 255, 0.2);
1077
+ color: white;
1078
+ }
1079
+
1080
+ .opencode-iframe-container {
1081
+ flex: 1;
1082
+ position: relative;
1083
+ overflow: hidden;
1084
+ display: flex;
1085
+ flex-direction: column;
1086
+ }
1087
+
1088
+ .opencode-loading-overlay {
1089
+ position: absolute;
1090
+ top: 0;
1091
+ left: 0;
1092
+ right: 0;
1093
+ bottom: 0;
1094
+ background: rgba(255, 255, 255, 0.9);
1095
+ display: none;
1096
+ flex-direction: column;
1097
+ align-items: center;
1098
+ justify-content: center;
1099
+ z-index: 10;
1100
+ transition: opacity 0.3s ease;
1101
+ }
1102
+
1103
+ .opencode-loading-overlay.visible {
1104
+ display: flex;
1105
+ }
1106
+
1107
+ .opencode-loading-spinner {
1108
+ width: 40px;
1109
+ height: 40px;
1110
+ border: 3px solid #e5e7eb;
1111
+ border-top-color: #3b82f6;
1112
+ border-radius: 50%;
1113
+ animation: spin 0.8s linear infinite;
1114
+ }
1115
+
1116
+ @keyframes spin {
1117
+ to { transform: rotate(360deg); }
1118
+ }
1119
+
1120
+ .opencode-loading-text {
1121
+ margin-top: 12px;
1122
+ font-size: 14px;
1123
+ color: #6b7280;
1124
+ }
1125
+
1126
+ .opencode-iframe {
1127
+ position: absolute;
1128
+ top: -136px;
1129
+ width: 100%;
1130
+ height: calc(100% + 136px);
1131
+ border: none;
1132
+ transition: opacity 0.3s ease;
1133
+ }
1134
+
1135
+ .opencode-dark .opencode-chat {
1136
+ background: #1a1a1a;
1137
+ }
1138
+
1139
+ .opencode-dark .opencode-session-list {
1140
+ background: #111827;
1141
+ border-right-color: #374151;
1142
+ }
1143
+
1144
+ .opencode-dark .opencode-session-toggle {
1145
+ color: #9ca3af;
1146
+ border-bottom-color: #374151;
1147
+ }
1148
+
1149
+ .opencode-dark .opencode-session-toggle:hover {
1150
+ background: #374151;
1151
+ color: #f3f4f6;
1152
+ }
1153
+
1154
+ .opencode-dark .opencode-session-list-header {
1155
+ border-bottom-color: #374151;
1156
+ color: #f3f4f6;
1157
+ }
1158
+
1159
+ .opencode-dark .opencode-session-item:hover {
1160
+ background: #374151;
1161
+ }
1162
+
1163
+ .opencode-dark .opencode-session-item.active {
1164
+ background: #3b82f6;
1165
+ }
1166
+
1167
+ .opencode-dark .opencode-loading-overlay {
1168
+ background: rgba(26, 26, 26, 0.9);
1169
+ }
1170
+
1171
+ .opencode-dark .opencode-loading-spinner {
1172
+ border-color: #374151;
1173
+ border-top-color: #3b82f6;
1174
+ }
1175
+
1176
+ .opencode-dark .opencode-loading-text {
1177
+ color: #9ca3af;
1178
+ }
1179
+
1180
+ .opencode-right-toolbar {
1181
+ width: 140px;
1182
+ background: #f8f9fa;
1183
+ border-left: 1px solid #e5e7eb;
1184
+ display: flex;
1185
+ flex-direction: column;
1186
+ flex-shrink: 0;
1187
+ transition: width 0.2s ease;
1188
+ }
1189
+
1190
+ .opencode-right-toolbar.collapsed {
1191
+ width: 40px;
1192
+ }
1193
+
1194
+ .opencode-right-toolbar.collapsed .opencode-selected-nodes {
1195
+ display: none;
1196
+ }
1197
+
1198
+ .opencode-select-btn {
1199
+ width: 100%;
1200
+ height: 44px;
1201
+ border: none;
1202
+ background: transparent;
1203
+ color: #6b7280;
1204
+ cursor: pointer;
1205
+ display: flex;
1206
+ align-items: center;
1207
+ justify-content: center;
1208
+ gap: 8px;
1209
+ transition: all 0.2s;
1210
+ border-bottom: 1px solid #e5e7eb;
1211
+ font-size: 13px;
1212
+ font-weight: 500;
1213
+ }
1214
+
1215
+ .opencode-select-btn:hover {
1216
+ background: #e5e7eb;
1217
+ color: #374151;
1218
+ }
1219
+
1220
+ .opencode-select-btn.active {
1221
+ background: #3b82f6;
1222
+ color: white;
1223
+ }
1224
+
1225
+ .opencode-selected-nodes {
1226
+ flex: 1;
1227
+ display: flex;
1228
+ flex-direction: column;
1229
+ padding: 8px;
1230
+ gap: 6px;
1231
+ overflow-y: auto;
1232
+ overflow-x: hidden;
1233
+ }
1234
+
1235
+ .opencode-selected-nodes:empty::before {
1236
+ content: '暂无选中元素';
1237
+ color: #9ca3af;
1238
+ font-size: 12px;
1239
+ text-align: center;
1240
+ padding: 20px 10px;
1241
+ }
1242
+
1243
+ .opencode-selected-node {
1244
+ display: flex;
1245
+ align-items: center;
1246
+ gap: 8px;
1247
+ padding: 8px 10px;
1248
+ background: white;
1249
+ border: 1px solid #e5e7eb;
1250
+ border-radius: 6px;
1251
+ font-size: 12px;
1252
+ transition: all 0.2s;
1253
+ }
1254
+
1255
+ .opencode-selected-node:hover {
1256
+ border-color: #3b82f6;
1257
+ box-shadow: 0 2px 4px rgba(59, 130, 246, 0.1);
1258
+ }
1259
+
1260
+ .opencode-node-content {
1261
+ flex: 1;
1262
+ min-width: 0;
1263
+ display: flex;
1264
+ flex-direction: column;
1265
+ gap: 2px;
1266
+ }
1267
+
1268
+ .opencode-node-text {
1269
+ color: #374151;
1270
+ font-weight: 500;
1271
+ overflow: hidden;
1272
+ text-overflow: ellipsis;
1273
+ white-space: nowrap;
1274
+ }
1275
+
1276
+ .opencode-node-file {
1277
+ color: #9ca3af;
1278
+ font-size: 11px;
1279
+ overflow: hidden;
1280
+ text-overflow: ellipsis;
1281
+ white-space: nowrap;
1282
+ }
1283
+
1284
+ .opencode-node-remove {
1285
+ width: 18px;
1286
+ height: 18px;
1287
+ border-radius: 4px;
1288
+ border: none;
1289
+ background: transparent;
1290
+ color: #9ca3af;
1291
+ cursor: pointer;
1292
+ display: flex;
1293
+ align-items: center;
1294
+ justify-content: center;
1295
+ font-size: 14px;
1296
+ transition: all 0.2s;
1297
+ flex-shrink: 0;
1298
+ }
1299
+
1300
+ .opencode-node-remove:hover {
1301
+ background: #ef4444;
1302
+ color: white;
1303
+ }
1304
+
1305
+ .opencode-dark .opencode-right-toolbar {
1306
+ background: #111827;
1307
+ border-left-color: #374151;
1308
+ }
1309
+
1310
+ .opencode-dark .opencode-select-btn {
1311
+ color: #9ca3af;
1312
+ border-bottom-color: #374151;
1313
+ }
1314
+
1315
+ .opencode-dark .opencode-select-btn:hover {
1316
+ background: #374151;
1317
+ color: #f3f4f6;
1318
+ }
1319
+
1320
+ .opencode-dark .opencode-selected-nodes:empty::before {
1321
+ color: #6b7280;
1322
+ }
1323
+
1324
+ .opencode-dark .opencode-selected-node {
1325
+ background: #1f2937;
1326
+ border-color: #374151;
1327
+ }
1328
+
1329
+ .opencode-dark .opencode-selected-node:hover {
1330
+ border-color: #3b82f6;
1331
+ }
1332
+
1333
+ .opencode-dark .opencode-node-text {
1334
+ color: #f3f4f6;
1335
+ }
1336
+
1337
+ .opencode-dark .opencode-node-file {
1338
+ color: #6b7280;
1339
+ }
1340
+
1341
+ .opencode-dark .opencode-node-remove {
1342
+ color: #6b7280;
1343
+ }
1344
+
1345
+ .opencode-dark .opencode-node-remove:hover {
1346
+ background: #ef4444;
1347
+ color: white;
1348
+ }
1349
+
1350
+ .opencode-dark .opencode-button {
1351
+ background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
1352
+ box-shadow: 0 4px 15px rgba(102, 126, 234, 0.5);
1353
+ }
1354
+
1355
+ .opencode-notification {
1356
+ position: absolute;
1357
+ top: 20px;
1358
+ left: 50%;
1359
+ transform: translateX(-50%);
1360
+ padding: 12px 20px;
1361
+ background: #10b981;
1362
+ color: white;
1363
+ border-radius: 8px;
1364
+ font-size: 14px;
1365
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
1366
+ animation: slideDown 0.3s ease;
1367
+ z-index: 10;
1368
+ }
1369
+
1370
+ @keyframes slideDown {
1371
+ from {
1372
+ transform: translateX(-50%) translateY(-100%);
1373
+ opacity: 0;
1374
+ }
1375
+ to {
1376
+ transform: translateX(-50%) translateY(0);
1377
+ opacity: 1;
1378
+ }
1379
+ }
1380
+
1381
+ .opencode-select-mode-hint {
1382
+ position: fixed;
1383
+ top: 20px;
1384
+ left: 50%;
1385
+ transform: translateX(-50%);
1386
+ padding: 10px 16px;
1387
+ background: #3b82f6;
1388
+ color: white;
1389
+ border-radius: 8px;
1390
+ font-size: 13px;
1391
+ box-shadow: 0 4px 12px rgba(59, 130, 246, 0.3);
1392
+ z-index: 9999999;
1393
+ display: none;
1394
+ align-items: center;
1395
+ gap: 12px;
1396
+ }
1397
+
1398
+ .opencode-select-mode-hint.visible {
1399
+ display: flex;
1400
+ animation: slideDown 0.3s ease;
1401
+ }
1402
+
1403
+ .opencode-hint-shortcut {
1404
+ padding: 2px 6px;
1405
+ background: rgba(255, 255, 255, 0.2);
1406
+ border-radius: 4px;
1407
+ font-size: 12px;
1408
+ }
1409
+
1410
+ .opencode-selected-bubbles {
1411
+ position: absolute;
1412
+ bottom: 70px;
1413
+ right: 0;
1414
+ display: none;
1415
+ flex-direction: column;
1416
+ gap: 6px;
1417
+ max-width: 220px;
1418
+ max-height: 300px;
1419
+ overflow-y: auto;
1420
+ }
1421
+
1422
+ .opencode-selected-bubbles.visible {
1423
+ display: flex;
1424
+ }
1425
+
1426
+ .opencode-selected-bubble {
1427
+ display: flex;
1428
+ flex-direction: column;
1429
+ gap: 2px;
1430
+ padding: 8px 10px;
1431
+ background: white;
1432
+ border: 1px solid #e5e7eb;
1433
+ border-radius: 8px;
1434
+ font-size: 12px;
1435
+ box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
1436
+ position: relative;
1437
+ }
1438
+
1439
+ .opencode-bubble-text {
1440
+ color: #374151;
1441
+ font-weight: 500;
1442
+ overflow: hidden;
1443
+ text-overflow: ellipsis;
1444
+ white-space: nowrap;
1445
+ }
1446
+
1447
+ .opencode-bubble-file {
1448
+ color: #9ca3af;
1449
+ font-size: 11px;
1450
+ overflow: hidden;
1451
+ text-overflow: ellipsis;
1452
+ white-space: nowrap;
1453
+ }
1454
+
1455
+ .opencode-bubble-remove {
1456
+ position: absolute;
1457
+ top: 4px;
1458
+ right: 4px;
1459
+ width: 16px;
1460
+ height: 16px;
1461
+ border-radius: 50%;
1462
+ border: none;
1463
+ background: transparent;
1464
+ color: #9ca3af;
1465
+ cursor: pointer;
1466
+ display: flex;
1467
+ align-items: center;
1468
+ justify-content: center;
1469
+ font-size: 12px;
1470
+ transition: all 0.2s;
1471
+ }
1472
+
1473
+ .opencode-bubble-remove:hover {
1474
+ background: #ef4444;
1475
+ color: white;
1476
+ }
1477
+
1478
+ .opencode-bubble-empty {
1479
+ padding: 8px 12px;
1480
+ background: white;
1481
+ border: 1px dashed #d1d5db;
1482
+ border-radius: 8px;
1483
+ color: #9ca3af;
1484
+ font-size: 12px;
1485
+ text-align: center;
1486
+ }
1487
+
1488
+ .opencode-dark .opencode-selected-bubble {
1489
+ background: #1f2937;
1490
+ border-color: #374151;
1491
+ }
1492
+
1493
+ .opencode-dark .opencode-bubble-text {
1494
+ color: #f3f4f6;
1495
+ }
1496
+
1497
+ .opencode-dark .opencode-bubble-file {
1498
+ color: #6b7280;
1499
+ }
1500
+
1501
+ .opencode-dark .opencode-bubble-remove {
1502
+ color: #6b7280;
1503
+ }
1504
+
1505
+ .opencode-dark .opencode-bubble-empty {
1506
+ background: #1f2937;
1507
+ border-color: #374151;
1508
+ color: #6b7280;
1509
+ }
1510
+
1511
+ @media (max-width: 768px) {
1512
+ .opencode-chat {
1513
+ width: calc(100vw - 40px);
1514
+ height: calc(100vh - 120px);
1515
+ }
1516
+ }
1517
+ `;
1518
+ }
1519
+ /**
1520
+ * 自动初始化挂件
1521
+ */
1522
+ function autoInit() {
1523
+ const script = document.currentScript || document.querySelector('script[data-opencode-config]');
1524
+ if (script) {
1525
+ const configBase64 = script.getAttribute('data-opencode-config');
1526
+ if (configBase64) {
1527
+ try {
1528
+ const config = JSON.parse(atob(configBase64));
1529
+ if (document.readyState === 'loading') {
1530
+ document.addEventListener('DOMContentLoaded', function () {
1531
+ initOpenCodeWidget(config);
1532
+ });
1533
+ }
1534
+ else {
1535
+ initOpenCodeWidget(config);
1536
+ }
1537
+ }
1538
+ catch (e) {
1539
+ console.error('[OpenCode Widget] Failed to parse config:', e);
1540
+ }
1541
+ }
1542
+ }
1543
+ }
1544
+ // 导出全局初始化函数
1545
+ window.initOpenCodeWidget = initOpenCodeWidget;
1546
+ // 自动初始化
1547
+ autoInit();
1548
+ })();
1549
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../src/client.js"],"names":[],"mappings":"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","sourcesContent":["/**\n * @fileoverview OpenCode 挂件客户端脚本\n * @description 用于在浏览器中显示 OpenCode AI 助手挂件\n */\n\n(function() {\n  'use strict'\n\n  /** @type {string} 初始化标记 */\n  const INIT_MARKER = '__OPENCODE_INITIALIZED__'\n\n  /** @type {string} 选中元素存储键 */\n  const SELECTED_ELEMENTS_KEY = '__opencode_selected_elements__'\n\n  /** @type {number} 服务器同步间隔（毫秒） */\n  const SERVER_SYNC_INTERVAL = 2000\n\n  /** @type {number} 检查 Vue Inspector 间隔（毫秒） */\n  const INSPECTOR_CHECK_INTERVAL = 500\n\n  /** @type {number} 自动打开延迟（毫秒） */\n  const AUTO_OPEN_DELAY = 1000\n\n  /** @type {number} 通知显示时间（毫秒） */\n  const NOTIFICATION_DURATION = 3000\n\n  /**\n   * @typedef {Object} HotkeyConfig\n   * @property {boolean} ctrl - 是否需要 Ctrl/Meta 键\n   * @property {boolean} shift - 是否需要 Shift 键\n   * @property {boolean} alt - 是否需要 Alt 键\n   * @property {string} key - 主键\n   */\n\n  /**\n   * @typedef {Object} SelectedElement\n   * @property {string|null} filePath - 文件路径\n   * @property {number|null} line - 行号\n   * @property {number|null} column - 列号\n   * @property {string} innerText - 元素内部文本\n   * @property {string} description - 元素描述（标签名+选择器）\n   */\n\n  /**\n   * @typedef {Object} WidgetConfig\n   * @property {string} webUrl - Web 服务 URL\n   * @property {string} position - 挂件位置\n   * @property {string} theme - 主题模式\n   * @property {boolean} open - 是否自动打开\n   * @property {string} sessionUrl - 会话 URL\n   * @property {boolean} lazy - 是否懒加载\n   * @property {string} hotkey - 快捷键配置\n   */\n\n  /**\n   * 初始化 OpenCode 挂件\n   * @param {WidgetConfig} config - 挂件配置\n   */\n  function initOpenCodeWidget(config) {\n    if (window[INIT_MARKER]) return\n    window[INIT_MARKER] = true\n\n    const { webUrl, position, theme, open, sessionUrl, lazy, hotkey, cwd } = config\n\n    /** @type {string} 当前页面 URL */\n    let currentPageUrl = ''\n\n    /** @type {string} 当前页面标题 */\n    let currentPageTitle = ''\n\n    /** @type {boolean} 服务是否已启动 */\n    let servicesStarted = !lazy\n\n    /** @type {boolean} 挂件是否打开 */\n    let isOpen = false\n\n    /**\n     * 解析快捷键字符串\n     * @param {string} hotkeyStr - 快捷键字符串，如 'ctrl+k'\n     * @returns {HotkeyConfig} 快捷键配置\n     */\n    function parseHotkey(hotkeyStr) {\n      if (!hotkeyStr) return { ctrl: true, shift: false, alt: false, key: 'k' }\n\n      const parts = hotkeyStr.toLowerCase().split('+')\n      const key = parts.pop()\n\n      return {\n        ctrl: parts.includes('ctrl') || parts.includes('cmd') || parts.includes('meta'),\n        shift: parts.includes('shift'),\n        alt: parts.includes('alt'),\n        key: key || 'k'\n      }\n    }\n\n    /** @type {HotkeyConfig} 主快捷键配置 */\n    const mainHotkey = parseHotkey(hotkey)\n\n    /**\n     * 检查键盘事件是否匹配快捷键\n     * @param {KeyboardEvent} e - 键盘事件\n     * @param {HotkeyConfig} hotkeyConfig - 快捷键配置\n     * @returns {boolean} 是否匹配\n     */\n    function matchHotkey(e, hotkeyConfig) {\n      const ctrlMatch = hotkeyConfig.ctrl ? (e.ctrlKey || e.metaKey) : !(e.ctrlKey || e.metaKey)\n      const shiftMatch = hotkeyConfig.shift ? e.shiftKey : !e.shiftKey\n      const altMatch = hotkeyConfig.alt ? e.altKey : !e.altKey\n      const keyMatch = e.key.toLowerCase() === hotkeyConfig.key.toLowerCase()\n\n      return ctrlMatch && shiftMatch && altMatch && keyMatch\n    }\n\n    /**\n     * 从 sessionStorage 加载选中的元素\n     * @returns {SelectedElement[]} 选中的元素列表\n     */\n    function loadSelectedElements() {\n      try {\n        const stored = sessionStorage.getItem(SELECTED_ELEMENTS_KEY)\n        if (stored) {\n          return JSON.parse(stored)\n        }\n      } catch (e) {\n        // 忽略错误\n      }\n      return []\n    }\n\n    /**\n     * 保存选中的元素到 sessionStorage\n     * @param {SelectedElement[]} elements - 选中的元素列表\n     */\n    function saveSelectedElements(elements) {\n      try {\n        sessionStorage.setItem(SELECTED_ELEMENTS_KEY, JSON.stringify(elements))\n      } catch (e) {\n        // 忽略错误\n      }\n    }\n\n    /** @type {SelectedElement[]} 选中的元素列表 */\n    let selectedElements = loadSelectedElements()\n\n    /**\n     * 确保服务已启动\n     * @returns {Promise<boolean>} 是否成功启动\n     */\n    async function ensureServicesStarted() {\n      if (servicesStarted) return true\n\n      try {\n        const res = await fetch('/__opencode_start__')\n        const data = await res.json()\n        if (data.success) {\n          servicesStarted = true\n          if (data.sessionUrl && iframe) {\n            iframe.src = data.sessionUrl\n          }\n          return true\n        }\n      } catch (e) {\n        console.error('[OpenCode Widget] Failed to start services:', e)\n      }\n      return false\n    }\n\n    /**\n     * 更新页面上下文\n     * @param {boolean} [force=false] - 是否强制更新\n     */\n    function updateContext(force = false) {\n      if (!servicesStarted) return\n\n      const newUrl = window.location.href\n      const newTitle = document.title\n\n      if (force || newUrl !== currentPageUrl || newTitle !== currentPageTitle) {\n        currentPageUrl = newUrl\n        currentPageTitle = newTitle\n\n        fetch('/__opencode_context__', {\n          method: 'POST',\n          headers: { 'Content-Type': 'application/json' },\n          body: JSON.stringify({ url: newUrl, title: newTitle, selectedElements })\n        }).catch(() => {})\n      }\n    }\n\n    // 监听路由变化\n    const originalPushState = history.pushState\n    const originalReplaceState = history.replaceState\n\n    history.pushState = function(...args) {\n      originalPushState.apply(this, args)\n      setTimeout(updateContext, 0)\n    }\n\n    history.replaceState = function(...args) {\n      originalReplaceState.apply(this, args)\n      setTimeout(updateContext, 0)\n    }\n\n    window.addEventListener('popstate', () => setTimeout(updateContext, 0))\n    window.addEventListener('hashchange', () => setTimeout(updateContext, 0))\n\n    // 监听标题变化\n    const titleObserver = new MutationObserver(() => {\n      if (document.title !== currentPageTitle) {\n        updateContext()\n      }\n    })\n\n    if (document.head) {\n      titleObserver.observe(document.head, { childList: true, subtree: true })\n    }\n\n    if (servicesStarted) {\n      updateContext(true)\n    }\n\n    /** @type {string} iframe URL */\n    const iframeUrl = sessionUrl || webUrl\n\n    // 创建样式\n    const style = document.createElement('style')\n    style.textContent = buildWidgetStyles()\n    document.head.appendChild(style)\n\n    // 创建容器\n    const container = document.createElement('div')\n    container.className = `opencode-widget ${position}`\n\n    // 创建按钮\n    const button = document.createElement('button')\n    button.className = 'opencode-button'\n    button.innerHTML = `\n      <svg t=\"1775402599580\" class=\"icon\" viewBox=\"0 0 1024 1024\" version=\"1.1\" xmlns=\" http://www.w3.org/2000/svg \" p-id=\"5390\" xmlns:xlink=\" http://www.w3.org/1999/xlink \" width=\"200\" height=\"200\"><path d=\"M512 981.33H85.34c-15.85 0-30.38-8.77-37.77-22.81a42.624 42.624 0 0 1 2.6-44.02L135 791.08C75.25 710.5 42.67 612.6 42.67 512 42.67 253.21 253.21 42.67 512 42.67S981.34 253.21 981.34 512 770.8 981.33 512 981.33zM166.44 896H512c211.73 0 384-172.27 384-384S723.73 128 512 128 128 300.27 128 512c0 91.29 32.83 179.9 92.46 249.46 12.58 14.69 13.73 36 2.77 51.94L166.44 896z\" fill=\"white\" p-id=\"5391\"></path><path d=\"M384 448m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z\" fill=\"white\" p-id=\"5392\"></path><path d=\"M640 448m-64 0a64 64 0 1 0 128 0 64 64 0 1 0-128 0Z\" fill=\"white\" p-id=\"5393\"></path></svg>\n    `\n    button.setAttribute('aria-label', 'Open AI Assistant')\n\n    // 创建聊天面板\n    const chat = document.createElement('div')\n    chat.className = 'opencode-chat'\n\n    // 创建会话列表\n    const sessionList = document.createElement('div')\n    sessionList.className = 'opencode-session-list collapsed'\n\n    // 创建折叠按钮\n    const toggleBtn = document.createElement('button')\n    toggleBtn.className = 'opencode-session-toggle'\n    toggleBtn.innerHTML = `\n      <svg viewBox=\"0 0 24 24\" width=\"16\" height=\"16\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M4 6h16M4 12h16M4 18h16\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\"/>\n      </svg>\n    `\n    toggleBtn.title = '展开会话列表'\n\n    // 创建会话列表头部\n    const sessionListHeader = document.createElement('div')\n    sessionListHeader.className = 'opencode-session-list-header'\n    sessionListHeader.innerHTML = `\n      <span>会话列表</span>\n      <button class=\"opencode-new-session-btn\" title=\"新建会话\">+</button>\n    `\n\n    // 创建会话列表内容\n    const sessionListContent = document.createElement('div')\n    sessionListContent.className = 'opencode-session-list-content'\n\n    sessionList.appendChild(toggleBtn)\n    sessionList.appendChild(sessionListHeader)\n    sessionList.appendChild(sessionListContent)\n\n    // 折叠/展开会话列表\n    let isSessionListCollapsed = true\n    function toggleSessionList() {\n      isSessionListCollapsed = !isSessionListCollapsed\n      sessionList.classList.toggle('collapsed', isSessionListCollapsed)\n      toggleBtn.title = isSessionListCollapsed ? '展开会话列表' : '折叠会话列表'\n    }\n    toggleBtn.addEventListener('click', toggleSessionList)\n\n    // 创建 iframe 容器\n    const iframeContainer = document.createElement('div')\n    iframeContainer.className = 'opencode-iframe-container'\n\n    // 创建加载指示器\n    const loadingOverlay = document.createElement('div')\n    loadingOverlay.className = 'opencode-loading-overlay'\n    loadingOverlay.innerHTML = `\n      <div class=\"opencode-loading-spinner\"></div>\n      <div class=\"opencode-loading-text\">加载中...</div>\n    `\n\n    // 创建 iframe\n    const iframe = document.createElement('iframe')\n    iframe.className = 'opencode-iframe'\n    iframe.src = servicesStarted ? iframeUrl : 'about:blank'\n    iframe.allow = 'clipboard-write; clipboard-read'\n    iframe.referrerPolicy = 'origin'\n\n    iframe.onload = function() {\n      if (servicesStarted) {\n        updateContext()\n        loadSessions()\n      }\n      hideLoading()\n    }\n\n    iframeContainer.appendChild(loadingOverlay)\n    iframeContainer.appendChild(iframe)\n\n    // 创建右侧工具栏\n    const rightToolbar = document.createElement('div')\n    rightToolbar.className = `opencode-right-toolbar${selectedElements.length === 0 ? ' collapsed' : ''}`\n\n    // 创建元素选择按钮\n    const selectButton = document.createElement('button')\n    selectButton.className = 'opencode-select-btn'\n    selectButton.innerHTML = `\n      <svg viewBox=\"0 0 1024 1024\" width=\"18\" height=\"18\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path fill=\"currentColor\" d=\"M512 896a384 384 0 1 0 0-768 384 384 0 0 0 0 768m0 64a448 448 0 1 1 0-896 448 448 0 0 1 0 896\"></path><path fill=\"currentColor\" d=\"M512 96a32 32 0 0 1 32 32v192a32 32 0 0 1-64 0V128a32 32 0 0 1 32-32m0 576a32 32 0 0 1 32 32v192a32 32 0 1 1-64 0V704a32 32 0 0 1 32-32M96 512a32 32 0 0 1 32-32h192a32 32 0 0 1 0 64H128a32 32 0 0 1-32-32m576 0a32 32 0 0 1 32-32h192a32 32 0 1 1 0 64H704a32 32 0 0 1-32-32\"></path>\n      </svg>\n    `\n    selectButton.title = '选择页面元素'\n\n    // 创建已选节点容器\n    const selectedNodesContainer = document.createElement('div')\n    selectedNodesContainer.className = 'opencode-selected-nodes'\n\n    rightToolbar.appendChild(selectButton)\n    rightToolbar.appendChild(selectedNodesContainer)\n\n    // 创建选择模式常驻提示（固定到页面顶部）\n    const selectModeHint = document.createElement('div')\n    selectModeHint.className = 'opencode-select-mode-hint'\n    selectModeHint.innerHTML = `\n      <span>🎯 选择模式已开启 - 点击元素进行选择</span>\n      <span class=\"opencode-hint-shortcut\">按 ESC 退出</span>\n    `\n\n    // 创建已选节点气泡容器（气泡按钮上方）\n    const selectedBubbles = document.createElement('div')\n    selectedBubbles.className = 'opencode-selected-bubbles'\n\n    chat.appendChild(sessionList)\n    chat.appendChild(iframeContainer)\n    chat.appendChild(rightToolbar)\n\n    container.appendChild(button)\n    container.appendChild(selectedBubbles)\n    container.appendChild(chat)\n    document.body.appendChild(container)\n    document.body.appendChild(selectModeHint)\n\n    if (selectedElements.length > 0) {\n      renderSelectedNodes()\n    }\n\n    /** @type {Array} 会话列表 */\n    let sessions = []\n\n    /** @type {string|null} 当前会话 ID */\n    let currentSessionId = null\n\n    /**\n     * 从 URL 中提取会话 ID\n     */\n    function extractSessionId(url) {\n      if (!url) return null\n      const match = url.match(/\\/session\\/([^/?]+)/)\n      return match ? match[1] : null\n    }\n\n    // 从初始 URL 中提取会话 ID\n    currentSessionId = extractSessionId(sessionUrl)\n\n    /**\n     * 显示加载状态\n     */\n    function showLoading() {\n      loadingOverlay.classList.add('visible')\n    }\n\n    /**\n     * 隐藏加载状态\n     */\n    function hideLoading() {\n      loadingOverlay.classList.remove('visible')\n    }\n\n    /**\n     * 加载会话列表\n     */\n    async function loadSessions() {\n      try {\n        const response = await fetch('/__opencode_sessions__')\n        sessions = await response.json()\n        renderSessionList()\n      } catch (e) {\n        console.error('Failed to load sessions:', e)\n      }\n    }\n\n    /**\n     * 渲染会话列表\n     */\n    function renderSessionList() {\n      sessionListContent.innerHTML = ''\n      \n      const currentProjectSessions = sessions.filter(session => session.directory === cwd)\n      \n      currentProjectSessions.forEach(session => {\n        const item = document.createElement('div')\n        item.className = 'opencode-session-item'\n        if (session.id === currentSessionId) {\n          item.classList.add('active')\n        }\n        \n        const header = document.createElement('div')\n        header.className = 'opencode-session-header'\n        \n        const title = document.createElement('div')\n        title.className = 'opencode-session-title'\n        title.textContent = session.title || '新会话'\n        \n        const deleteBtn = document.createElement('button')\n        deleteBtn.className = 'opencode-session-delete-btn'\n        deleteBtn.innerHTML = '×'\n        deleteBtn.title = '删除会话'\n        deleteBtn.addEventListener('click', (e) => {\n          e.stopPropagation()\n          confirmDeleteSession(session)\n        })\n        \n        header.appendChild(title)\n        header.appendChild(deleteBtn)\n        \n        const meta = document.createElement('div')\n        meta.className = 'opencode-session-meta'\n        const date = new Date(session.time.updated)\n        meta.textContent = date.toLocaleDateString() + ' ' + date.toLocaleTimeString()\n        \n        item.appendChild(header)\n        item.appendChild(meta)\n        \n        item.addEventListener('click', () => {\n          switchSession(session)\n        })\n        \n        sessionListContent.appendChild(item)\n      })\n    }\n\n    /**\n     * 切换会话\n     */\n    function switchSession(session) {\n      if (session.id === currentSessionId) {\n        return\n      }\n      \n      currentSessionId = session.id\n      const encodedDir = btoa(cwd)\n      const baseUrl = iframeUrl.split('/').slice(0, 3).join('/')\n      showLoading()\n      iframe.src = `${baseUrl}/${encodedDir}/session/${session.id}`\n      renderSessionList()\n    }\n\n    /**\n     * 创建新会话\n     */\n    async function createNewSession() {\n      try {\n        const response = await fetch('/__opencode_sessions__', {\n          method: 'POST'\n        })\n        const newSession = await response.json()\n        sessions.unshift(newSession)\n        switchSession(newSession)\n      } catch (e) {\n        console.error('Failed to create session:', e)\n        showNotification('创建会话失败')\n      }\n    }\n\n    /**\n     * 确认删除会话\n     */\n    function confirmDeleteSession(session) {\n      const confirmed = confirm(`确定要删除会话 \"${session.title || '新会话'}\" 吗？`)\n      if (confirmed) {\n        deleteSession(session)\n      }\n    }\n\n    /**\n     * 删除会话\n     */\n    async function deleteSession(session) {\n      try {\n        const response = await fetch(`/__opencode_sessions__?id=${session.id}`, {\n          method: 'DELETE'\n        })\n        \n        if (!response.ok) {\n          throw new Error('Delete failed')\n        }\n        \n        sessions = sessions.filter(s => s.id !== session.id)\n        \n        if (session.id === currentSessionId) {\n          const remainingSessions = sessions.filter(s => s.directory === cwd)\n          if (remainingSessions.length > 0) {\n            switchSession(remainingSessions[0])\n          } else {\n            currentSessionId = null\n            iframe.src = 'about:blank'\n          }\n        }\n        \n        renderSessionList()\n        showNotification('会话已删除')\n      } catch (e) {\n        console.error('Failed to delete session:', e)\n        showNotification('删除会话失败')\n      }\n    }\n\n    // 绑定新建会话按钮\n    sessionListHeader.querySelector('.opencode-new-session-btn').addEventListener('click', createNewSession)\n\n    /**\n     * 应用主题\n     */\n    function applyTheme() {\n      if (theme === 'auto') {\n        const prefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches\n        container.classList.toggle('opencode-dark', prefersDark)\n      } else {\n        container.classList.toggle('opencode-dark', theme === 'dark')\n      }\n    }\n\n    applyTheme()\n\n    if (theme === 'auto') {\n      window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', applyTheme)\n    }\n\n    /**\n     * 切换挂件显示状态\n     */\n    async function toggle() {\n      if (lazy && !servicesStarted) {\n        button.classList.add('loading')\n        const started = await ensureServicesStarted()\n        button.classList.remove('loading')\n        if (!started) {\n          showNotification('服务启动失败，请检查控制台')\n          return\n        }\n      }\n\n      isOpen = !isOpen\n      chat.classList.toggle('open', isOpen)\n      button.classList.toggle('active', isOpen)\n\n      if (isOpen) {\n        updateContext()\n      }\n    }\n\n    button.addEventListener('click', toggle)\n\n    document.addEventListener('keydown', (e) => {\n      if (matchHotkey(e, mainHotkey)) {\n        e.preventDefault()\n        toggle()\n      }\n    })\n\n    if (open && servicesStarted) {\n      setTimeout(() => {\n        toggle()\n      }, AUTO_OPEN_DELAY)\n    }\n\n    /**\n     * 显示通知\n     * @param {string} message - 通知消息\n     */\n    function showNotification(message) {\n      const notification = document.createElement('div')\n      notification.className = 'opencode-notification'\n      notification.textContent = message\n      chat.appendChild(notification)\n\n      setTimeout(() => {\n        notification.remove()\n      }, NOTIFICATION_DURATION)\n    }\n\n    /**\n     * 添加选中元素\n     * @param {SelectedElement} elementInfo - 元素信息\n     */\n    function addElement(elementInfo) {\n      const key = elementInfo.filePath && elementInfo.line\n        ? `${elementInfo.filePath}:${elementInfo.line}`\n        : null\n\n      const exists = key && selectedElements.some(el => {\n        const elKey = el.filePath && el.line ? `${el.filePath}:${el.line}` : null\n        return elKey === key\n      })\n\n      if (!exists) {\n        selectedElements.push(elementInfo)\n        saveSelectedElements(selectedElements)\n        renderSelectedNodes()\n        if (isSelectMode) {\n          renderSelectedBubbles()\n        }\n        showNotification(`已选中元素 (${selectedElements.length}个)`)\n      } else {\n        showNotification('该元素已选中')\n      }\n    }\n\n    /**\n     * 移除选中元素\n     * @param {number} index - 元素索引\n     */\n    function removeElement(index) {\n      selectedElements.splice(index, 1)\n      saveSelectedElements(selectedElements)\n      renderSelectedNodes()\n      updateToolbarState()\n      sendSelectedElements()\n    }\n\n    /**\n     * 清除所有选中元素\n     */\n    function clearAllElements() {\n      selectedElements = []\n      saveSelectedElements(selectedElements)\n      renderSelectedNodes()\n      updateToolbarState()\n      sendSelectedElements()\n      showNotification('已清除所有选中元素')\n    }\n\n    /**\n     * 更新工具栏状态\n     */\n    function updateToolbarState() {\n      if (selectedElements.length > 0) {\n        rightToolbar.classList.remove('collapsed')\n      } else {\n        rightToolbar.classList.add('collapsed')\n      }\n    }\n\n    /** @type {boolean} 是否处于元素选择模式 */\n    let isSelectMode = false\n\n    /**\n     * 切换元素选择模式\n     */\n    function toggleSelectMode() {\n      const inspector = window.__VUE_INSPECTOR__\n      \n      if (!inspector) {\n        showNotification('Vue Inspector 未加载，无法使用元素选择功能')\n        return\n      }\n      \n      isSelectMode = !isSelectMode\n      selectButton.classList.toggle('active', isSelectMode)\n      selectModeHint.classList.toggle('visible', isSelectMode)\n      selectedBubbles.classList.toggle('visible', isSelectMode)\n      \n      if (isSelectMode) {\n        chat.style.display = 'none'\n        inspector.enable()\n        renderSelectedBubbles()\n      } else {\n        chat.style.display = ''\n        inspector.disable()\n      }\n    }\n\n    /**\n     * 退出选择模式\n     */\n    function exitSelectMode() {\n      if (!isSelectMode) return\n      \n      const inspector = window.__VUE_INSPECTOR__\n      if (inspector) {\n        inspector.disable()\n      }\n      \n      isSelectMode = false\n      selectButton.classList.remove('active')\n      selectModeHint.classList.remove('visible')\n      selectedBubbles.classList.remove('visible')\n      chat.style.display = ''\n    }\n\n    // ESC 键退出选择模式\n    document.addEventListener('keydown', (e) => {\n      if (e.key === 'Escape' && isSelectMode) {\n        exitSelectMode()\n      }\n    })\n\n    /**\n     * 渲染已选节点气泡\n     */\n    function renderSelectedBubbles() {\n      selectedBubbles.innerHTML = ''\n      \n      if (selectedElements.length === 0) {\n        selectedBubbles.innerHTML = '<div class=\"opencode-bubble-empty\">暂无选中元素</div>'\n        return\n      }\n      \n      selectedElements.forEach((element, index) => {\n        const bubble = document.createElement('div')\n        bubble.className = 'opencode-selected-bubble'\n        \n        const description = element.description || '未知元素'\n        const fileName = element.filePath ? element.filePath.split('/').pop() : ''\n        const lineInfo = element.line ? `:${element.line}` : ''\n        \n        bubble.innerHTML = `\n          <span class=\"opencode-bubble-text\">${description}</span>\n          ${fileName ? `<span class=\"opencode-bubble-file\">${fileName}${lineInfo}</span>` : ''}\n          <button class=\"opencode-bubble-remove\" data-index=\"${index}\">×</button>\n        `\n        \n        bubble.querySelector('.opencode-bubble-remove').addEventListener('click', (e) => {\n          e.stopPropagation()\n          removeElement(index)\n          renderSelectedBubbles()\n        })\n        \n        selectedBubbles.appendChild(bubble)\n      })\n    }\n    function renderSelectedNodes() {\n      selectedNodesContainer.innerHTML = ''\n      \n      if (selectedElements.length === 0) {\n        rightToolbar.classList.add('collapsed')\n      } else {\n        rightToolbar.classList.remove('collapsed')\n      }\n      \n      selectedElements.forEach((element, index) => {\n        const node = document.createElement('div')\n        node.className = 'opencode-selected-node'\n        \n        const description = element.description || '未知元素'\n        const textPreview = element.innerText ? element.innerText.substring(0, 30) : ''\n        const fileName = element.filePath ? element.filePath.split('/').pop() : '未知文件'\n        const lineInfo = element.line ? `:${element.line}` : ''\n        \n        node.innerHTML = `\n          <div class=\"opencode-node-content\">\n            <span class=\"opencode-node-text\">${description}</span>\n            <span class=\"opencode-node-file\">${textPreview ? textPreview + ' · ' : ''}${fileName}${lineInfo}</span>\n          </div>\n          <button class=\"opencode-node-remove\" data-index=\"${index}\">×</button>\n        `\n        \n        node.querySelector('.opencode-node-remove').addEventListener('click', (e) => {\n          e.stopPropagation()\n          removeElement(index)\n        })\n        \n        selectedNodesContainer.appendChild(node)\n      })\n    }\n\n    // 绑定选择按钮点击事件\n    selectButton.addEventListener('click', toggleSelectMode)\n\n    /**\n     * 发送选中元素到服务器\n     */\n    function sendSelectedElements() {\n      if (!servicesStarted) return\n\n      fetch('/__opencode_context__', {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({\n          url: currentPageUrl,\n          title: currentPageTitle,\n          selectedElements: selectedElements\n        })\n      }).catch(() => {})\n    }\n\n    /**\n     * 与服务器同步选中元素\n     */\n    function syncWithServer() {\n      if (!servicesStarted) return\n\n      fetch('/__opencode_context__')\n        .then(res => res.json())\n        .then(data => {\n          if (data.selectedElements && data.selectedElements.length === 0 && selectedElements.length > 0) {\n            selectedElements = []\n            saveSelectedElements(selectedElements)\n            renderSelectedNodes()\n          }\n        })\n        .catch(() => {})\n    }\n\n    setInterval(syncWithServer, SERVER_SYNC_INTERVAL)\n\n    /**\n     * 截断字符串\n     * @param {string} str - 原字符串\n     * @param {number} maxLength - 最大长度\n     * @returns {string} 截断后的字符串\n     */\n    function truncate(str, maxLength) {\n      if (!str) return ''\n      return str.length > maxLength ? str.substring(0, maxLength) + '...' : str\n    }\n\n    /**\n     * 获取元素的直接文本内容\n     * @param {Element} element - DOM 元素\n     * @returns {string} 直接文本内容\n     */\n    function getDirectText(element) {\n      let text = ''\n      for (const child of element.childNodes) {\n        if (child.nodeType === Node.TEXT_NODE) {\n          text += child.textContent || ''\n        }\n      }\n      return text.trim()\n    }\n\n    /**\n     * 获取元素描述信息\n     * @param {Element} element - DOM 元素\n     * @returns {string} 元素描述\n     */\n    function getElementDescription(element) {\n      const tag = element.tagName.toLowerCase()\n      const parts = [tag]\n      \n      const id = element.id\n      if (id) parts.push(`#${id}`)\n      \n      const className = element.className && typeof element.className === 'string'\n        ? element.className.trim().split(/\\s+/).filter(Boolean).slice(0, 2).join('.')\n        : ''\n      if (className) parts.push(`.${className}`)\n      \n      const name = element.getAttribute('name')\n      if (name) parts.push(`[name=\"${name}\"]`)\n      \n      const placeholder = element.getAttribute('placeholder')\n      if (placeholder) parts.push(`[placeholder=\"${placeholder.substring(0, 20)}\"]`)\n      \n      const src = element.getAttribute('src')\n      if (src) parts.push(`[src]`)\n      \n      const href = element.getAttribute('href')\n      if (href && href !== '#') parts.push(`[href]`)\n      \n      return parts.join('')\n    }\n\n    /**\n     * 设置 Vue Inspector 钩子\n     */\n    function setupInspectorHook() {\n      if (window.__VUE_INSPECTOR__) {\n        const inspector = window.__VUE_INSPECTOR__\n        const originalHandleClick = inspector.handleClick.bind(inspector)\n\n        inspector.handleClick = function(e) {\n          if (isSelectMode) {\n            const { targetNode, params } = inspector.getTargetNode(e)\n            if (targetNode && params) {\n              const innerText = getDirectText(targetNode)\n              const description = getElementDescription(targetNode)\n\n              const elementInfo = {\n                filePath: params.file,\n                line: params.line,\n                column: params.column,\n                innerText: truncate(innerText, 200),\n                description\n              }\n              addElement(elementInfo)\n              sendSelectedElements()\n            }\n            return\n          }\n\n          return originalHandleClick.call(inspector, e)\n        }\n      }\n    }\n\n    if (window.__VUE_INSPECTOR__) {\n      setupInspectorHook()\n    } else {\n      const checkInspector = setInterval(() => {\n        if (window.__VUE_INSPECTOR__) {\n          setupInspectorHook()\n          clearInterval(checkInspector)\n        }\n      }, INSPECTOR_CHECK_INTERVAL)\n    }\n\n    // 导出全局 API\n    window.OpenCodeWidget = {\n      open: () => { if (!isOpen) toggle() },\n      close: () => { if (isOpen) toggle() },\n      toggle,\n      showNotification,\n      updateContext,\n    }\n  }\n\n  /**\n   * 构建挂件样式\n   * @returns {string} CSS 样式字符串\n   */\n  function buildWidgetStyles() {\n    return `\n      .opencode-widget {\n        position: fixed;\n        z-index: 999999;\n        font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n      }\n\n      .opencode-widget.bottom-right {\n        bottom: 20px;\n        right: 20px;\n      }\n\n      .opencode-widget.bottom-left {\n        bottom: 20px;\n        left: 20px;\n      }\n\n      .opencode-widget.top-right {\n        top: 20px;\n        right: 20px;\n      }\n\n      .opencode-widget.top-left {\n        top: 20px;\n        left: 20px;\n      }\n\n      .opencode-button {\n        width: 60px;\n        height: 60px;\n        border-radius: 50%;\n        background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);\n        border: none;\n        cursor: pointer;\n        box-shadow: 0 4px 15px rgba(102, 126, 234, 0.4);\n        transition: all 0.3s ease;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        color: white;\n        padding: 0;\n      }\n\n      .opencode-button:hover {\n        transform: scale(1.1);\n        box-shadow: 0 6px 20px rgba(102, 126, 234, 0.6);\n        background: linear-gradient(135deg, #764ba2 0%, #667eea 100%);\n      }\n\n      .opencode-button.active {\n        background: linear-gradient(135deg, #f093fb 0%, #f5576c 100%);\n        box-shadow: 0 6px 20px rgba(240, 147, 251, 0.4);\n        transform: rotate(180deg);\n      }\n\n      .opencode-button.loading {\n        animation: pulse 1s infinite;\n      }\n\n      @keyframes pulse {\n        0%, 100% { opacity: 1; }\n        50% { opacity: 0.5; }\n      }\n\n      .opencode-chat {\n        position: absolute;\n        width: 700px;\n        height: 80vh;\n        background: white;\n        border-radius: 16px;\n        box-shadow: 0 8px 32px rgba(0, 0, 0, 0.12);\n        overflow: hidden;\n        opacity: 0;\n        visibility: hidden;\n        transform: translateY(20px) scale(0.95);\n        transition: all 0.3s ease;\n        display: flex;\n      }\n\n      .opencode-widget.bottom-right .opencode-chat {\n        bottom: 80px;\n        right: 0;\n      }\n\n      .opencode-widget.bottom-left .opencode-chat {\n        bottom: 80px;\n        left: 0;\n      }\n\n      .opencode-widget.top-right .opencode-chat {\n        top: 80px;\n        right: 0;\n      }\n\n      .opencode-widget.top-left .opencode-chat {\n        top: 80px;\n        left: 0;\n      }\n\n      .opencode-chat.open {\n        opacity: 1;\n        visibility: visible;\n        transform: translateY(0) scale(1);\n      }\n\n      .opencode-session-list {\n        width: 240px;\n        background: #f8f9fa;\n        border-right: 1px solid #e5e7eb;\n        display: flex;\n        flex-direction: column;\n        flex-shrink: 0;\n        transition: width 0.2s ease;\n      }\n\n      .opencode-session-list.collapsed {\n        width: 40px;\n      }\n\n      .opencode-session-toggle {\n        width: 100%;\n        height: 40px;\n        border: none;\n        background: transparent;\n        color: #6b7280;\n        cursor: pointer;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        transition: all 0.2s;\n        border-bottom: 1px solid #e5e7eb;\n      }\n\n      .opencode-session-toggle:hover {\n        background: #e5e7eb;\n        color: #374151;\n      }\n\n      .opencode-session-list.collapsed .opencode-session-list-header,\n      .opencode-session-list.collapsed .opencode-session-list-content {\n        display: none;\n      }\n\n      .opencode-session-list-header {\n        padding: 16px;\n        border-bottom: 1px solid #e5e7eb;\n        display: flex;\n        justify-content: space-between;\n        align-items: center;\n        font-weight: 600;\n        font-size: 14px;\n        color: #374151;\n      }\n\n      .opencode-new-session-btn {\n        width: 28px;\n        height: 28px;\n        border-radius: 6px;\n        border: none;\n        background: #3b82f6;\n        color: white;\n        font-size: 18px;\n        cursor: pointer;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        transition: all 0.2s;\n      }\n\n      .opencode-new-session-btn:hover {\n        background: #2563eb;\n        transform: scale(1.05);\n      }\n\n      .opencode-session-list-content {\n        flex: 1;\n        overflow-y: auto;\n        padding: 8px;\n      }\n\n      .opencode-session-item {\n        padding: 12px;\n        border-radius: 8px;\n        cursor: pointer;\n        transition: all 0.2s;\n        margin-bottom: 4px;\n      }\n\n      .opencode-session-item:hover {\n        background: #e5e7eb;\n      }\n\n      .opencode-session-item.active {\n        background: #3b82f6;\n        color: white;\n      }\n\n      .opencode-session-title {\n        font-size: 14px;\n        font-weight: 500;\n        margin-bottom: 4px;\n        overflow: hidden;\n        text-overflow: ellipsis;\n        white-space: nowrap;\n      }\n\n      .opencode-session-meta {\n        font-size: 12px;\n        opacity: 0.6;\n      }\n\n      .opencode-session-header {\n        display: flex;\n        justify-content: space-between;\n        align-items: center;\n        margin-bottom: 4px;\n      }\n\n      .opencode-session-delete-btn {\n        width: 20px;\n        height: 20px;\n        border-radius: 4px;\n        border: none;\n        background: transparent;\n        color: #6b7280;\n        font-size: 16px;\n        cursor: pointer;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        transition: all 0.2s;\n        opacity: 0;\n        flex-shrink: 0;\n      }\n\n      .opencode-session-item:hover .opencode-session-delete-btn {\n        opacity: 1;\n      }\n\n      .opencode-session-delete-btn:hover {\n        background: #ef4444;\n        color: white;\n      }\n\n      .opencode-session-item.active .opencode-session-delete-btn {\n        color: rgba(255, 255, 255, 0.7);\n      }\n\n      .opencode-session-item.active .opencode-session-delete-btn:hover {\n        background: rgba(255, 255, 255, 0.2);\n        color: white;\n      }\n\n      .opencode-iframe-container {\n        flex: 1;\n        position: relative;\n        overflow: hidden;\n        display: flex;\n        flex-direction: column;\n      }\n\n      .opencode-loading-overlay {\n        position: absolute;\n        top: 0;\n        left: 0;\n        right: 0;\n        bottom: 0;\n        background: rgba(255, 255, 255, 0.9);\n        display: none;\n        flex-direction: column;\n        align-items: center;\n        justify-content: center;\n        z-index: 10;\n        transition: opacity 0.3s ease;\n      }\n\n      .opencode-loading-overlay.visible {\n        display: flex;\n      }\n\n      .opencode-loading-spinner {\n        width: 40px;\n        height: 40px;\n        border: 3px solid #e5e7eb;\n        border-top-color: #3b82f6;\n        border-radius: 50%;\n        animation: spin 0.8s linear infinite;\n      }\n\n      @keyframes spin {\n        to { transform: rotate(360deg); }\n      }\n\n      .opencode-loading-text {\n        margin-top: 12px;\n        font-size: 14px;\n        color: #6b7280;\n      }\n\n      .opencode-iframe {\n        position: absolute;\n        top: -136px;\n        width: 100%;\n        height: calc(100% + 136px);\n        border: none;\n        transition: opacity 0.3s ease;\n      }\n\n      .opencode-dark .opencode-chat {\n        background: #1a1a1a;\n      }\n\n      .opencode-dark .opencode-session-list {\n        background: #111827;\n        border-right-color: #374151;\n      }\n\n      .opencode-dark .opencode-session-toggle {\n        color: #9ca3af;\n        border-bottom-color: #374151;\n      }\n\n      .opencode-dark .opencode-session-toggle:hover {\n        background: #374151;\n        color: #f3f4f6;\n      }\n\n      .opencode-dark .opencode-session-list-header {\n        border-bottom-color: #374151;\n        color: #f3f4f6;\n      }\n\n      .opencode-dark .opencode-session-item:hover {\n        background: #374151;\n      }\n\n      .opencode-dark .opencode-session-item.active {\n        background: #3b82f6;\n      }\n\n      .opencode-dark .opencode-loading-overlay {\n        background: rgba(26, 26, 26, 0.9);\n      }\n\n      .opencode-dark .opencode-loading-spinner {\n        border-color: #374151;\n        border-top-color: #3b82f6;\n      }\n\n      .opencode-dark .opencode-loading-text {\n        color: #9ca3af;\n      }\n\n      .opencode-right-toolbar {\n        width: 140px;\n        background: #f8f9fa;\n        border-left: 1px solid #e5e7eb;\n        display: flex;\n        flex-direction: column;\n        flex-shrink: 0;\n        transition: width 0.2s ease;\n      }\n\n      .opencode-right-toolbar.collapsed {\n        width: 40px;\n      }\n\n      .opencode-right-toolbar.collapsed .opencode-selected-nodes {\n        display: none;\n      }\n\n      .opencode-select-btn {\n        width: 100%;\n        height: 44px;\n        border: none;\n        background: transparent;\n        color: #6b7280;\n        cursor: pointer;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        gap: 8px;\n        transition: all 0.2s;\n        border-bottom: 1px solid #e5e7eb;\n        font-size: 13px;\n        font-weight: 500;\n      }\n\n      .opencode-select-btn:hover {\n        background: #e5e7eb;\n        color: #374151;\n      }\n\n      .opencode-select-btn.active {\n        background: #3b82f6;\n        color: white;\n      }\n\n      .opencode-selected-nodes {\n        flex: 1;\n        display: flex;\n        flex-direction: column;\n        padding: 8px;\n        gap: 6px;\n        overflow-y: auto;\n        overflow-x: hidden;\n      }\n\n      .opencode-selected-nodes:empty::before {\n        content: '暂无选中元素';\n        color: #9ca3af;\n        font-size: 12px;\n        text-align: center;\n        padding: 20px 10px;\n      }\n\n      .opencode-selected-node {\n        display: flex;\n        align-items: center;\n        gap: 8px;\n        padding: 8px 10px;\n        background: white;\n        border: 1px solid #e5e7eb;\n        border-radius: 6px;\n        font-size: 12px;\n        transition: all 0.2s;\n      }\n\n      .opencode-selected-node:hover {\n        border-color: #3b82f6;\n        box-shadow: 0 2px 4px rgba(59, 130, 246, 0.1);\n      }\n\n      .opencode-node-content {\n        flex: 1;\n        min-width: 0;\n        display: flex;\n        flex-direction: column;\n        gap: 2px;\n      }\n\n      .opencode-node-text {\n        color: #374151;\n        font-weight: 500;\n        overflow: hidden;\n        text-overflow: ellipsis;\n        white-space: nowrap;\n      }\n\n      .opencode-node-file {\n        color: #9ca3af;\n        font-size: 11px;\n        overflow: hidden;\n        text-overflow: ellipsis;\n        white-space: nowrap;\n      }\n\n      .opencode-node-remove {\n        width: 18px;\n        height: 18px;\n        border-radius: 4px;\n        border: none;\n        background: transparent;\n        color: #9ca3af;\n        cursor: pointer;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        font-size: 14px;\n        transition: all 0.2s;\n        flex-shrink: 0;\n      }\n\n      .opencode-node-remove:hover {\n        background: #ef4444;\n        color: white;\n      }\n\n      .opencode-dark .opencode-right-toolbar {\n        background: #111827;\n        border-left-color: #374151;\n      }\n\n      .opencode-dark .opencode-select-btn {\n        color: #9ca3af;\n        border-bottom-color: #374151;\n      }\n\n      .opencode-dark .opencode-select-btn:hover {\n        background: #374151;\n        color: #f3f4f6;\n      }\n\n      .opencode-dark .opencode-selected-nodes:empty::before {\n        color: #6b7280;\n      }\n\n      .opencode-dark .opencode-selected-node {\n        background: #1f2937;\n        border-color: #374151;\n      }\n\n      .opencode-dark .opencode-selected-node:hover {\n        border-color: #3b82f6;\n      }\n\n      .opencode-dark .opencode-node-text {\n        color: #f3f4f6;\n      }\n\n      .opencode-dark .opencode-node-file {\n        color: #6b7280;\n      }\n\n      .opencode-dark .opencode-node-remove {\n        color: #6b7280;\n      }\n\n      .opencode-dark .opencode-node-remove:hover {\n        background: #ef4444;\n        color: white;\n      }\n\n      .opencode-dark .opencode-button {\n        background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);\n        box-shadow: 0 4px 15px rgba(102, 126, 234, 0.5);\n      }\n\n      .opencode-notification {\n        position: absolute;\n        top: 20px;\n        left: 50%;\n        transform: translateX(-50%);\n        padding: 12px 20px;\n        background: #10b981;\n        color: white;\n        border-radius: 8px;\n        font-size: 14px;\n        box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\n        animation: slideDown 0.3s ease;\n        z-index: 10;\n      }\n\n      @keyframes slideDown {\n        from {\n          transform: translateX(-50%) translateY(-100%);\n          opacity: 0;\n        }\n        to {\n          transform: translateX(-50%) translateY(0);\n          opacity: 1;\n        }\n      }\n\n      .opencode-select-mode-hint {\n        position: fixed;\n        top: 20px;\n        left: 50%;\n        transform: translateX(-50%);\n        padding: 10px 16px;\n        background: #3b82f6;\n        color: white;\n        border-radius: 8px;\n        font-size: 13px;\n        box-shadow: 0 4px 12px rgba(59, 130, 246, 0.3);\n        z-index: 9999999;\n        display: none;\n        align-items: center;\n        gap: 12px;\n      }\n\n      .opencode-select-mode-hint.visible {\n        display: flex;\n        animation: slideDown 0.3s ease;\n      }\n\n      .opencode-hint-shortcut {\n        padding: 2px 6px;\n        background: rgba(255, 255, 255, 0.2);\n        border-radius: 4px;\n        font-size: 12px;\n      }\n\n      .opencode-selected-bubbles {\n        position: absolute;\n        bottom: 70px;\n        right: 0;\n        display: none;\n        flex-direction: column;\n        gap: 6px;\n        max-width: 220px;\n        max-height: 300px;\n        overflow-y: auto;\n      }\n\n      .opencode-selected-bubbles.visible {\n        display: flex;\n      }\n\n      .opencode-selected-bubble {\n        display: flex;\n        flex-direction: column;\n        gap: 2px;\n        padding: 8px 10px;\n        background: white;\n        border: 1px solid #e5e7eb;\n        border-radius: 8px;\n        font-size: 12px;\n        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);\n        position: relative;\n      }\n\n      .opencode-bubble-text {\n        color: #374151;\n        font-weight: 500;\n        overflow: hidden;\n        text-overflow: ellipsis;\n        white-space: nowrap;\n      }\n\n      .opencode-bubble-file {\n        color: #9ca3af;\n        font-size: 11px;\n        overflow: hidden;\n        text-overflow: ellipsis;\n        white-space: nowrap;\n      }\n\n      .opencode-bubble-remove {\n        position: absolute;\n        top: 4px;\n        right: 4px;\n        width: 16px;\n        height: 16px;\n        border-radius: 50%;\n        border: none;\n        background: transparent;\n        color: #9ca3af;\n        cursor: pointer;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        font-size: 12px;\n        transition: all 0.2s;\n      }\n\n      .opencode-bubble-remove:hover {\n        background: #ef4444;\n        color: white;\n      }\n\n      .opencode-bubble-empty {\n        padding: 8px 12px;\n        background: white;\n        border: 1px dashed #d1d5db;\n        border-radius: 8px;\n        color: #9ca3af;\n        font-size: 12px;\n        text-align: center;\n      }\n\n      .opencode-dark .opencode-selected-bubble {\n        background: #1f2937;\n        border-color: #374151;\n      }\n\n      .opencode-dark .opencode-bubble-text {\n        color: #f3f4f6;\n      }\n\n      .opencode-dark .opencode-bubble-file {\n        color: #6b7280;\n      }\n\n      .opencode-dark .opencode-bubble-remove {\n        color: #6b7280;\n      }\n\n      .opencode-dark .opencode-bubble-empty {\n        background: #1f2937;\n        border-color: #374151;\n        color: #6b7280;\n      }\n\n      @media (max-width: 768px) {\n        .opencode-chat {\n          width: calc(100vw - 40px);\n          height: calc(100vh - 120px);\n        }\n      }\n    `\n  }\n\n  /**\n   * 自动初始化挂件\n   */\n  function autoInit() {\n    const script = document.currentScript || document.querySelector('script[data-opencode-config]')\n    if (script) {\n      const configBase64 = script.getAttribute('data-opencode-config')\n      if (configBase64) {\n        try {\n          const config = JSON.parse(atob(configBase64))\n          if (document.readyState === 'loading') {\n            document.addEventListener('DOMContentLoaded', function() {\n              initOpenCodeWidget(config)\n            })\n          } else {\n            initOpenCodeWidget(config)\n          }\n        } catch (e) {\n          console.error('[OpenCode Widget] Failed to parse config:', e)\n        }\n      }\n    }\n  }\n\n  // 导出全局初始化函数\n  window.initOpenCodeWidget = initOpenCodeWidget\n\n  // 自动初始化\n  autoInit()\n})();"]}