@adhdev/daemon-core 0.5.3

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 (217) hide show
  1. package/dist/index.d.ts +2662 -0
  2. package/dist/index.js +11341 -0
  3. package/dist/index.js.map +1 -0
  4. package/package.json +48 -0
  5. package/providers/_builtin/.github/workflows/generate-registry.yml +57 -0
  6. package/providers/_builtin/COMPATIBILITY.md +217 -0
  7. package/providers/_builtin/CONTRIBUTING.md +200 -0
  8. package/providers/_builtin/README.md +119 -0
  9. package/providers/_builtin/_helpers/index.js +188 -0
  10. package/providers/_builtin/acp/agentpool/provider.json +54 -0
  11. package/providers/_builtin/acp/amp/provider.json +52 -0
  12. package/providers/_builtin/acp/auggie/provider.json +57 -0
  13. package/providers/_builtin/acp/autodev/provider.json +54 -0
  14. package/providers/_builtin/acp/autohand/provider.json +52 -0
  15. package/providers/_builtin/acp/blackbox-ai/provider.json +54 -0
  16. package/providers/_builtin/acp/claude-agent/provider.json +57 -0
  17. package/providers/_builtin/acp/cline-acp/provider.json +54 -0
  18. package/providers/_builtin/acp/codebuddy/provider.json +54 -0
  19. package/providers/_builtin/acp/codex-cli/provider.json +57 -0
  20. package/providers/_builtin/acp/corust-agent/provider.json +52 -0
  21. package/providers/_builtin/acp/crow-cli/provider.json +54 -0
  22. package/providers/_builtin/acp/cursor-acp/provider.json +54 -0
  23. package/providers/_builtin/acp/deepagents/provider.json +52 -0
  24. package/providers/_builtin/acp/dimcode/provider.json +54 -0
  25. package/providers/_builtin/acp/docker-cagent/provider.json +57 -0
  26. package/providers/_builtin/acp/factory-droid/provider.json +60 -0
  27. package/providers/_builtin/acp/fast-agent/provider.json +52 -0
  28. package/providers/_builtin/acp/gemini-cli/provider.json +114 -0
  29. package/providers/_builtin/acp/github-copilot/provider.json +54 -0
  30. package/providers/_builtin/acp/goose/provider.json +57 -0
  31. package/providers/_builtin/acp/junie/provider.json +52 -0
  32. package/providers/_builtin/acp/kilo/provider.json +54 -0
  33. package/providers/_builtin/acp/kimi-cli/provider.json +57 -0
  34. package/providers/_builtin/acp/minion-code/provider.json +52 -0
  35. package/providers/_builtin/acp/mistral-vibe/provider.json +57 -0
  36. package/providers/_builtin/acp/nova/provider.json +54 -0
  37. package/providers/_builtin/acp/openclaw/provider.json +54 -0
  38. package/providers/_builtin/acp/opencode/provider.json +52 -0
  39. package/providers/_builtin/acp/openhands/provider.json +54 -0
  40. package/providers/_builtin/acp/pi-acp/provider.json +52 -0
  41. package/providers/_builtin/acp/qoder/provider.json +54 -0
  42. package/providers/_builtin/acp/qwen-code/provider.json +60 -0
  43. package/providers/_builtin/acp/stakpak/provider.json +54 -0
  44. package/providers/_builtin/acp/vtcode/provider.json +54 -0
  45. package/providers/_builtin/cli/claude-cli/provider.json +100 -0
  46. package/providers/_builtin/cli/codex-cli/provider.json +89 -0
  47. package/providers/_builtin/cli/gemini-cli/provider.json +93 -0
  48. package/providers/_builtin/docs/CDP_SELECTOR_GUIDE.md +370 -0
  49. package/providers/_builtin/docs/PROVIDER_GUIDE.md +916 -0
  50. package/providers/_builtin/extension/cline/provider.json +35 -0
  51. package/providers/_builtin/extension/cline/scripts/focus_editor.js +48 -0
  52. package/providers/_builtin/extension/cline/scripts/list_chats.js +100 -0
  53. package/providers/_builtin/extension/cline/scripts/list_models.js +43 -0
  54. package/providers/_builtin/extension/cline/scripts/list_modes.js +35 -0
  55. package/providers/_builtin/extension/cline/scripts/new_session.js +85 -0
  56. package/providers/_builtin/extension/cline/scripts/open_panel.js +25 -0
  57. package/providers/_builtin/extension/cline/scripts/read_chat.js +257 -0
  58. package/providers/_builtin/extension/cline/scripts/resolve_action.js +83 -0
  59. package/providers/_builtin/extension/cline/scripts/send_message.js +95 -0
  60. package/providers/_builtin/extension/cline/scripts/set_mode.js +36 -0
  61. package/providers/_builtin/extension/cline/scripts/set_model.js +36 -0
  62. package/providers/_builtin/extension/cline/scripts/switch_session.js +206 -0
  63. package/providers/_builtin/extension/cline/scripts.js +73 -0
  64. package/providers/_builtin/extension/roo-code/provider.json +35 -0
  65. package/providers/_builtin/extension/roo-code/scripts.js +659 -0
  66. package/providers/_builtin/ide/antigravity/provider.json +68 -0
  67. package/providers/_builtin/ide/antigravity/scripts/1.106/focus_editor.js +20 -0
  68. package/providers/_builtin/ide/antigravity/scripts/1.106/list_chats.js +137 -0
  69. package/providers/_builtin/ide/antigravity/scripts/1.106/list_models.js +38 -0
  70. package/providers/_builtin/ide/antigravity/scripts/1.106/list_modes.js +48 -0
  71. package/providers/_builtin/ide/antigravity/scripts/1.106/new_session.js +75 -0
  72. package/providers/_builtin/ide/antigravity/scripts/1.106/read_chat.js +262 -0
  73. package/providers/_builtin/ide/antigravity/scripts/1.106/resolve_action.js +68 -0
  74. package/providers/_builtin/ide/antigravity/scripts/1.106/scripts.js +57 -0
  75. package/providers/_builtin/ide/antigravity/scripts/1.106/send_message.js +56 -0
  76. package/providers/_builtin/ide/antigravity/scripts/1.106/set_mode.js +34 -0
  77. package/providers/_builtin/ide/antigravity/scripts/1.106/set_model.js +47 -0
  78. package/providers/_builtin/ide/antigravity/scripts/1.106/switch_session.js +114 -0
  79. package/providers/_builtin/ide/antigravity/scripts/1.107/focus_editor.js +20 -0
  80. package/providers/_builtin/ide/antigravity/scripts/1.107/list_chats.js +137 -0
  81. package/providers/_builtin/ide/antigravity/scripts/1.107/list_models.js +61 -0
  82. package/providers/_builtin/ide/antigravity/scripts/1.107/list_modes.js +72 -0
  83. package/providers/_builtin/ide/antigravity/scripts/1.107/new_session.js +75 -0
  84. package/providers/_builtin/ide/antigravity/scripts/1.107/read_chat.js +262 -0
  85. package/providers/_builtin/ide/antigravity/scripts/1.107/resolve_action.js +68 -0
  86. package/providers/_builtin/ide/antigravity/scripts/1.107/scripts.js +67 -0
  87. package/providers/_builtin/ide/antigravity/scripts/1.107/send_message.js +56 -0
  88. package/providers/_builtin/ide/antigravity/scripts/1.107/set_mode.js +67 -0
  89. package/providers/_builtin/ide/antigravity/scripts/1.107/set_model.js +72 -0
  90. package/providers/_builtin/ide/antigravity/scripts/1.107/switch_session.js +114 -0
  91. package/providers/_builtin/ide/cursor/provider.json +70 -0
  92. package/providers/_builtin/ide/cursor/scripts/0.49/dismiss_notification.js +30 -0
  93. package/providers/_builtin/ide/cursor/scripts/0.49/focus_editor.js +13 -0
  94. package/providers/_builtin/ide/cursor/scripts/0.49/list_models.js +78 -0
  95. package/providers/_builtin/ide/cursor/scripts/0.49/list_modes.js +40 -0
  96. package/providers/_builtin/ide/cursor/scripts/0.49/list_notifications.js +23 -0
  97. package/providers/_builtin/ide/cursor/scripts/0.49/list_sessions.js +42 -0
  98. package/providers/_builtin/ide/cursor/scripts/0.49/new_session.js +20 -0
  99. package/providers/_builtin/ide/cursor/scripts/0.49/open_panel.js +23 -0
  100. package/providers/_builtin/ide/cursor/scripts/0.49/read_chat.js +75 -0
  101. package/providers/_builtin/ide/cursor/scripts/0.49/resolve_action.js +19 -0
  102. package/providers/_builtin/ide/cursor/scripts/0.49/scripts.js +78 -0
  103. package/providers/_builtin/ide/cursor/scripts/0.49/send_message.js +23 -0
  104. package/providers/_builtin/ide/cursor/scripts/0.49/set_mode.js +38 -0
  105. package/providers/_builtin/ide/cursor/scripts/0.49/set_model.js +81 -0
  106. package/providers/_builtin/ide/cursor/scripts/0.49/switch_session.js +28 -0
  107. package/providers/_builtin/ide/kiro/provider.json +67 -0
  108. package/providers/_builtin/ide/kiro/scripts/focus_editor.js +20 -0
  109. package/providers/_builtin/ide/kiro/scripts/open_panel.js +47 -0
  110. package/providers/_builtin/ide/kiro/scripts/resolve_action.js +54 -0
  111. package/providers/_builtin/ide/kiro/scripts/send_message.js +29 -0
  112. package/providers/_builtin/ide/kiro/scripts/webview_list_models.js +39 -0
  113. package/providers/_builtin/ide/kiro/scripts/webview_list_modes.js +39 -0
  114. package/providers/_builtin/ide/kiro/scripts/webview_list_sessions.js +21 -0
  115. package/providers/_builtin/ide/kiro/scripts/webview_new_session.js +34 -0
  116. package/providers/_builtin/ide/kiro/scripts/webview_read_chat.js +68 -0
  117. package/providers/_builtin/ide/kiro/scripts/webview_send_message.js +72 -0
  118. package/providers/_builtin/ide/kiro/scripts/webview_set_mode.js +15 -0
  119. package/providers/_builtin/ide/kiro/scripts/webview_set_model.js +15 -0
  120. package/providers/_builtin/ide/kiro/scripts/webview_switch_session.js +26 -0
  121. package/providers/_builtin/ide/kiro/scripts.js +62 -0
  122. package/providers/_builtin/ide/pearai/provider.json +67 -0
  123. package/providers/_builtin/ide/pearai/scripts/focus_editor.js +20 -0
  124. package/providers/_builtin/ide/pearai/scripts/list_sessions.js +38 -0
  125. package/providers/_builtin/ide/pearai/scripts/new_session.js +55 -0
  126. package/providers/_builtin/ide/pearai/scripts/open_panel.js +46 -0
  127. package/providers/_builtin/ide/pearai/scripts/resolve_action.js +54 -0
  128. package/providers/_builtin/ide/pearai/scripts/send_message.js +29 -0
  129. package/providers/_builtin/ide/pearai/scripts/webview_list_models.js +43 -0
  130. package/providers/_builtin/ide/pearai/scripts/webview_list_modes.js +35 -0
  131. package/providers/_builtin/ide/pearai/scripts/webview_list_sessions.js +62 -0
  132. package/providers/_builtin/ide/pearai/scripts/webview_new_session.js +49 -0
  133. package/providers/_builtin/ide/pearai/scripts/webview_read_chat.js +92 -0
  134. package/providers/_builtin/ide/pearai/scripts/webview_resolve_action.js +59 -0
  135. package/providers/_builtin/ide/pearai/scripts/webview_send_message.js +72 -0
  136. package/providers/_builtin/ide/pearai/scripts/webview_set_mode.js +36 -0
  137. package/providers/_builtin/ide/pearai/scripts/webview_set_model.js +36 -0
  138. package/providers/_builtin/ide/pearai/scripts/webview_switch_session.js +34 -0
  139. package/providers/_builtin/ide/pearai/scripts.js +74 -0
  140. package/providers/_builtin/ide/trae/provider.json +66 -0
  141. package/providers/_builtin/ide/trae/scripts/focus_editor.js +20 -0
  142. package/providers/_builtin/ide/trae/scripts/list_chats.js +24 -0
  143. package/providers/_builtin/ide/trae/scripts/list_models.js +39 -0
  144. package/providers/_builtin/ide/trae/scripts/list_modes.js +39 -0
  145. package/providers/_builtin/ide/trae/scripts/new_session.js +30 -0
  146. package/providers/_builtin/ide/trae/scripts/open_panel.js +44 -0
  147. package/providers/_builtin/ide/trae/scripts/read_chat.js +113 -0
  148. package/providers/_builtin/ide/trae/scripts/resolve_action.js +54 -0
  149. package/providers/_builtin/ide/trae/scripts/send_message.js +69 -0
  150. package/providers/_builtin/ide/trae/scripts/set_mode.js +15 -0
  151. package/providers/_builtin/ide/trae/scripts/set_model.js +15 -0
  152. package/providers/_builtin/ide/trae/scripts/switch_session.js +23 -0
  153. package/providers/_builtin/ide/trae/scripts.js +57 -0
  154. package/providers/_builtin/ide/vscode/provider.json +64 -0
  155. package/providers/_builtin/ide/vscode-insiders/provider.json +62 -0
  156. package/providers/_builtin/ide/vscodium/provider.json +63 -0
  157. package/providers/_builtin/ide/windsurf/provider.json +53 -0
  158. package/providers/_builtin/ide/windsurf/scripts/focus_editor.js +30 -0
  159. package/providers/_builtin/ide/windsurf/scripts/list_chats.js +117 -0
  160. package/providers/_builtin/ide/windsurf/scripts/list_models.js +39 -0
  161. package/providers/_builtin/ide/windsurf/scripts/list_modes.js +39 -0
  162. package/providers/_builtin/ide/windsurf/scripts/new_session.js +69 -0
  163. package/providers/_builtin/ide/windsurf/scripts/open_panel.js +58 -0
  164. package/providers/_builtin/ide/windsurf/scripts/read_chat.js +297 -0
  165. package/providers/_builtin/ide/windsurf/scripts/resolve_action.js +68 -0
  166. package/providers/_builtin/ide/windsurf/scripts/send_message.js +87 -0
  167. package/providers/_builtin/ide/windsurf/scripts/set_mode.js +15 -0
  168. package/providers/_builtin/ide/windsurf/scripts/set_model.js +15 -0
  169. package/providers/_builtin/ide/windsurf/scripts/switch_session.js +58 -0
  170. package/providers/_builtin/ide/windsurf/scripts.js +57 -0
  171. package/providers/_builtin/registry.json +266 -0
  172. package/providers/_builtin/validate.js +156 -0
  173. package/src/agent-stream/index.ts +6 -0
  174. package/src/agent-stream/manager.ts +286 -0
  175. package/src/agent-stream/poller.ts +154 -0
  176. package/src/agent-stream/provider-adapter.ts +138 -0
  177. package/src/agent-stream/types.ts +61 -0
  178. package/src/boot/daemon-lifecycle.ts +252 -0
  179. package/src/cdp/devtools.ts +335 -0
  180. package/src/cdp/initializer.ts +191 -0
  181. package/src/cdp/manager.ts +897 -0
  182. package/src/cdp/scanner.ts +185 -0
  183. package/src/cdp/setup.ts +150 -0
  184. package/src/cli-adapter-types.ts +25 -0
  185. package/src/cli-adapters/provider-cli-adapter.ts +448 -0
  186. package/src/commands/cdp-commands.ts +208 -0
  187. package/src/commands/chat-commands.ts +675 -0
  188. package/src/commands/cli-manager.ts +353 -0
  189. package/src/commands/handler.ts +328 -0
  190. package/src/commands/router.ts +258 -0
  191. package/src/commands/stream-commands.ts +325 -0
  192. package/src/config/chat-history.ts +211 -0
  193. package/src/config/config.ts +219 -0
  194. package/src/daemon/dev-server.ts +2378 -0
  195. package/src/daemon/scaffold-template.ts +394 -0
  196. package/src/daemon-core.ts +50 -0
  197. package/src/detection/cli-detector.ts +89 -0
  198. package/src/detection/ide-detector.ts +157 -0
  199. package/src/index.ts +103 -0
  200. package/src/installer.ts +263 -0
  201. package/src/ipc-protocol.ts +133 -0
  202. package/src/launch.ts +433 -0
  203. package/src/logging/command-log.ts +180 -0
  204. package/src/logging/logger.ts +316 -0
  205. package/src/providers/acp-provider-instance.ts +1140 -0
  206. package/src/providers/cli-provider-instance.ts +207 -0
  207. package/src/providers/contracts.ts +524 -0
  208. package/src/providers/extension-provider-instance.ts +156 -0
  209. package/src/providers/ide-provider-instance.ts +377 -0
  210. package/src/providers/index.ts +18 -0
  211. package/src/providers/provider-instance-manager.ts +182 -0
  212. package/src/providers/provider-instance.ts +112 -0
  213. package/src/providers/provider-loader.ts +1031 -0
  214. package/src/providers/status-monitor.ts +125 -0
  215. package/src/providers/version-archive.ts +266 -0
  216. package/src/status/reporter.ts +294 -0
  217. package/src/types.ts +206 -0
