adhdev 0.2.0 → 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (145) hide show
  1. package/dist/cli-entrypoint.js +18710 -1198
  2. package/dist/cli-entrypoint.js.map +1 -1
  3. package/dist/index.js +18720 -1228
  4. package/dist/index.js.map +1 -1
  5. package/package.json +2 -2
  6. package/providers/_builtin/CONTRIBUTING.md +0 -141
  7. package/providers/_builtin/README.md +0 -51
  8. package/providers/_builtin/_helpers/index.js +0 -188
  9. package/providers/_builtin/acp/agentpool/provider.json +0 -47
  10. package/providers/_builtin/acp/amp/provider.json +0 -45
  11. package/providers/_builtin/acp/auggie/provider.json +0 -50
  12. package/providers/_builtin/acp/autodev/provider.json +0 -47
  13. package/providers/_builtin/acp/autohand/provider.json +0 -45
  14. package/providers/_builtin/acp/blackbox-ai/provider.json +0 -47
  15. package/providers/_builtin/acp/claude-agent/provider.json +0 -50
  16. package/providers/_builtin/acp/cline-acp/provider.json +0 -47
  17. package/providers/_builtin/acp/code-assistant/provider.json +0 -47
  18. package/providers/_builtin/acp/codebuddy/provider.json +0 -47
  19. package/providers/_builtin/acp/codex-cli/provider.json +0 -50
  20. package/providers/_builtin/acp/corust-agent/provider.json +0 -45
  21. package/providers/_builtin/acp/crow-cli/provider.json +0 -47
  22. package/providers/_builtin/acp/cursor-acp/provider.json +0 -47
  23. package/providers/_builtin/acp/deepagents/provider.json +0 -45
  24. package/providers/_builtin/acp/dimcode/provider.json +0 -47
  25. package/providers/_builtin/acp/docker-cagent/provider.json +0 -50
  26. package/providers/_builtin/acp/factory-droid/provider.json +0 -53
  27. package/providers/_builtin/acp/fast-agent/provider.json +0 -45
  28. package/providers/_builtin/acp/fount/provider.json +0 -47
  29. package/providers/_builtin/acp/gemini-cli/provider.json +0 -107
  30. package/providers/_builtin/acp/github-copilot/provider.json +0 -47
  31. package/providers/_builtin/acp/goose/provider.json +0 -50
  32. package/providers/_builtin/acp/junie/provider.json +0 -45
  33. package/providers/_builtin/acp/kilo/provider.json +0 -45
  34. package/providers/_builtin/acp/kimi-cli/provider.json +0 -50
  35. package/providers/_builtin/acp/kiro-cli/provider.json +0 -47
  36. package/providers/_builtin/acp/minion-code/provider.json +0 -45
  37. package/providers/_builtin/acp/mistral-vibe/provider.json +0 -50
  38. package/providers/_builtin/acp/nova/provider.json +0 -47
  39. package/providers/_builtin/acp/openclaw/provider.json +0 -47
  40. package/providers/_builtin/acp/opencode/provider.json +0 -45
  41. package/providers/_builtin/acp/openhands/provider.json +0 -47
  42. package/providers/_builtin/acp/pi-acp/provider.json +0 -45
  43. package/providers/_builtin/acp/qoder/provider.json +0 -47
  44. package/providers/_builtin/acp/qwen-code/provider.json +0 -53
  45. package/providers/_builtin/acp/stakpak/provider.json +0 -47
  46. package/providers/_builtin/acp/vtcode/provider.json +0 -47
  47. package/providers/_builtin/cli/claude-cli/provider.json +0 -78
  48. package/providers/_builtin/cli/codex-cli/provider.json +0 -60
  49. package/providers/_builtin/cli/gemini-cli/provider.json +0 -64
  50. package/providers/_builtin/extension/cline/provider.json +0 -11
  51. package/providers/_builtin/extension/cline/scripts/focus_editor.js +0 -48
  52. package/providers/_builtin/extension/cline/scripts/list_chats.js +0 -100
  53. package/providers/_builtin/extension/cline/scripts/list_models.js +0 -43
  54. package/providers/_builtin/extension/cline/scripts/list_modes.js +0 -35
  55. package/providers/_builtin/extension/cline/scripts/new_session.js +0 -85
  56. package/providers/_builtin/extension/cline/scripts/open_panel.js +0 -25
  57. package/providers/_builtin/extension/cline/scripts/read_chat.js +0 -257
  58. package/providers/_builtin/extension/cline/scripts/resolve_action.js +0 -83
  59. package/providers/_builtin/extension/cline/scripts/send_message.js +0 -95
  60. package/providers/_builtin/extension/cline/scripts/set_mode.js +0 -36
  61. package/providers/_builtin/extension/cline/scripts/set_model.js +0 -36
  62. package/providers/_builtin/extension/cline/scripts/switch_session.js +0 -206
  63. package/providers/_builtin/extension/cline/scripts.js +0 -73
  64. package/providers/_builtin/extension/roo-code/provider.json +0 -11
  65. package/providers/_builtin/extension/roo-code/scripts.js +0 -659
  66. package/providers/_builtin/ide/antigravity/provider.json +0 -32
  67. package/providers/_builtin/ide/antigravity/scripts/focus_editor.js +0 -20
  68. package/providers/_builtin/ide/antigravity/scripts/list_chats.js +0 -137
  69. package/providers/_builtin/ide/antigravity/scripts/list_models.js +0 -38
  70. package/providers/_builtin/ide/antigravity/scripts/list_modes.js +0 -48
  71. package/providers/_builtin/ide/antigravity/scripts/new_session.js +0 -75
  72. package/providers/_builtin/ide/antigravity/scripts/read_chat.js +0 -262
  73. package/providers/_builtin/ide/antigravity/scripts/resolve_action.js +0 -68
  74. package/providers/_builtin/ide/antigravity/scripts/send_message.js +0 -56
  75. package/providers/_builtin/ide/antigravity/scripts/set_mode.js +0 -34
  76. package/providers/_builtin/ide/antigravity/scripts/set_model.js +0 -47
  77. package/providers/_builtin/ide/antigravity/scripts/switch_session.js +0 -114
  78. package/providers/_builtin/ide/antigravity/scripts.js +0 -73
  79. package/providers/_builtin/ide/cursor/provider.json +0 -35
  80. package/providers/_builtin/ide/cursor/scripts.js +0 -452
  81. package/providers/_builtin/ide/kiro/provider.json +0 -36
  82. package/providers/_builtin/ide/kiro/scripts/focus_editor.js +0 -20
  83. package/providers/_builtin/ide/kiro/scripts/open_panel.js +0 -47
  84. package/providers/_builtin/ide/kiro/scripts/resolve_action.js +0 -54
  85. package/providers/_builtin/ide/kiro/scripts/send_message.js +0 -29
  86. package/providers/_builtin/ide/kiro/scripts/webview_list_models.js +0 -39
  87. package/providers/_builtin/ide/kiro/scripts/webview_list_modes.js +0 -39
  88. package/providers/_builtin/ide/kiro/scripts/webview_list_sessions.js +0 -21
  89. package/providers/_builtin/ide/kiro/scripts/webview_new_session.js +0 -34
  90. package/providers/_builtin/ide/kiro/scripts/webview_read_chat.js +0 -68
  91. package/providers/_builtin/ide/kiro/scripts/webview_send_message.js +0 -72
  92. package/providers/_builtin/ide/kiro/scripts/webview_set_mode.js +0 -15
  93. package/providers/_builtin/ide/kiro/scripts/webview_set_model.js +0 -15
  94. package/providers/_builtin/ide/kiro/scripts/webview_switch_session.js +0 -26
  95. package/providers/_builtin/ide/kiro/scripts.js +0 -62
  96. package/providers/_builtin/ide/pearai/provider.json +0 -36
  97. package/providers/_builtin/ide/pearai/scripts/focus_editor.js +0 -20
  98. package/providers/_builtin/ide/pearai/scripts/list_sessions.js +0 -38
  99. package/providers/_builtin/ide/pearai/scripts/new_session.js +0 -55
  100. package/providers/_builtin/ide/pearai/scripts/open_panel.js +0 -46
  101. package/providers/_builtin/ide/pearai/scripts/resolve_action.js +0 -54
  102. package/providers/_builtin/ide/pearai/scripts/send_message.js +0 -29
  103. package/providers/_builtin/ide/pearai/scripts/webview_list_models.js +0 -43
  104. package/providers/_builtin/ide/pearai/scripts/webview_list_modes.js +0 -35
  105. package/providers/_builtin/ide/pearai/scripts/webview_list_sessions.js +0 -62
  106. package/providers/_builtin/ide/pearai/scripts/webview_new_session.js +0 -49
  107. package/providers/_builtin/ide/pearai/scripts/webview_read_chat.js +0 -92
  108. package/providers/_builtin/ide/pearai/scripts/webview_resolve_action.js +0 -59
  109. package/providers/_builtin/ide/pearai/scripts/webview_send_message.js +0 -72
  110. package/providers/_builtin/ide/pearai/scripts/webview_set_mode.js +0 -36
  111. package/providers/_builtin/ide/pearai/scripts/webview_set_model.js +0 -36
  112. package/providers/_builtin/ide/pearai/scripts/webview_switch_session.js +0 -34
  113. package/providers/_builtin/ide/pearai/scripts.js +0 -74
  114. package/providers/_builtin/ide/trae/provider.json +0 -35
  115. package/providers/_builtin/ide/trae/scripts/focus_editor.js +0 -20
  116. package/providers/_builtin/ide/trae/scripts/list_chats.js +0 -24
  117. package/providers/_builtin/ide/trae/scripts/list_models.js +0 -39
  118. package/providers/_builtin/ide/trae/scripts/list_modes.js +0 -39
  119. package/providers/_builtin/ide/trae/scripts/new_session.js +0 -30
  120. package/providers/_builtin/ide/trae/scripts/open_panel.js +0 -44
  121. package/providers/_builtin/ide/trae/scripts/read_chat.js +0 -113
  122. package/providers/_builtin/ide/trae/scripts/resolve_action.js +0 -54
  123. package/providers/_builtin/ide/trae/scripts/send_message.js +0 -69
  124. package/providers/_builtin/ide/trae/scripts/set_mode.js +0 -15
  125. package/providers/_builtin/ide/trae/scripts/set_model.js +0 -15
  126. package/providers/_builtin/ide/trae/scripts/switch_session.js +0 -23
  127. package/providers/_builtin/ide/trae/scripts.js +0 -57
  128. package/providers/_builtin/ide/vscode/provider.json +0 -33
  129. package/providers/_builtin/ide/vscode-insiders/provider.json +0 -31
  130. package/providers/_builtin/ide/vscodium/provider.json +0 -32
  131. package/providers/_builtin/ide/windsurf/provider.json +0 -22
  132. package/providers/_builtin/ide/windsurf/scripts/focus_editor.js +0 -30
  133. package/providers/_builtin/ide/windsurf/scripts/list_chats.js +0 -117
  134. package/providers/_builtin/ide/windsurf/scripts/list_models.js +0 -39
  135. package/providers/_builtin/ide/windsurf/scripts/list_modes.js +0 -39
  136. package/providers/_builtin/ide/windsurf/scripts/new_session.js +0 -69
  137. package/providers/_builtin/ide/windsurf/scripts/open_panel.js +0 -58
  138. package/providers/_builtin/ide/windsurf/scripts/read_chat.js +0 -297
  139. package/providers/_builtin/ide/windsurf/scripts/resolve_action.js +0 -68
  140. package/providers/_builtin/ide/windsurf/scripts/send_message.js +0 -87
  141. package/providers/_builtin/ide/windsurf/scripts/set_mode.js +0 -15
  142. package/providers/_builtin/ide/windsurf/scripts/set_model.js +0 -15
  143. package/providers/_builtin/ide/windsurf/scripts/switch_session.js +0 -58
  144. package/providers/_builtin/ide/windsurf/scripts.js +0 -57
  145. package/providers/_builtin/validate.js +0 -156
