adhdev 0.1.42 → 0.1.44

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 (47) hide show
  1. package/dist/index.js +7 -0
  2. package/package.json +2 -1
  3. package/providers/_builtin/cli/claude-cli/provider.js +125 -0
  4. package/providers/_builtin/cli/codex-cli/provider.js +77 -0
  5. package/providers/_builtin/cli/gemini-cli/provider.js +121 -0
  6. package/providers/_builtin/extension/cline/provider.js +77 -0
  7. package/providers/_builtin/extension/cline/scripts/focus_editor.js +48 -0
  8. package/providers/_builtin/extension/cline/scripts/list_chats.js +100 -0
  9. package/providers/_builtin/extension/cline/scripts/new_session.js +85 -0
  10. package/providers/_builtin/extension/cline/scripts/open_panel.js +25 -0
  11. package/providers/_builtin/extension/cline/scripts/read_chat.js +257 -0
  12. package/providers/_builtin/extension/cline/scripts/resolve_action.js +83 -0
  13. package/providers/_builtin/extension/cline/scripts/send_message.js +95 -0
  14. package/providers/_builtin/extension/cline/scripts/switch_session.js +206 -0
  15. package/providers/_builtin/extension/roo-code/provider.js +578 -0
  16. package/providers/_builtin/ide/antigravity/provider.js +92 -0
  17. package/providers/_builtin/ide/antigravity/scripts/focus_editor.js +20 -0
  18. package/providers/_builtin/ide/antigravity/scripts/list_chats.js +137 -0
  19. package/providers/_builtin/ide/antigravity/scripts/new_session.js +75 -0
  20. package/providers/_builtin/ide/antigravity/scripts/read_chat.js +240 -0
  21. package/providers/_builtin/ide/antigravity/scripts/resolve_action.js +64 -0
  22. package/providers/_builtin/ide/antigravity/scripts/send_message.js +56 -0
  23. package/providers/_builtin/ide/antigravity/scripts/switch_session.js +114 -0
  24. package/providers/_builtin/ide/cursor/provider.js +242 -0
  25. package/providers/_builtin/ide/cursor/provider.js.backup +116 -0
  26. package/providers/_builtin/ide/cursor/provider.js.bak +127 -0
  27. package/providers/_builtin/ide/cursor/scripts_backup/focus_editor.js +20 -0
  28. package/providers/_builtin/ide/cursor/scripts_backup/list_chats.js +111 -0
  29. package/providers/_builtin/ide/cursor/scripts_backup/new_session.js +62 -0
  30. package/providers/_builtin/ide/cursor/scripts_backup/open_panel.js +31 -0
  31. package/providers/_builtin/ide/cursor/scripts_backup/read_chat.js +433 -0
  32. package/providers/_builtin/ide/cursor/scripts_backup/resolve_action.js +90 -0
  33. package/providers/_builtin/ide/cursor/scripts_backup/send_message.js +86 -0
  34. package/providers/_builtin/ide/cursor/scripts_backup/switch_session.js +63 -0
  35. package/providers/_builtin/ide/vscode/provider.js +36 -0
  36. package/providers/_builtin/ide/vscode-insiders/provider.js +27 -0
  37. package/providers/_builtin/ide/vscodium/provider.js +27 -0
  38. package/providers/_builtin/ide/windsurf/provider.js +64 -0
  39. package/providers/_builtin/ide/windsurf/scripts/focus_editor.js +30 -0
  40. package/providers/_builtin/ide/windsurf/scripts/list_chats.js +117 -0
  41. package/providers/_builtin/ide/windsurf/scripts/new_session.js +69 -0
  42. package/providers/_builtin/ide/windsurf/scripts/open_panel.js +58 -0
  43. package/providers/_builtin/ide/windsurf/scripts/read_chat.js +297 -0
  44. package/providers/_builtin/ide/windsurf/scripts/resolve_action.js +68 -0
  45. package/providers/_builtin/ide/windsurf/scripts/send_message.js +87 -0
  46. package/providers/_builtin/ide/windsurf/scripts/switch_session.js +58 -0
  47. package/providers/_helpers/index.js +188 -0