@@ -0,0 +1,68 @@
1
+ {
2
+ "type": "antigravity",
3
+ "name": "Antigravity",
4
+ "category": "ide",
5
+ "displayName": "Antigravity",
6
+ "icon": "🚀",
7
+ "cli": "antigravity",
8
+ "cdpPorts": [
9
+ 9335,
10
+ 9336
11
+ ],
12
+ "targetFilter": {
13
+ "urlIncludes": "workbench.html",
14
+ "urlExcludes": ["agent"],
15
+ "titleExcludes": "extension-output|ADHDev CDP|Debug Console|Output\\s*$|Launchpad"
16
+ },
17
+ "processNames": {
18
+ "darwin": "Antigravity",
19
+ "win32": [
20
+ "Antigravity.exe"
21
+ ]
22
+ },
23
+ "paths": {
24
+ "darwin": [
25
+ "/Applications/Antigravity.app"
26
+ ],
27
+ "win32": [
28
+ "C:\\Users\\*\\AppData\\Local\\Programs\\antigravity\\Antigravity.exe"
29
+ ],
30
+ "linux": [
31
+ "/opt/Antigravity",
32
+ "/usr/share/antigravity"
33
+ ]
34
+ },
35
+ "inputMethod": "cdp-type-and-send",
36
+ "inputSelector": "[contenteditable=\"true\"][role=\"textbox\"]",
37
+ "versionCommand": "antigravity --version",
38
+ "providerVersion": "1.0.0",
39
+ "compatibility": [
40
+ { "ideVersion": ">=1.107.0", "scriptDir": "scripts/1.107" },
41
+ { "ideVersion": ">=1.105.0", "scriptDir": "scripts/1.106" }
42
+ ],
43
+ "defaultScriptDir": "scripts/1.107",
44
+ "settings": {
45
+ "approvalAlert": {
46
+ "type": "boolean",
47
+ "default": true,
48
+ "public": true,
49
+ "label": "Approval Notifications",
50
+ "description": "Show notification when approval is needed"
51
+ },
52
+ "longGeneratingAlert": {
53
+ "type": "boolean",
54
+ "default": true,
55
+ "public": true,
56
+ "label": "Long Generation Alert",
57
+ "description": "Alert when generation takes too long"
58
+ },
59
+ "longGeneratingThresholdSec": {
60
+ "type": "number",
61
+ "default": 180,
62
+ "public": true,
63
+ "label": "Long Generation Threshold (sec)",
64
+ "min": 30,
65
+ "max": 600
66
+ }
67
+ }
68
+ }
@@ -0,0 +1,20 @@
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
+ })()
@@ -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,38 @@
1
+ /**
2
+ * Antigravity — list_models
3
+ * antigravity-agent-side-panel 내부 모델 드롭다운에서 목록 + 현재 모델 추출
4
+ * → { models: string[], current: string }
5
+ */
6
+ (() => {
7
+ try {
8
+ const models = [];
9
+ let current = '';
10
+
11
+ // 1. 모델 항목에서 목록 추출
12
+ // 셀렉터: .px-2.py-1.flex.items-center.justify-between.cursor-pointer
13
+ const items = document.querySelectorAll('.px-2.py-1.flex.items-center.justify-between.cursor-pointer');
14
+ for (const item of items) {
15
+ const label = item.querySelector('.text-xs.font-medium');
16
+ const text = (label || item).textContent?.trim();
17
+ if (!text || text.length > 60) continue;
18
+ // 모델명 검증 (Claude, Gemini, GPT, etc.)
19
+ models.push(text);
20
+ // 선택된 항목: bg-gray-500/20
21
+ if ((item.className || '').includes('bg-gray-500/20')) {
22
+ current = text;
23
+ }
24
+ }
25
+
26
+ // 2. 모델 목록이 없으면 (드롭다운 닫힘) → 트리거 버튼에서 현재 모델만
27
+ if (models.length === 0) {
28
+ const trigger = document.querySelector('.flex.min-w-0.max-w-full.cursor-pointer.items-center');
29
+ if (trigger) {
30
+ current = trigger.textContent?.trim() || '';
31
+ }
32
+ }
33
+
34
+ return JSON.stringify({ models, current });
35
+ } catch (e) {
36
+ return JSON.stringify({ models: [], current: '', error: e.message });
37
+ }
38
+ })()
@@ -0,0 +1,48 @@
1
+ /**
2
+ * Antigravity — list_modes
3
+ * Conversation mode: Planning / Fast 세그먼트 컨트롤
4
+ * inputBox 근처의 "Conversation mode" 패널에서 읽기
5
+ * → { modes: string[], current: string }
6
+ */
7
+ (() => {
8
+ try {
9
+ const modes = [];
10
+ let current = '';
11
+
12
+ // "Conversation mode" 헤더를 포함하는 패널 찾기
13
+ const headers = document.querySelectorAll('.text-xs.px-2.pb-1.opacity-80');
14
+ for (const header of headers) {
15
+ if (header.textContent?.trim() === 'Conversation mode') {
16
+ // 형제 요소들에서 모드 항목 추출
17
+ const parent = header.parentElement;
18
+ if (!parent) continue;
19
+ const items = parent.querySelectorAll('.font-medium');
20
+ for (const item of items) {
21
+ const text = item.textContent?.trim();
22
+ if (text && text.length < 20) {
23
+ modes.push(text);
24
+ }
25
+ }
26
+ break;
27
+ }
28
+ }
29
+
30
+ // 현재 모드: Fast 버튼의 텍스트 (현재 활성 모드 표시)
31
+ const modeBtn = [...document.querySelectorAll('button')].find(b => {
32
+ const cls = b.className || '';
33
+ return cls.includes('py-1') && cls.includes('pl-1') && cls.includes('pr-2') && b.offsetWidth > 0;
34
+ });
35
+ if (modeBtn) {
36
+ current = modeBtn.textContent?.trim() || '';
37
+ }
38
+
39
+ // modes가 비어있으면 기본값
40
+ if (modes.length === 0) {
41
+ modes.push('Planning', 'Fast');
42
+ }
43
+
44
+ return JSON.stringify({ modes, current });
45
+ } catch (e) {
46
+ return JSON.stringify({ modes: [], current: '', error: e.message });
47
+ }
48
+ })()
@@ -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,262 @@
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
+ // C: footer 기반 사용량/quota 다이얼로그 (Dismiss / See Plans / Enable Overages 등)
233
+ // <footer> 요소가 conv 안에 존재하고 2개 이상 버튼이 있으면 사용자 액션이 필요한 카드로 판단
234
+ if (!activeModal) {
235
+ const footers = Array.from(conv.querySelectorAll('footer')).filter(f => f.offsetWidth > 0 && f.offsetHeight > 0);
236
+ for (const footer of footers) {
237
+ const footerBtns = Array.from(footer.querySelectorAll('button, a')).filter(b => b.offsetWidth > 0);
238
+ if (footerBtns.length >= 2) {
239
+ // 카드 컨테이너: footer 상위에서 충분한 높이를 가진 첫 번째 요소
240
+ let card = footer.parentElement;
241
+ for (let up = 0; up < 4 && card; up++) {
242
+ if (card.offsetHeight > 60) break;
243
+ card = card.parentElement;
244
+ }
245
+ const msg = card ? (card.innerText || '').trim().slice(0, 300) : '';
246
+ const btnTexts = footerBtns.map(b => (b.innerText || '').trim()).filter(t => t.length > 0 && t.length < 40);
247
+ if (btnTexts.length >= 2) {
248
+ activeModal = { message: msg, buttons: btnTexts, width: card ? card.offsetWidth : 300, height: card ? card.offsetHeight : 100 };
249
+ break;
250
+ }
251
+ }
252
+ }
253
+ }
254
+ // 모달이 감지되면 status를 waiting_approval로 변경
255
+ if (activeModal) status = 'waiting_approval';
256
+ } catch (e) { activeModal = null; }
257
+
258
+ return { id: 'active_session', status, title, messages: final, inputContent, activeModal };
259
+ } catch (e) {
260
+ return { id: 'error', status: 'error', error: e.message, messages: [] };
261
+ }
262
+ })()