@@ -1,43 +0,0 @@
1
- /**
2
- * Cline — list_models
3
- * 드롭다운에서 사용 가능한 모델 목록 + 현재 선택된 모델 반환
4
- * → { models: string[], current: string }
5
- */
6
- (async () => {
7
- try {
8
- const inner = document.querySelector('iframe');
9
- const doc = inner?.contentDocument || inner?.contentWindow?.document;
10
- if (!doc) return JSON.stringify({ models: [], current: '', error: 'no doc' });
11
-
12
- // 현재 모델: mode-switch 또는 model selector에서 읽기
13
- let current = '';
14
- const modeSwitch = doc.querySelector('[data-testid="mode-switch"]');
15
- if (modeSwitch) current = (modeSwitch.textContent || '').trim();
16
- if (!current) {
17
- const modelSel = doc.querySelector('[data-testid*="model"], [aria-label*="model" i]');
18
- if (modelSel) current = (modelSel.textContent || '').trim();
19
- }
20
-
21
- // 드롭다운 트리거 찾기
22
- const trigger = doc.querySelector('[data-testid="model-selector"], [data-testid*="model-dropdown"], [data-testid="dropdown-trigger"]');
23
- if (!trigger) return JSON.stringify({ models: [], current, error: 'no model trigger' });
24
-
25
- // 드롭다운 열기
26
- trigger.click();
27
- await new Promise(r => setTimeout(r, 300));
28
-
29
- // 옵션 수집
30
- const options = doc.querySelectorAll('[data-testid*="dropdown-option"], [role="option"], [class*="dropdown"] [class*="item"], [class*="listbox"] [class*="option"]');
31
- const models = [];
32
- for (const opt of options) {
33
- const text = (opt.textContent || '').trim();
34
- if (text && text.length > 1 && text.length < 100) models.push(text);
35
- }
36
-
37
- // 닫기
38
- doc.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', bubbles: true }));
39
- if (trigger.click) trigger.click(); // fallback close
40
-
41
- return JSON.stringify({ models: [...new Set(models)], current });
42
- } catch (e) { return JSON.stringify({ models: [], current: '', error: e.message }); }
43
- })()
@@ -1,35 +0,0 @@
1
- /**
2
- * Cline — list_modes
3
- * 모드 selector에서 사용 가능한 모드 목록 + 현재 모드 반환
4
- * → { modes: string[], current: string }
5
- */
6
- (async () => {
7
- try {
8
- const inner = document.querySelector('iframe');
9
- const doc = inner?.contentDocument || inner?.contentWindow?.document;
10
- if (!doc) return JSON.stringify({ modes: [], current: '', error: 'no doc' });
11
-
12
- // 현재 모드
13
- const trigger = doc.querySelector('[data-testid="mode-selector-trigger"], [data-testid="mode-switch"]');
14
- if (!trigger) return JSON.stringify({ modes: [], current: '', error: 'no mode trigger' });
15
- const current = (trigger.textContent || '').trim();
16
-
17
- // 드롭다운 열기
18
- trigger.click();
19
- await new Promise(r => setTimeout(r, 300));
20
-
21
- // 옵션 수집
22
- const options = doc.querySelectorAll('[data-testid*="mode-option"], [role="option"], [class*="dropdown"] [class*="item"], [class*="listbox"] [class*="option"]');
23
- const modes = [];
24
- for (const opt of options) {
25
- const text = (opt.textContent || '').trim();
26
- if (text && text.length > 1 && text.length < 50) modes.push(text);
27
- }
28
-
29
- // 닫기
30
- doc.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', bubbles: true }));
31
- if (trigger.click) trigger.click(); // fallback close
32
-
33
- return JSON.stringify({ modes: [...new Set(modes)], current });
34
- } catch (e) { return JSON.stringify({ modes: [], current: '', error: e.message }); }
35
- })()
@@ -1,62 +0,0 @@
1
- /**
2
- * PearAI — webview_list_sessions (webview iframe 내부에서 실행)
3
- *
4
- * PearAI(Roo Code/Cline 기반) 히스토리 뷰에서 세션 목록을 추출.
5
- * 각 항목은 data-testid="task-item-{UUID}" 로 식별됨.
6
- */
7
- (() => {
8
- try {
9
- // ─── 히스토리 항목: data-testid="task-item-*" ───
10
- const taskItems = document.querySelectorAll('[data-testid^="task-item-"]');
11
-
12
- if (taskItems.length > 0) {
13
- const sessions = [];
14
- for (let i = 0; i < taskItems.length; i++) {
15
- const item = taskItems[i];
16
- if (!item) continue;
17
-
18
- const testId = item.getAttribute('data-testid') || '';
19
- const taskId = testId.replace('task-item-', '');
20
-
21
- // 전체 텍스트에서 제목 추출
22
- const fullText = (item.textContent || '').trim();
23
-
24
- // 구조: "MARCH 16, 1:31 AM 173 B test123 Tokens:10.4k 229"
25
- // 전략: Tokens: 앞의 마지막 의미있는 텍스트를 제목으로 사용
26
- let title = '';
27
- let date = '';
28
-
29
- // 날짜 추출
30
- const dateMatch = fullText.match(/^([A-Z]+\s+\d+,\s*\d+:\d+\s*[AP]M)/);
31
- if (dateMatch) date = dateMatch[1];
32
-
33
- // 제목 추출: 날짜와 크기(B/kB) 뒤, Tokens: 앞
34
- const titleMatch = fullText.match(/[AP]M[\d.\s]*[kMG]?B\s*(.*?)(?:Tokens:|$)/s);
35
- if (titleMatch && titleMatch[1]) {
36
- title = titleMatch[1].trim();
37
- }
38
-
39
- // fallback: 첫 80자
40
- if (!title) {
41
- title = fullText.substring(0, 80);
42
- }
43
-
44
- sessions.push({
45
- id: taskId,
46
- title: title.substring(0, 100),
47
- date: date,
48
- active: false
49
- });
50
- }
51
- return JSON.stringify({ sessions: sessions });
52
- }
53
-
54
- // ─── 히스토리 뷰가 열려 있지 않음 ───
55
- return JSON.stringify({
56
- sessions: [],
57
- note: 'History view not open. Toggle history first.'
58
- });
59
- } catch (e) {
60
- return JSON.stringify({ sessions: [], error: String(e.message || e) });
61
- }
62
- })()
@@ -1,49 +0,0 @@
1
- /**
2
- * PearAI — webview_new_session (webview iframe 내부에서 실행)
3
- *
4
- * Roo Code/Cline 기반 PearAI에서 새 세션(태스크)을 시작합니다.
5
- * 1) "New Task" 텍스트를 가진 버튼 클릭 시도
6
- * 2) 실패 시, vscode postMessage API로 newTask 명령 전송 시도
7
- * 3) 실패 시, chat-text-area를 초기화하여 새 세션 상태로 진입
8
- */
9
- (() => {
10
- try {
11
- // Strategy 1: Find "New Task" button text
12
- const buttons = document.querySelectorAll('button, [role="button"]');
13
- for (const btn of buttons) {
14
- const text = (btn.textContent || '').trim();
15
- if (text.includes('Start New Task') || text.includes('New Task') || text.includes('New Chat')) {
16
- btn.click();
17
- return JSON.stringify({ created: true, method: 'button', text });
18
- }
19
- }
20
-
21
- // Strategy 2: Look for a "+" (plus/add) button that might create new tasks
22
- for (const btn of buttons) {
23
- const title = btn.getAttribute('title') || '';
24
- const ariaLabel = btn.getAttribute('aria-label') || '';
25
- if (title.includes('New') || ariaLabel.includes('New') ||
26
- title.includes('new') || ariaLabel.includes('new')) {
27
- btn.click();
28
- return JSON.stringify({ created: true, method: 'title', title: title || ariaLabel });
29
- }
30
- }
31
-
32
- // Strategy 3: Use vscode API to post message for new task
33
- if (typeof acquireVsCodeApi !== 'undefined') {
34
- const vscode = acquireVsCodeApi();
35
- vscode.postMessage({ type: 'newTask' });
36
- return JSON.stringify({ created: true, method: 'vscodeApi' });
37
- }
38
-
39
- // Strategy 4: If already in empty state (no messages), report as already new
40
- const messages = document.querySelectorAll('[class*="message"], [class*="chat-row"]');
41
- if (messages.length === 0) {
42
- return JSON.stringify({ created: true, method: 'alreadyNew', note: 'No messages - already in new session state' });
43
- }
44
-
45
- return JSON.stringify({ created: false, error: 'New Task button not found' });
46
- } catch (e) {
47
- return JSON.stringify({ created: false, error: e.message });
48
- }
49
- })()
@@ -1,92 +0,0 @@
1
- /**
2
- * PearAI — webview_read_chat (webview iframe 내부에서 실행)
3
- *
4
- * PearAI는 Roo Code / Cline 기반의 Agent 패널을 사용.
5
- * React로 렌더링된 DOM에서 메시지를 파싱.
6
- *
7
- * DOM 단서:
8
- * - textarea.chat-text-area: 입력 필드
9
- * - Task: ... : 유저 메시지
10
- * - API Request 등: 시스템 메시지
11
- *
12
- * 반환: ReadChatResult { id, status, messages, inputContent? }
13
- */
14
- (() => {
15
- try {
16
- const messages = [];
17
-
18
- // 메시지 블록 찾기 — 다양한 셀렉터 시도
19
- // Cline/Roo Code 기반이므로 유사한 구조
20
- const allText = document.body?.textContent || '';
21
-
22
- // Task 헤더에서 유저 메시지
23
- const taskEl = document.querySelector('[class*="task-header"], [class*="TaskHeader"]');
24
- if (taskEl) {
25
- const taskText = taskEl.textContent?.trim() || '';
26
- if (taskText) {
27
- messages.push({ role: 'user', content: taskText.replace(/^Task:\s*/i, ''), index: 0 });
28
- }
29
- }
30
-
31
- // 채팅 메시지 영역 파싱 — 공통 클래스
32
- const chatMessages = document.querySelectorAll('[class*="chat-row"], [class*="ChatRow"], [class*="message-row"], [class*="MessageRow"]');
33
- chatMessages.forEach((msg, idx) => {
34
- const text = msg.textContent?.trim() || '';
35
- if (!text) return;
36
-
37
- // role 판정 — 클래스명이나 아이콘으로
38
- const isUser = msg.querySelector('[class*="user"]') != null || msg.classList.toString().includes('user');
39
- messages.push({
40
- role: isUser ? 'user' : 'assistant',
41
- content: text.substring(0, 2000),
42
- index: idx,
43
- });
44
- });
45
-
46
- // 메시지가 없으면 전체 텍스트에서 추출
47
- if (messages.length === 0 && allText.length > 50) {
48
- // Task: 패턴 찾기
49
- const taskMatch = allText.match(/Task:\s*(.+?)(?:\d+[\d.,]*k|\d+\s*$)/s);
50
- if (taskMatch) {
51
- messages.push({ role: 'user', content: taskMatch[1].trim().substring(0, 500), index: 0 });
52
- }
53
- // AI 응답 찾기 (간략화)
54
- const bodyContent = allText.substring(0, 2000);
55
- if (bodyContent.includes('API Request') || bodyContent.includes('Initial Checkpoint')) {
56
- // 응답이 있음을 표시
57
- messages.push({ role: 'assistant', content: '(response in progress or completed)', index: 1 });
58
- }
59
- }
60
-
61
- // 상태 감지
62
- let status = 'idle';
63
- const cancelBtn = document.querySelector('button[class*="cancel"], [aria-label*="cancel" i]');
64
- if (cancelBtn && cancelBtn.offsetWidth > 0 && cancelBtn.textContent?.toLowerCase().includes('cancel')) {
65
- status = 'generating';
66
- }
67
-
68
- // API Request... (로딩) → generating
69
- const loadingEl = document.querySelector('[class*="loading"], [class*="spinner"]');
70
- if (loadingEl && loadingEl.offsetWidth > 0) {
71
- status = 'generating';
72
- }
73
-
74
- // "Retrying" → error/generating
75
- if (allText.includes('Retrying in') || allText.includes('Retry attempt')) {
76
- status = 'generating';
77
- }
78
-
79
- // 입력 필드
80
- const textarea = document.querySelector('textarea[placeholder*="Type"], textarea.chat-text-area, textarea');
81
- const inputContent = textarea ? textarea.value || textarea.textContent?.trim() : '';
82
-
83
- return JSON.stringify({
84
- id: 'pearai-agent',
85
- status,
86
- messages,
87
- inputContent: inputContent || undefined,
88
- });
89
- } catch (e) {
90
- return JSON.stringify({ id: '', status: 'error', messages: [], error: e.message });
91
- }
92
- })()
@@ -1,59 +0,0 @@
1
- /**
2
- * PearAI — webview_resolve_action
3
- *
4
- * PearAI Agent(Roo Code 기반)는 승인/거절 버튼을 webview 내부에 렌더링함.
5
- * 버튼을 찾아 직접 click() 이벤트 발생.
6
- *
7
- * 파라미터: ${ BUTTON_TEXT }
8
- */
9
- (() => {
10
- try {
11
- const want = ${ BUTTON_TEXT };
12
- const wantNorm = (want || '').replace(/\s+/g, ' ').trim().toLowerCase();
13
-
14
- function norm(t) { return (t || '').replace(/\s+/g, ' ').trim().toLowerCase(); }
15
-
16
- function matches(el) {
17
- const t = norm(el.textContent);
18
- if (!t || t.length > 80) return false;
19
- if (t === wantNorm) return true;
20
- if (t.indexOf(wantNorm) === 0) return true;
21
- if (wantNorm.indexOf(t) >= 0 && t.length > 2) return true;
22
- if (/^(run|approve|allow|accept|yes)\b/.test(wantNorm)) {
23
- if (/^(run|allow|accept|approve)\b/.test(t)) return true;
24
- }
25
- if (/^(reject|deny|no|abort)\b/.test(wantNorm)) {
26
- if (/^(reject|deny)\b/.test(t)) return true;
27
- }
28
- return false;
29
- }
30
-
31
- const sel = 'button, [role="button"], .vsc-button';
32
- const allBtns = [...document.querySelectorAll(sel)].filter(b => b.offsetWidth > 0 && b.offsetHeight > 0);
33
-
34
- let found = null;
35
- for (let i = allBtns.length - 1; i >= 0; i--) {
36
- if (matches(allBtns[i])) { found = allBtns[i]; break; }
37
- }
38
-
39
- if (found) {
40
- found.focus?.();
41
- const rect = found.getBoundingClientRect();
42
- const x = rect.left + rect.width / 2;
43
- const y = rect.top + rect.height / 2;
44
- for (const type of ['pointerdown', 'mousedown', 'pointerup', 'mouseup', 'click']) {
45
- found.dispatchEvent(new PointerEvent(type, {
46
- bubbles: true, cancelable: true, view: window,
47
- clientX: x, clientY: y, pointerId: 1, pointerType: 'mouse'
48
- }));
49
- }
50
- return JSON.stringify({
51
- resolved: true,
52
- clicked: found.textContent || wantNorm
53
- });
54
- }
55
- return JSON.stringify({ resolved: false, error: 'Button not found: ' + wantNorm });
56
- } catch (e) {
57
- return JSON.stringify({ resolved: false, error: e.message });
58
- }
59
- })()
@@ -1,72 +0,0 @@
1
- /**
2
- * Kiro — webview_send_message (webview iframe 내부에서 실행)
3
- *
4
- * Kiro의 채팅 입력은 webview iframe 안의 ProseMirror/tiptap 에디터.
5
- * execCommand('insertText') + Enter 키 이벤트로 메시지 전송.
6
- *
7
- * 파라미터: ${ MESSAGE }
8
- */
9
- (async () => {
10
- try {
11
- const msg = ${ MESSAGE };
12
-
13
- // ─── 1. 입력 필드 찾기 ───
14
- const editor =
15
- document.querySelector('.tiptap.ProseMirror') ||
16
- document.querySelector('[contenteditable="true"]') ||
17
- document.querySelector('textarea');
18
-
19
- if (!editor) return JSON.stringify({ sent: false, error: 'no input found in webview' });
20
-
21
- const isTextarea = editor.tagName === 'TEXTAREA';
22
-
23
- if (isTextarea) {
24
- editor.focus();
25
- const nativeSetter = Object.getOwnPropertyDescriptor(HTMLTextAreaElement.prototype, 'value')?.set;
26
- if (nativeSetter) nativeSetter.call(editor, msg);
27
- else editor.value = msg;
28
- editor.dispatchEvent(new Event('input', { bubbles: true }));
29
- await new Promise(r => setTimeout(r, 300));
30
-
31
- const enterOpts = { key: 'Enter', code: 'Enter', keyCode: 13, which: 13, bubbles: true, cancelable: true, composed: true };
32
- editor.dispatchEvent(new KeyboardEvent('keydown', enterOpts));
33
- editor.dispatchEvent(new KeyboardEvent('keypress', enterOpts));
34
- editor.dispatchEvent(new KeyboardEvent('keyup', enterOpts));
35
- return JSON.stringify({ sent: true });
36
- }
37
-
38
- // ─── 2. contenteditable (ProseMirror / tiptap) ───
39
- editor.focus();
40
- await new Promise(r => setTimeout(r, 100));
41
-
42
- // 전체 선택 + 삭제 + 삽입
43
- const sel = window.getSelection();
44
- const range = document.createRange();
45
- range.selectNodeContents(editor);
46
- sel.removeAllRanges();
47
- sel.addRange(range);
48
- await new Promise(r => setTimeout(r, 50));
49
-
50
- document.execCommand('delete', false, null);
51
- await new Promise(r => setTimeout(r, 50));
52
- document.execCommand('insertText', false, msg);
53
-
54
- editor.dispatchEvent(new Event('input', { bubbles: true }));
55
- await new Promise(r => setTimeout(r, 400));
56
-
57
- // ─── 3. Enter 키 전송 ───
58
- const enterOpts = {
59
- key: 'Enter', code: 'Enter',
60
- keyCode: 13, which: 13,
61
- bubbles: true, cancelable: true, composed: true,
62
- };
63
- editor.dispatchEvent(new KeyboardEvent('keydown', enterOpts));
64
- await new Promise(r => setTimeout(r, 50));
65
- editor.dispatchEvent(new KeyboardEvent('keypress', enterOpts));
66
- editor.dispatchEvent(new KeyboardEvent('keyup', enterOpts));
67
-
68
- return JSON.stringify({ sent: true });
69
- } catch (e) {
70
- return JSON.stringify({ sent: false, error: e.message });
71
- }
72
- })()
@@ -1,36 +0,0 @@
1
- /**
2
- * Cline — set_mode
3
- * 모드 selector에서 지정된 모드 선택
4
- * ${MODE} → JSON.stringify(modeName)
5
- * → { success: boolean }
6
- */
7
- (async () => {
8
- try {
9
- const inner = document.querySelector('iframe');
10
- const doc = inner?.contentDocument || inner?.contentWindow?.document;
11
- if (!doc) return JSON.stringify({ success: false, error: 'no doc' });
12
-
13
- const target = ${MODE};
14
- const trigger = doc.querySelector('[data-testid="mode-selector-trigger"], [data-testid="mode-switch"]');
15
- if (!trigger) return JSON.stringify({ success: false, error: 'no mode trigger' });
16
-
17
- // 드롭다운 열기
18
- trigger.click();
19
- await new Promise(r => setTimeout(r, 300));
20
-
21
- // 옵션에서 타겟 모드 찾기
22
- const options = doc.querySelectorAll('[data-testid*="mode-option"], [role="option"], [class*="dropdown"] [class*="item"], [class*="listbox"] [class*="option"]');
23
- for (const opt of options) {
24
- const text = (opt.textContent || '').trim();
25
- if (text === target || text.toLowerCase().includes(target.toLowerCase())) {
26
- opt.click();
27
- await new Promise(r => setTimeout(r, 200));
28
- return JSON.stringify({ success: true, mode: text });
29
- }
30
- }
31
-
32
- // 닫기
33
- doc.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', bubbles: true }));
34
- return JSON.stringify({ success: false, error: 'mode not found: ' + target });
35
- } catch (e) { return JSON.stringify({ success: false, error: e.message }); }
36
- })()
@@ -1,36 +0,0 @@
1
- /**
2
- * Cline — set_model
3
- * 드롭다운에서 지정된 모델 선택
4
- * ${MODEL} → JSON.stringify(modelName)
5
- * → { success: boolean }
6
- */
7
- (async () => {
8
- try {
9
- const inner = document.querySelector('iframe');
10
- const doc = inner?.contentDocument || inner?.contentWindow?.document;
11
- if (!doc) return JSON.stringify({ success: false, error: 'no doc' });
12
-
13
- const target = ${MODEL};
14
- const trigger = doc.querySelector('[data-testid="model-selector"], [data-testid*="model-dropdown"], [data-testid="dropdown-trigger"]');
15
- if (!trigger) return JSON.stringify({ success: false, error: 'no model trigger' });
16
-
17
- // 드롭다운 열기
18
- trigger.click();
19
- await new Promise(r => setTimeout(r, 300));
20
-
21
- // 옵션에서 타겟 모델 찾기
22
- const options = doc.querySelectorAll('[data-testid*="dropdown-option"], [role="option"], [class*="dropdown"] [class*="item"], [class*="listbox"] [class*="option"]');
23
- for (const opt of options) {
24
- const text = (opt.textContent || '').trim();
25
- if (text === target || text.includes(target)) {
26
- opt.click();
27
- await new Promise(r => setTimeout(r, 200));
28
- return JSON.stringify({ success: true, model: text });
29
- }
30
- }
31
-
32
- // 닫기
33
- doc.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', bubbles: true }));
34
- return JSON.stringify({ success: false, error: 'model not found: ' + target });
35
- } catch (e) { return JSON.stringify({ success: false, error: e.message }); }
36
- })()
@@ -1,34 +0,0 @@
1
- /**
2
- * PearAI — webview_switch_session (webview iframe 내부에서 실행)
3
- *
4
- * data-testid="task-item-{UUID}" 항목을 클릭하여 세션 전환.
5
- * SESSION_ID는 task UUID 형태 (예: "51a08aba-1078-410c-a601-0e859205b12c")
6
- */
7
- (() => {
8
- try {
9
- const targetId = ${SESSION_ID};
10
-
11
- // ─── UUID 기반 매칭 ───
12
- const selector = '[data-testid="task-item-' + targetId + '"]';
13
- const targetItem = document.querySelector(selector);
14
- if (targetItem) {
15
- targetItem.click();
16
- return JSON.stringify({ switched: true, id: targetId });
17
- }
18
-
19
- // ─── 인덱스 기반 fallback (숫자로 전달된 경우) ───
20
- const index = parseInt(targetId, 10);
21
- if (!isNaN(index)) {
22
- const allItems = document.querySelectorAll('[data-testid^="task-item-"]');
23
- if (allItems[index]) {
24
- allItems[index].click();
25
- const taskId = (allItems[index].getAttribute('data-testid') || '').replace('task-item-', '');
26
- return JSON.stringify({ switched: true, id: taskId, method: 'index' });
27
- }
28
- }
29
-
30
- return JSON.stringify({ switched: false, error: 'Task item not found: ' + targetId });
31
- } catch (e) {
32
- return JSON.stringify({ switched: false, error: e.message });
33
- }
34
- })()
@@ -1,74 +0,0 @@
1
- /**
2
- * CDP Scripts for PearAI
3
- */
4
-
5
- const fs = require('fs');
6
- const path = require('path');
7
- const SCRIPTS_DIR = path.join(__dirname, 'scripts');
8
- function loadScript(name) {
9
- try { return fs.readFileSync(path.join(SCRIPTS_DIR, name), 'utf8'); }
10
- catch { return null; }
11
- }
12
-
13
-
14
- module.exports.webviewReadChat = function webviewReadChat() { return loadScript('webview_read_chat.js'); };
15
-
16
- module.exports.webviewSendMessage = function webviewSendMessage(text) {
17
- const s = loadScript('webview_send_message.js');
18
- return s ? s.replace(/\$\{\s*MESSAGE\s*\}/g, JSON.stringify(text)) : null;
19
- };
20
-
21
- module.exports.sendMessage = function sendMessage(text) {
22
- const s = loadScript('send_message.js');
23
- return s ? s.replace(/\$\{\s*MESSAGE\s*\}/g, JSON.stringify(text)) : null;
24
- };
25
-
26
- module.exports.resolveAction = function resolveAction(params) {
27
- const action = typeof params === 'string' ? params : params?.action || 'approve';
28
- const buttonText = params?.button || params?.buttonText
29
- || (action === 'approve' ? 'Accept' : action === 'reject' ? 'Reject' : action);
30
- const s = loadScript('resolve_action.js');
31
- return s ? s.replace(/\$\{\s*BUTTON_TEXT\s*\}/g, JSON.stringify(buttonText)) : null;
32
- };
33
-
34
- module.exports.webviewResolveAction = function webviewResolveAction(params) {
35
- const action = typeof params === 'string' ? params : params?.action || 'approve';
36
- const buttonText = params?.button || params?.buttonText
37
- || (action === 'approve' ? 'Accept' : action === 'reject' ? 'Reject' : action);
38
- const s = loadScript('webview_resolve_action.js');
39
- return s ? s.replace(/\$\{\s*BUTTON_TEXT\s*\}/g, JSON.stringify(buttonText)) : null;
40
- };
41
-
42
- module.exports.openPanel = function openPanel() { return loadScript('open_panel.js'); };
43
-
44
- module.exports.focusEditor = function focusEditor() { return loadScript('focus_editor.js'); };
45
-
46
- module.exports.newSession = function newSession() { return loadScript('new_session.js'); };
47
-
48
- module.exports.listSessions = function listSessions() { return loadScript('list_sessions.js'); };
49
-
50
- module.exports.webviewNewSession = function webviewNewSession() { return loadScript('webview_new_session.js'); };
51
-
52
- module.exports.webviewListSessions = function webviewListSessions() { return loadScript('webview_list_sessions.js'); };
53
-
54
- module.exports.webviewSwitchSession = function webviewSwitchSession(sessionId) {
55
- const s = loadScript('webview_switch_session.js');
56
- return s ? s.replace(/\$\{\s*SESSION_ID\s*\}/g, JSON.stringify(sessionId)) : null;
57
- };
58
-
59
- module.exports.webviewListModels = function webviewListModels() { return loadScript('webview_list_models.js'); };
60
-
61
- module.exports.webviewSetModel = function webviewSetModel(params) {
62
- const model = typeof params === 'string' ? params : params?.model;
63
- const s = loadScript('webview_set_model.js');
64
- return s ? s.replace(/\$\{\s*MODEL\s*\}/g, JSON.stringify(model)) : null;
65
- };
66
-
67
- module.exports.webviewListModes = function webviewListModes() { return loadScript('webview_list_modes.js'); };
68
-
69
- module.exports.webviewSetMode = function webviewSetMode(params) {
70
- const mode = typeof params === 'string' ? params : params?.mode;
71
- const s = loadScript('webview_set_mode.js');
72
- return s ? s.replace(/\$\{\s*MODE\s*\}/g, JSON.stringify(mode)) : null;
73
- };
74
-
@@ -1,35 +0,0 @@
1
- {
2
- "type": "trae",
3
- "name": "Trae",
4
- "category": "ide",
5
- "displayName": "Trae",
6
- "icon": "🔮",
7
- "cli": "trae",
8
- "cdpPorts": [
9
- 9353,
10
- 9354
11
- ],
12
- "processNames": {
13
- "darwin": "Trae",
14
- "win32": [
15
- "Trae.exe"
16
- ],
17
- "linux": [
18
- "trae"
19
- ]
20
- },
21
- "paths": {
22
- "darwin": [
23
- "/Applications/Trae.app"
24
- ],
25
- "win32": [
26
- "C:\\Users\\*\\AppData\\Local\\Programs\\trae\\Trae.exe"
27
- ],
28
- "linux": [
29
- "/opt/trae",
30
- "/usr/share/trae"
31
- ]
32
- },
33
- "inputMethod": "cdp-type-and-send",
34
- "inputSelector": ".chat-input-v2-input-box-editable, [contenteditable=\"true\"][role=\"textbox\"]"
35
- }
@@ -1,20 +0,0 @@
1
- /**
2
- * Cursor v1 — focus_editor
3
- *
4
- * CURSOR.md 4-5: 셀렉터 우선순위
5
- * [contenteditable="true"][role="textbox"]
6
- * → .chat-input textarea
7
- * → .composer-input
8
- * → textarea
9
- *
10
- * 최종 확인: 2026-03-06
11
- */
12
- (() => {
13
- const editor = document.querySelector('[contenteditable="true"][role="textbox"]')
14
- || document.querySelector('.chat-input textarea')
15
- || document.querySelector('.composer-input')
16
- || document.querySelector('textarea.native-input')
17
- || document.querySelector('textarea');
18
- if (editor) { editor.focus(); return 'focused'; }
19
- return 'no editor found';
20
- })()