@@ -0,0 +1,137 @@
1
+ /**
2
+ * Antigravity v1 — list_chats (v3 — 강화된 셀렉터)
3
+ *
4
+ * 히스토리 토글을 클릭하여 대화 목록 패널을 열고,
5
+ * DOM에서 직접 대화 목록을 파싱한 뒤 패널을 닫고 결과를 반환.
6
+ *
7
+ * DOM 구조 (2026-03-03 확인):
8
+ * 토글: [data-past-conversations-toggle="true"]
9
+ * 패널: input[placeholder="Select a conversation"] 근처
10
+ * 섹션 헤더: .opacity-50 텍스트 (Current, Recent, Other)
11
+ * 행: .cursor-pointer.justify-between.rounded-md
12
+ * ├── 제목: span > span
13
+ * ├── 워크스페이스: .opacity-50.truncate > span
14
+ * └── 시간: .opacity-50.flex-shrink-0
15
+ */
16
+ (async () => {
17
+ const sleep = (ms) => new Promise(r => setTimeout(r, ms));
18
+
19
+ try {
20
+ // 1. 토글 클릭하여 히스토리 패널 열기
21
+ const toggle = document.querySelector('[data-past-conversations-toggle="true"]');
22
+ if (!toggle) return [];
23
+ toggle.click();
24
+ await sleep(1000);
25
+
26
+ // 2. 정확한 검색 input 찾기 (placeholder 정확 매칭)
27
+ const allInputs = document.querySelectorAll('input[type="text"]');
28
+ let searchInput = null;
29
+ for (const inp of allInputs) {
30
+ if (inp.placeholder === 'Select a conversation' && inp.offsetWidth > 0) {
31
+ searchInput = inp;
32
+ break;
33
+ }
34
+ }
35
+
36
+ if (!searchInput) {
37
+ document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', code: 'Escape', bubbles: true }));
38
+ return [];
39
+ }
40
+
41
+ // 3. "Current" 텍스트를 기준으로 대화 목록 스크롤 컨테이너 찾기
42
+ let container = null;
43
+ const walker = document.createTreeWalker(document.body, NodeFilter.SHOW_TEXT, null);
44
+ while (walker.nextNode()) {
45
+ if (walker.currentNode.textContent.trim() === 'Current') {
46
+ // "Current" 텍스트의 조상 — overflow-auto/scroll이 있는 스크롤 컨테이너까지 올라감
47
+ let el = walker.currentNode.parentElement;
48
+ for (let i = 0; i < 10 && el; i++) {
49
+ const cls = (el.className || '');
50
+ if (typeof cls === 'string' && (cls.includes('overflow-auto') || cls.includes('overflow-y-scroll'))) {
51
+ container = el;
52
+ break;
53
+ }
54
+ el = el.parentElement;
55
+ }
56
+ // overflow 컨테이너 못 찾으면 행이 가장 많은 조상 사용
57
+ if (!container) {
58
+ el = walker.currentNode.parentElement;
59
+ let bestEl = null;
60
+ let bestCount = 0;
61
+ for (let i = 0; i < 8 && el; i++) {
62
+ const rows = el.querySelectorAll('[class*="cursor-pointer"][class*="justify-between"][class*="rounded-md"]');
63
+ if (rows.length > bestCount) {
64
+ bestCount = rows.length;
65
+ bestEl = el;
66
+ }
67
+ el = el.parentElement;
68
+ }
69
+ container = bestEl;
70
+ }
71
+ if (container) break;
72
+ }
73
+ }
74
+
75
+ // 폴백: "Current" 텍스트가 없을 때 (새 대화 직후 등) searchInput 기반
76
+ if (!container && searchInput) {
77
+ let el = searchInput.parentElement;
78
+ for (let i = 0; i < 10 && el; i++) {
79
+ const cls = (el.className || '');
80
+ if (typeof cls === 'string' && (cls.includes('overflow-auto') || cls.includes('overflow-y-scroll'))) {
81
+ container = el; break;
82
+ }
83
+ const rows = el.querySelectorAll('[class*="cursor-pointer"][class*="justify-between"][class*="rounded-md"]');
84
+ if (rows.length > 0 && !container) {
85
+ container = el;
86
+ }
87
+ el = el.parentElement;
88
+ }
89
+ }
90
+
91
+ if (!container) {
92
+ document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', code: 'Escape', bubbles: true }));
93
+ return [];
94
+ }
95
+
96
+ // 4. 대화 행 파싱
97
+ const rows = container.querySelectorAll('[class*="cursor-pointer"][class*="justify-between"][class*="rounded-md"]');
98
+ const chats = [];
99
+
100
+ for (const row of rows) {
101
+ const titleEl = row.querySelector('span span');
102
+ const title = titleEl ? titleEl.textContent.trim() : '';
103
+ if (!title) continue;
104
+
105
+ const timeEl = row.querySelector('span[class*="opacity-50"][class*="flex-shrink-0"]');
106
+ const time = timeEl ? timeEl.textContent.trim() : '';
107
+
108
+ const wsEl = row.querySelector('span[class*="opacity-50"][class*="truncate"] span');
109
+ const workspace = wsEl ? wsEl.textContent.trim() : '';
110
+
111
+ const isCurrent = (row.className || '').includes('focusBackground');
112
+
113
+ let section = '';
114
+ const sectionHeader = row.parentElement?.querySelector('[class*="opacity-50"]:not([class*="cursor-pointer"])');
115
+ if (sectionHeader) {
116
+ section = sectionHeader.textContent.trim();
117
+ }
118
+
119
+ chats.push({
120
+ id: title,
121
+ title,
122
+ status: isCurrent ? 'current' : '',
123
+ time,
124
+ workspace,
125
+ section,
126
+ });
127
+ }
128
+
129
+ // 5. 패널 닫기
130
+ document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', code: 'Escape', bubbles: true }));
131
+
132
+ return chats;
133
+ } catch (e) {
134
+ try { document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Escape', code: 'Escape', bubbles: true })); } catch (_) { }
135
+ return [];
136
+ }
137
+ })()
@@ -0,0 +1,75 @@
1
+ /**
2
+ * Antigravity v1 — new_session
3
+ *
4
+ * Antigravity Agent 패널 상단의 + 아이콘 클릭으로 새 대화 시작.
5
+ *
6
+ * DOM 구조:
7
+ * .antigravity-agent-side-panel 상단 아이콘 바
8
+ * [0] <a> + (Plus) → 새 대화 ← 이것을 클릭
9
+ * [1] <a> 히스토리 (data-past-conversations-toggle)
10
+ * [2] <div> 더보기 (⋯)
11
+ * [3] <a> 닫기 (✕)
12
+ *
13
+ * 전략:
14
+ * 1. 히스토리 토글의 이전 형제 <a> 클릭 (Plus 아이콘)
15
+ * 2. SVG path "M12 4.5v15m7.5-7.5h-15" (+ 모양) 찾기
16
+ * 3. 패널 상단 첫 번째 <a> 클릭 폴백
17
+ *
18
+ * 최종 확인: 2026-03-11
19
+ */
20
+ (() => {
21
+ try {
22
+ // ─── 1. 히스토리 토글의 이전 형제 요소 (Plus 아이콘) ───
23
+ const toggle = document.querySelector('[data-past-conversations-toggle="true"]');
24
+ if (toggle) {
25
+ const parent = toggle.parentElement;
26
+ if (parent) {
27
+ const children = Array.from(parent.children).filter(c => c.offsetWidth > 0);
28
+ const toggleIdx = children.indexOf(toggle);
29
+ // Plus 아이콘은 히스토리 바로 앞
30
+ if (toggleIdx > 0) {
31
+ const plusBtn = children[toggleIdx - 1];
32
+ plusBtn.click();
33
+ return 'clicked (sibling of toggle)';
34
+ }
35
+ }
36
+ }
37
+
38
+ // ─── 2. SVG path 기반: Plus(+) 모양 SVG ───
39
+ const panel = document.querySelector('.antigravity-agent-side-panel') || document.querySelector('#conversation');
40
+ if (panel) {
41
+ const svgs = panel.querySelectorAll('svg');
42
+ for (const svg of svgs) {
43
+ const path = svg.querySelector('path');
44
+ if (path) {
45
+ const d = path.getAttribute('d') || '';
46
+ // Plus 아이콘 SVG: 세로선 + 가로선
47
+ if (d.includes('v15') && d.includes('h-15')) {
48
+ const clickable = svg.closest('a') || svg.closest('button') || svg.parentElement;
49
+ if (clickable) {
50
+ clickable.click();
51
+ return 'clicked (svg plus)';
52
+ }
53
+ }
54
+ }
55
+ }
56
+ }
57
+
58
+ // ─── 3. 폴백: 패널 상단 영역의 첫 번째 <a> ───
59
+ if (panel) {
60
+ const topLinks = Array.from(panel.querySelectorAll('a')).filter(a => {
61
+ const r = a.getBoundingClientRect();
62
+ const pr = panel.getBoundingClientRect();
63
+ return r.y < pr.y + 30 && a.offsetWidth > 0 && a.offsetWidth < 30;
64
+ });
65
+ if (topLinks.length > 0) {
66
+ topLinks[0].click();
67
+ return 'clicked (first top link)';
68
+ }
69
+ }
70
+
71
+ return 'no new session button found';
72
+ } catch (e) {
73
+ return 'error: ' + e.message;
74
+ }
75
+ })()
@@ -0,0 +1,240 @@
1
+ /**
2
+ * Antigravity v1 — read_chat (v4 — 스크롤 아래만, 가시 영역만 수집)
3
+ *
4
+ * 원칙:
5
+ * - 스크롤은 아래로만 (위로 절대 안 감)
6
+ * - 가상화로 사라진 과거 사용자 메시지는 무시
7
+ * - 현재 보이는 메시지만 수집 (최신 턴 중심)
8
+ *
9
+ * DOM 구조:
10
+ * 사용자: bg-gray-500/15 + select-text + p-2, 내부 whitespace-pre-wrap
11
+ * 어시스턴트: .leading-relaxed.select-text
12
+ */
13
+ (() => {
14
+ try {
15
+ const conv = document.querySelector('#conversation') || document.querySelector('.antigravity-agent-side-panel') || document.body;
16
+ const scroll = conv.querySelector('.overflow-y-auto') || conv;
17
+
18
+ // 1. 상태 감지 — 사이드바 하단 Send/Stop 버튼 기반 (오탐 방지)
19
+ let status = 'idle';
20
+
21
+ // 시그널 A (1순위): 사이드바 하단 빨간 Stop 사각형 (.bg-red-500) — generating 중일 때 표시됨
22
+ const stopSquare = conv.querySelector('[class*="bg-red-500"]') || conv.querySelector('button[class*="rounded"] [class*="bg-red"]');
23
+ if (stopSquare && stopSquare.offsetWidth > 0) {
24
+ status = 'generating';
25
+ }
26
+
27
+ // 시그널 B: conv 내부의 animate-markdown (생성 중 마크다운 렌더링)
28
+ if (status === 'idle') {
29
+ const animMarkdown = scroll.querySelector('.leading-relaxed [class*="animate-markdown"]');
30
+ if (animMarkdown && animMarkdown.offsetWidth > 0) status = 'generating';
31
+ }
32
+
33
+ const title = document.title.split(' \u2014 ')[0].trim() || 'Active Session';
34
+
35
+ // ─── HTML → Markdown 변환기 (대시보드가 ReactMarkdown+remarkGfm 사용) ───
36
+ function htmlToMd(node) {
37
+ if (node.nodeType === 3) return node.textContent || '';
38
+ if (node.nodeType !== 1) return '';
39
+ const tag = node.tagName;
40
+
41
+ // 스타일/스크립트 제거
42
+ if (tag === 'STYLE' || tag === 'SCRIPT' || tag === 'SVG') return '';
43
+
44
+ // 테이블 → GFM
45
+ if (tag === 'TABLE') {
46
+ const rows = Array.from(node.querySelectorAll('tr'));
47
+ if (rows.length === 0) return '';
48
+ const table = rows.map(tr =>
49
+ Array.from(tr.querySelectorAll('th, td')).map(cell => (cell.textContent || '').trim().replace(/\|/g, '\\|'))
50
+ );
51
+ if (table.length === 0) return '';
52
+ const colCount = Math.max(...table.map(r => r.length));
53
+ const header = table[0];
54
+ const sep = Array(colCount).fill('---');
55
+ const body = table.slice(1);
56
+ let md = '| ' + header.join(' | ') + ' |\n';
57
+ md += '| ' + sep.join(' | ') + ' |\n';
58
+ for (const row of body) {
59
+ while (row.length < colCount) row.push('');
60
+ md += '| ' + row.join(' | ') + ' |\n';
61
+ }
62
+ return '\n' + md + '\n';
63
+ }
64
+
65
+ if (tag === 'UL') return '\n' + Array.from(node.children).map(li => '- ' + childrenToMd(li).trim()).join('\n') + '\n';
66
+ if (tag === 'OL') return '\n' + Array.from(node.children).map((li, i) => (i + 1) + '. ' + childrenToMd(li).trim()).join('\n') + '\n';
67
+ if (tag === 'LI') return childrenToMd(node);
68
+
69
+ if (tag === 'H1') return '\n# ' + childrenToMd(node).trim() + '\n';
70
+ if (tag === 'H2') return '\n## ' + childrenToMd(node).trim() + '\n';
71
+ if (tag === 'H3') return '\n### ' + childrenToMd(node).trim() + '\n';
72
+ if (tag === 'H4') return '\n#### ' + childrenToMd(node).trim() + '\n';
73
+
74
+ if (tag === 'STRONG' || tag === 'B') return '**' + childrenToMd(node).trim() + '**';
75
+ if (tag === 'EM' || tag === 'I') return '*' + childrenToMd(node).trim() + '*';
76
+
77
+ if (tag === 'PRE') {
78
+ const codeEl = node.querySelector('code');
79
+ const lang = codeEl ? (codeEl.className.match(/language-(\w+)/)?.[1] || '') : '';
80
+ const code = (codeEl || node).textContent || '';
81
+ return '\n```' + lang + '\n' + code.trim() + '\n```\n';
82
+ }
83
+ if (tag === 'CODE') {
84
+ if (node.parentElement && node.parentElement.tagName === 'PRE') return node.textContent || '';
85
+ return '`' + (node.textContent || '').trim() + '`';
86
+ }
87
+
88
+ if (tag === 'BLOCKQUOTE') return '\n> ' + childrenToMd(node).trim().replace(/\n/g, '\n> ') + '\n';
89
+ if (tag === 'A') return '[' + childrenToMd(node).trim() + '](' + (node.getAttribute('href') || '') + ')';
90
+ if (tag === 'BR') return '\n';
91
+ if (tag === 'P') return '\n' + childrenToMd(node).trim() + '\n';
92
+
93
+ return childrenToMd(node);
94
+ }
95
+ function childrenToMd(node) {
96
+ return Array.from(node.childNodes).map(htmlToMd).join('');
97
+ }
98
+
99
+ function getCleanMd(el) {
100
+ const clone = el.cloneNode(true);
101
+ // 노이즈 제거
102
+ clone.querySelectorAll('button, [role="button"], style, script, svg, .codicon, [class*="feedback"], [aria-label*="Good"], [aria-label*="Bad"]').forEach(n => n.remove());
103
+ // 상태 텍스트 제거 (leaf만, 60자 이하만)
104
+ clone.querySelectorAll('*').forEach(child => {
105
+ if (!child.parentNode) return;
106
+ const t = (child.textContent || '').trim();
107
+ if (t.length > 60) return;
108
+ const low = t.toLowerCase();
109
+ if (/^(analyzed\s+\d|edited\s+\d|ran\s+\S|terminal\s|reading|searching)/i.test(low)) child.remove();
110
+ if (/^(mcp|customizationmcp|serversexport)/i.test(low)) child.remove();
111
+ });
112
+ let md = htmlToMd(clone);
113
+ // "Thought for X seconds" 제거
114
+ md = md.replace(/^Thought for\s+[\d.]+\s*(seconds?|s)\s*/i, '');
115
+ md = md.replace(/\n{3,}/g, '\n\n').trim();
116
+ return md;
117
+ }
118
+
119
+ // 2. 메시지 수집 (스크롤 조작 없음 — 현재 DOM에 있는 것만)
120
+ const collected = [];
121
+ const seenHashes = new Set();
122
+
123
+ // 사용자 메시지 (bg-gray-500/15 + select-text + p-2)
124
+ const allDivs = scroll.querySelectorAll('div');
125
+ for (const el of allDivs) {
126
+ const cls = (el.className || '');
127
+ if (typeof cls !== 'string') continue;
128
+ if (cls.includes('bg-gray-500/15') && cls.includes('select-text') && cls.includes('p-2')) {
129
+ const textEl = el.querySelector('[class*="whitespace-pre-wrap"]') || el;
130
+ const text = (textEl.innerText || '').trim();
131
+ if (!text || text.length < 1) continue;
132
+ const hash = 'user:' + text.slice(0, 200);
133
+ if (seenHashes.has(hash)) continue;
134
+ seenHashes.add(hash);
135
+ collected.push({ role: 'user', text, el });
136
+ }
137
+ }
138
+
139
+ // 어시스턴트 메시지 (leading-relaxed.select-text) — HTML→Markdown 변환
140
+ const assistantBlocks = scroll.querySelectorAll('.leading-relaxed.select-text');
141
+ for (const ab of assistantBlocks) {
142
+ if (ab.offsetHeight < 10) continue;
143
+ if (ab.closest('[class*="max-h-"][class*="overflow-y-auto"]')) continue;
144
+
145
+ let text = getCleanMd(ab);
146
+ if (!text || text.length < 2) continue;
147
+ if (/^(Running command|Checked command|collectStatus)/i.test(text)) continue;
148
+
149
+ const hash = 'assistant:' + text.slice(0, 200);
150
+ if (seenHashes.has(hash)) continue;
151
+ seenHashes.add(hash);
152
+ collected.push({ role: 'assistant', text, el: ab });
153
+ }
154
+
155
+ // 3. DOM 순서 정렬
156
+ collected.sort((a, b) => {
157
+ const pos = a.el.compareDocumentPosition(b.el);
158
+ if (pos & Node.DOCUMENT_POSITION_FOLLOWING) return -1;
159
+ if (pos & Node.DOCUMENT_POSITION_PRECEDING) return 1;
160
+ return 0;
161
+ });
162
+
163
+ // 최신 30개만 유지 (대화 첫 로드 시 수백 개 수집 방지)
164
+ const trimmed = collected.length > 30 ? collected.slice(-30) : collected;
165
+
166
+ const final = trimmed.map((m, i) => ({
167
+ id: 'msg_' + i,
168
+ role: m.role,
169
+ content: m.text.length > 6000 ? m.text.slice(0, 6000) + '\n[... truncated]' : m.text,
170
+ index: i,
171
+ kind: 'standard',
172
+ vsc_history: true
173
+ }));
174
+
175
+ // 4. 입력창
176
+ const editor = conv.querySelector('[contenteditable="true"][role="textbox"]') ||
177
+ conv.querySelector('[data-lexical-editor="true"]') ||
178
+ conv.querySelector('textarea');
179
+ const inputContent = editor ? (editor.innerText || editor.value || '').trim() : '';
180
+
181
+ // 5. 모달/승인 감지 — Run⌥⏎/Reject 인라인 + Deny/Allow 브라우저 승인
182
+ let activeModal = null;
183
+ try {
184
+ const isApprovalLike = (el) => {
185
+ const t = (el.textContent || '').trim().toLowerCase();
186
+ // 드롭다운 옵션 제외
187
+ if (t === 'ask every time') return false;
188
+ return /^(run|reject|skip|approve|allow|deny|cancel|accept|yes|no)\b/i.test(t)
189
+ || t === 'always allow' || t === 'always deny'
190
+ || t.includes('run ') || t.includes('approve') || t.includes('reject')
191
+ || t.includes('skip');
192
+ };
193
+ // A: 전통적 모달 다이얼로그
194
+ const dialog = document.querySelector('.monaco-dialog-box, [role="dialog"], .monaco-modal-block');
195
+ if (dialog && dialog.offsetWidth > 80 && dialog.offsetHeight > 40) {
196
+ const msg = (dialog.querySelector('.dialog-message, .dialog-message-text') || dialog).innerText?.trim() || '';
197
+ const buttons = Array.from(dialog.querySelectorAll('.monaco-button, button'))
198
+ .map(b => (b.innerText || '').trim())
199
+ .filter(t => t.length > 0 && t.length < 30);
200
+ if (msg || buttons.length > 0) {
201
+ activeModal = { message: msg.slice(0, 300), buttons, width: dialog.offsetWidth, height: dialog.offsetHeight };
202
+ }
203
+ }
204
+ // B: 인라인 approval 버튼 (Run⌥⏎, Reject, Deny, Allow 등)
205
+ // ⚠ 사이드바(conv) 내부의 버튼만 검사 — 에디터의 Accept/Reject Changes 제외
206
+ if (!activeModal) {
207
+ const panelBtns = Array.from(conv.querySelectorAll('button')).filter(b => b.offsetWidth > 0 && b.offsetHeight > 0);
208
+ const approvalBtns = panelBtns.filter(isApprovalLike);
209
+ if (approvalBtns.length > 0) {
210
+ const hasActionBtn = approvalBtns.some(b => {
211
+ const t = (b.textContent || '').trim().toLowerCase();
212
+ return t.indexOf('run') === 0 || t === 'reject' || t.indexOf('reject') === 0
213
+ || t === 'skip' || t.indexOf('skip') === 0
214
+ || t === 'deny' || t === 'allow' || t === 'always allow' || t === 'always deny'
215
+ || t === 'accept' || t === 'approve';
216
+ });
217
+ if (hasActionBtn) {
218
+ const btnTexts = [...new Set(
219
+ approvalBtns.map(b => (b.textContent || '').trim())
220
+ .filter(t => t.length > 0 && t.length < 40)
221
+ )];
222
+ const firstApproval = approvalBtns[0];
223
+ let wrapper = firstApproval.parentElement;
224
+ for (let up = 0; up < 5 && wrapper; up++, wrapper = wrapper.parentElement) {
225
+ if (wrapper.offsetHeight > 40) break;
226
+ }
227
+ const msg = wrapper ? (wrapper.textContent || '').trim().slice(0, 300) : '';
228
+ activeModal = { message: msg, buttons: btnTexts, width: 400, height: 100 };
229
+ }
230
+ }
231
+ }
232
+ // 모달이 감지되면 status를 waiting_approval로 변경
233
+ if (activeModal) status = 'waiting_approval';
234
+ } catch (e) { activeModal = null; }
235
+
236
+ return { id: 'active_session', status, title, messages: final, inputContent, activeModal };
237
+ } catch (e) {
238
+ return { id: 'error', status: 'error', error: e.message, messages: [] };
239
+ }
240
+ })()
@@ -0,0 +1,64 @@
1
+ /**
2
+ * Antigravity v1 — resolve_action
3
+ *
4
+ * 승인/거부 버튼 클릭 (파라미터: ${BUTTON_TEXT})
5
+ * Antigravity는 모달이 아닌 인라인 버튼(Run⌥⏎, Reject 등)을 사용하므로,
6
+ * DOM 전체에서 startsWith 매칭으로 탐색.
7
+ */
8
+ (() => {
9
+ const want = ${ BUTTON_TEXT };
10
+ function norm(t) { return (t || '').replace(/\s+/g, ' ').trim().toLowerCase(); }
11
+ function matches(el) {
12
+ const t = norm(el.textContent);
13
+ if (t.length > 80) return false;
14
+ if (t === want) return true;
15
+ if (t.indexOf(want) === 0) return true; // startsWith: "run⌥⏎" starts with "run"
16
+ if (want === 'run' && (/^run\s*/.test(t) || t === 'enter' || t === '⏎')) return true;
17
+ if (want === 'skip' && t.indexOf('skip') >= 0) return true;
18
+ if (want === 'reject' && t.indexOf('reject') >= 0) return true;
19
+ if (want === 'accept' && t.indexOf('accept') >= 0) return true;
20
+ return false;
21
+ }
22
+ function doClick(el) {
23
+ if (!el) return;
24
+ el.focus && el.focus();
25
+ el.click && el.click();
26
+ try {
27
+ el.dispatchEvent(new MouseEvent('mousedown', { bubbles: true, cancelable: true, view: window }));
28
+ el.dispatchEvent(new MouseEvent('mouseup', { bubbles: true, cancelable: true, view: window }));
29
+ el.dispatchEvent(new MouseEvent('click', { bubbles: true, cancelable: true, view: window }));
30
+ } catch (e) { }
31
+ }
32
+ const sel = 'button, [role="button"], .monaco-button, .solid-dropdown-toggle';
33
+ // 1. Try run-command-review container first
34
+ const runReview = document.querySelector('[class*="run-command-review"]');
35
+ if (runReview) {
36
+ const btns = Array.from(runReview.querySelectorAll(sel)).filter(b => b.offsetWidth > 0);
37
+ for (const b of btns) { if (matches(b)) { doClick(b); return true; } }
38
+ }
39
+ // 2. Try overlay containers
40
+ const overlay = document.querySelector('.quick-agent-overlay-container, [class*="overlay-container"]');
41
+ if (overlay) {
42
+ const btns = Array.from(overlay.querySelectorAll(sel)).filter(b => b.offsetWidth > 0);
43
+ for (const b of btns) { if (matches(b)) { doClick(b); return true; } }
44
+ }
45
+ // 3. Try dialog boxes
46
+ const dialog = document.querySelector('.monaco-dialog-box, .monaco-modal-block, [role="dialog"]');
47
+ if (dialog) {
48
+ const btns = Array.from(dialog.querySelectorAll(sel)).filter(b => b.offsetWidth > 0);
49
+ for (const b of btns) { if (matches(b)) { doClick(b); return true; } }
50
+ }
51
+ // 4. Global search — find any visible matching button
52
+ const allBtns = Array.from(document.querySelectorAll(sel)).filter(b => b.offsetWidth > 0 && b.getBoundingClientRect().height > 0);
53
+ for (const b of allBtns) { if (matches(b)) { doClick(b); return true; } }
54
+ // 5. If Run was requested, try Enter key on focused element
55
+ if (want === 'run') {
56
+ const focused = document.activeElement;
57
+ if (focused) {
58
+ focused.dispatchEvent(new KeyboardEvent('keydown', { key: 'Enter', code: 'Enter', keyCode: 13, which: 13, bubbles: true }));
59
+ focused.dispatchEvent(new KeyboardEvent('keyup', { key: 'Enter', code: 'Enter', keyCode: 13, which: 13, bubbles: true }));
60
+ return true;
61
+ }
62
+ }
63
+ return false;
64
+ })()
@@ -0,0 +1,56 @@
1
+ /**
2
+ * Antigravity v1 — send_message
3
+ *
4
+ * Antigravity는 contenteditable div[role="textbox"]를 사용.
5
+ * 여러 개의 contenteditable이 있을 수 있으므로 y좌표가 가장 큰 (메인 채팅) 것을 선택.
6
+ *
7
+ * ⚠️ Enter 이벤트에 composed: true + which 필수 (Shadow DOM 경계 통과 + React 호환)
8
+ * ⚠️ keydown + keypress + keyup 전체 시퀀스 필요
9
+ *
10
+ * 파라미터: ${ MESSAGE }
11
+ * 최종 확인: 2026-03-10
12
+ */
13
+ (async () => {
14
+ try {
15
+ const msg = ${ MESSAGE };
16
+
17
+ // ─── 1. 메인 채팅 입력 필드 찾기 ───
18
+ const editors = document.querySelectorAll('[contenteditable="true"][role="textbox"]');
19
+ if (!editors.length) return 'error: no contenteditable textbox found';
20
+
21
+ // y좌표가 가장 큰 (화면 아래쪽 = 메인 채팅) 에디터 선택
22
+ const editor = [...editors].reduce((a, b) =>
23
+ b.getBoundingClientRect().y > a.getBoundingClientRect().y ? b : a
24
+ );
25
+
26
+ // ─── 2. 텍스트 삽입 ───
27
+ editor.focus();
28
+
29
+ // 기존 내용 삭제
30
+ document.execCommand('selectAll', false, null);
31
+ document.execCommand('delete', false, null);
32
+
33
+ // 텍스트 삽입
34
+ document.execCommand('insertText', false, msg);
35
+
36
+ // React에 변경 알림
37
+ editor.dispatchEvent(new Event('input', { bubbles: true }));
38
+
39
+ await new Promise(r => setTimeout(r, 300));
40
+
41
+ // ─── 3. Enter 키 전송 (full sequence) ───
42
+ const enterOpts = {
43
+ key: 'Enter', code: 'Enter',
44
+ keyCode: 13, which: 13,
45
+ bubbles: true, cancelable: true, composed: true,
46
+ };
47
+
48
+ editor.dispatchEvent(new KeyboardEvent('keydown', enterOpts));
49
+ editor.dispatchEvent(new KeyboardEvent('keypress', enterOpts));
50
+ editor.dispatchEvent(new KeyboardEvent('keyup', enterOpts));
51
+
52
+ return 'sent';
53
+ } catch (e) {
54
+ return 'error: ' + e.message;
55
+ }
56
+ })()