claude-mpm 5.1.9__py3-none-any.whl → 5.4.48__py3-none-any.whl

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.

Potentially problematic release.


This version of claude-mpm might be problematic. Click here for more details.

Files changed (248) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/__init__.py +4 -0
  3. claude_mpm/agents/BASE_AGENT.md +164 -0
  4. claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +1 -1
  5. claude_mpm/agents/MEMORY.md +1 -1
  6. claude_mpm/agents/PM_INSTRUCTIONS.md +843 -900
  7. claude_mpm/agents/WORKFLOW.md +5 -254
  8. claude_mpm/agents/agent_loader.py +13 -44
  9. claude_mpm/agents/base_agent.json +1 -1
  10. claude_mpm/agents/frontmatter_validator.py +2 -2
  11. claude_mpm/agents/templates/circuit-breakers.md +138 -1
  12. claude_mpm/cli/__main__.py +4 -0
  13. claude_mpm/cli/chrome_devtools_installer.py +175 -0
  14. claude_mpm/cli/commands/agent_state_manager.py +18 -27
  15. claude_mpm/cli/commands/agents.py +9 -40
  16. claude_mpm/cli/commands/auto_configure.py +210 -25
  17. claude_mpm/cli/commands/config.py +88 -2
  18. claude_mpm/cli/commands/configure.py +1098 -159
  19. claude_mpm/cli/commands/configure_agent_display.py +25 -6
  20. claude_mpm/cli/commands/mpm_init/core.py +225 -46
  21. claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
  22. claude_mpm/cli/commands/mpm_init/prompts.py +280 -0
  23. claude_mpm/cli/commands/postmortem.py +1 -1
  24. claude_mpm/cli/commands/profile.py +277 -0
  25. claude_mpm/cli/commands/skills.py +218 -197
  26. claude_mpm/cli/commands/summarize.py +413 -0
  27. claude_mpm/cli/executor.py +21 -3
  28. claude_mpm/cli/interactive/agent_wizard.py +2 -2
  29. claude_mpm/cli/parsers/agents_parser.py +0 -9
  30. claude_mpm/cli/parsers/auto_configure_parser.py +0 -138
  31. claude_mpm/cli/parsers/base_parser.py +12 -0
  32. claude_mpm/cli/parsers/config_parser.py +153 -83
  33. claude_mpm/cli/parsers/profile_parser.py +148 -0
  34. claude_mpm/cli/parsers/skills_parser.py +0 -5
  35. claude_mpm/cli/startup.py +876 -149
  36. claude_mpm/commands/mpm-config.md +28 -0
  37. claude_mpm/commands/mpm-doctor.md +9 -22
  38. claude_mpm/commands/mpm-help.md +5 -287
  39. claude_mpm/commands/mpm-init.md +81 -507
  40. claude_mpm/commands/mpm-monitor.md +15 -402
  41. claude_mpm/commands/mpm-organize.md +120 -0
  42. claude_mpm/commands/mpm-postmortem.md +6 -108
  43. claude_mpm/commands/mpm-session-resume.md +12 -363
  44. claude_mpm/commands/mpm-status.md +5 -69
  45. claude_mpm/commands/mpm-ticket-view.md +52 -495
  46. claude_mpm/commands/mpm-version.md +5 -107
  47. claude_mpm/config/agent_sources.py +27 -0
  48. claude_mpm/core/config.py +2 -4
  49. claude_mpm/core/framework/formatters/content_formatter.py +3 -13
  50. claude_mpm/core/framework/loaders/agent_loader.py +8 -5
  51. claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
  52. claude_mpm/core/framework_loader.py +4 -2
  53. claude_mpm/core/logger.py +13 -0
  54. claude_mpm/core/optimized_startup.py +59 -0
  55. claude_mpm/core/shared/config_loader.py +1 -1
  56. claude_mpm/core/socketio_pool.py +3 -3
  57. claude_mpm/core/unified_agent_registry.py +5 -15
  58. claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
  59. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +1 -0
  60. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +1 -0
  61. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +1 -0
  62. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +1 -0
  63. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +1 -0
  64. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +2 -0
  65. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DjhvlsAc.js +1 -0
  66. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/N4qtv3Hx.js +2 -0
  67. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/uj46x2Wr.js +1 -0
  68. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +2 -0
  69. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +1 -0
  70. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.CAGBuiOw.js +1 -0
  71. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +1 -0
  72. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +10 -0
  73. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
  74. claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
  75. claude_mpm/dashboard/static/svelte-build/index.html +36 -0
  76. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  77. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  78. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  79. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  80. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  81. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  82. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  83. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  84. claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
  85. claude_mpm/hooks/claude_hooks/event_handlers.py +211 -78
  86. claude_mpm/hooks/claude_hooks/hook_handler.py +155 -1
  87. claude_mpm/hooks/claude_hooks/installer.py +33 -10
  88. claude_mpm/hooks/claude_hooks/memory_integration.py +26 -9
  89. claude_mpm/hooks/claude_hooks/response_tracking.py +2 -3
  90. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  91. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  92. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  93. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  94. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  95. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  96. claude_mpm/hooks/claude_hooks/services/connection_manager.py +30 -6
  97. claude_mpm/hooks/kuzu_memory_hook.py +5 -5
  98. claude_mpm/hooks/memory_integration_hook.py +46 -1
  99. claude_mpm/init.py +63 -19
  100. claude_mpm/models/git_repository.py +3 -3
  101. claude_mpm/scripts/claude-hook-handler.sh +58 -18
  102. claude_mpm/scripts/launch_monitor.py +93 -13
  103. claude_mpm/services/agents/agent_builder.py +3 -3
  104. claude_mpm/services/agents/agent_recommendation_service.py +278 -0
  105. claude_mpm/services/agents/agent_review_service.py +280 -0
  106. claude_mpm/services/agents/cache_git_manager.py +6 -6
  107. claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
  108. claude_mpm/services/agents/deployment/agent_discovery_service.py +4 -5
  109. claude_mpm/services/agents/deployment/agent_format_converter.py +23 -13
  110. claude_mpm/services/agents/deployment/agent_template_builder.py +32 -20
  111. claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
  112. claude_mpm/services/agents/deployment/async_agent_deployment.py +31 -27
  113. claude_mpm/services/agents/deployment/local_template_deployment.py +3 -1
  114. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +247 -35
  115. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +392 -87
  116. claude_mpm/services/agents/git_source_manager.py +53 -4
  117. claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
  118. claude_mpm/services/agents/recommender.py +5 -3
  119. claude_mpm/services/agents/single_tier_deployment_service.py +2 -2
  120. claude_mpm/services/agents/sources/git_source_sync_service.py +120 -7
  121. claude_mpm/services/agents/startup_sync.py +22 -2
  122. claude_mpm/services/agents/toolchain_detector.py +10 -6
  123. claude_mpm/services/analysis/__init__.py +11 -1
  124. claude_mpm/services/analysis/clone_detector.py +1030 -0
  125. claude_mpm/services/command_deployment_service.py +81 -10
  126. claude_mpm/services/diagnostics/checks/agent_check.py +2 -2
  127. claude_mpm/services/diagnostics/checks/agent_sources_check.py +1 -1
  128. claude_mpm/services/event_bus/config.py +3 -1
  129. claude_mpm/services/git/git_operations_service.py +101 -16
  130. claude_mpm/services/monitor/daemon.py +9 -2
  131. claude_mpm/services/monitor/daemon_manager.py +39 -3
  132. claude_mpm/services/monitor/management/lifecycle.py +8 -1
  133. claude_mpm/services/monitor/server.py +698 -22
  134. claude_mpm/services/pm_skills_deployer.py +711 -0
  135. claude_mpm/services/profile_manager.py +331 -0
  136. claude_mpm/services/self_upgrade_service.py +120 -12
  137. claude_mpm/services/skills/__init__.py +3 -0
  138. claude_mpm/services/skills/git_skill_source_manager.py +130 -2
  139. claude_mpm/services/skills/selective_skill_deployer.py +704 -0
  140. claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
  141. claude_mpm/services/skills_deployer.py +127 -9
  142. claude_mpm/services/socketio/dashboard_server.py +1 -0
  143. claude_mpm/services/socketio/event_normalizer.py +51 -6
  144. claude_mpm/services/socketio/server/core.py +386 -108
  145. claude_mpm/services/version_control/git_operations.py +103 -0
  146. claude_mpm/skills/skill_manager.py +92 -3
  147. claude_mpm/utils/agent_dependency_loader.py +14 -2
  148. claude_mpm/utils/agent_filters.py +17 -44
  149. claude_mpm/utils/migration.py +4 -4
  150. claude_mpm/utils/robust_installer.py +47 -3
  151. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/METADATA +53 -87
  152. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/RECORD +157 -197
  153. claude_mpm-5.4.48.dist-info/entry_points.txt +5 -0
  154. claude_mpm-5.4.48.dist-info/licenses/LICENSE +94 -0
  155. claude_mpm-5.4.48.dist-info/licenses/LICENSE-FAQ.md +153 -0
  156. claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
  157. claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
  158. claude_mpm/agents/BASE_OPS.md +0 -219
  159. claude_mpm/agents/BASE_PM.md +0 -480
  160. claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
  161. claude_mpm/agents/BASE_QA.md +0 -167
  162. claude_mpm/agents/BASE_RESEARCH.md +0 -53
  163. claude_mpm/agents/base_agent_loader.py +0 -601
  164. claude_mpm/cli/commands/agents_detect.py +0 -380
  165. claude_mpm/cli/commands/agents_recommend.py +0 -309
  166. claude_mpm/cli/ticket_cli.py +0 -35
  167. claude_mpm/commands/mpm-agents-auto-configure.md +0 -278
  168. claude_mpm/commands/mpm-agents-detect.md +0 -177
  169. claude_mpm/commands/mpm-agents-list.md +0 -131
  170. claude_mpm/commands/mpm-agents-recommend.md +0 -223
  171. claude_mpm/commands/mpm-config-view.md +0 -150
  172. claude_mpm/commands/mpm-ticket-organize.md +0 -304
  173. claude_mpm/dashboard/analysis_runner.py +0 -455
  174. claude_mpm/dashboard/index.html +0 -13
  175. claude_mpm/dashboard/open_dashboard.py +0 -66
  176. claude_mpm/dashboard/static/css/activity.css +0 -1958
  177. claude_mpm/dashboard/static/css/connection-status.css +0 -370
  178. claude_mpm/dashboard/static/css/dashboard.css +0 -4701
  179. claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
  180. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
  181. claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
  182. claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
  183. claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
  184. claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
  185. claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
  186. claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
  187. claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
  188. claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
  189. claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
  190. claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
  191. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
  192. claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
  193. claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
  194. claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
  195. claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
  196. claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
  197. claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
  198. claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
  199. claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
  200. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
  201. claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
  202. claude_mpm/dashboard/static/js/connection-manager.js +0 -536
  203. claude_mpm/dashboard/static/js/dashboard.js +0 -1914
  204. claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
  205. claude_mpm/dashboard/static/js/socket-client.js +0 -1474
  206. claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
  207. claude_mpm/dashboard/static/socket.io.min.js +0 -7
  208. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
  209. claude_mpm/dashboard/templates/code_simple.html +0 -153
  210. claude_mpm/dashboard/templates/index.html +0 -606
  211. claude_mpm/dashboard/test_dashboard.html +0 -372
  212. claude_mpm/scripts/mcp_server.py +0 -75
  213. claude_mpm/scripts/mcp_wrapper.py +0 -39
  214. claude_mpm/services/mcp_gateway/__init__.py +0 -159
  215. claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
  216. claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
  217. claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
  218. claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
  219. claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
  220. claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
  221. claude_mpm/services/mcp_gateway/core/base.py +0 -312
  222. claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
  223. claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
  224. claude_mpm/services/mcp_gateway/core/process_pool.py +0 -977
  225. claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
  226. claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
  227. claude_mpm/services/mcp_gateway/main.py +0 -589
  228. claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
  229. claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
  230. claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
  231. claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
  232. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -414
  233. claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
  234. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -712
  235. claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
  236. claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
  237. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
  238. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
  239. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
  240. claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
  241. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -555
  242. claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
  243. claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
  244. claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
  245. claude_mpm-5.1.9.dist-info/entry_points.txt +0 -10
  246. claude_mpm-5.1.9.dist-info/licenses/LICENSE +0 -21
  247. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/WHEEL +0 -0
  248. {claude_mpm-5.1.9.dist-info → claude_mpm-5.4.48.dist-info}/top_level.txt +0 -0
@@ -1,857 +0,0 @@
1
- // Ultra-simple directory browser - maximum compatibility and debugging
2
- console.log('[code-simple.js] Script loaded at', new Date().toISOString());
3
-
4
- // Global function for onclick handlers
5
- function loadDir(path) {
6
- console.log('[loadDir] Called with path:', path);
7
- if (window.simpleCodeView) {
8
- window.simpleCodeView.loadDirectory(path);
9
- } else {
10
- console.error('[loadDir] simpleCodeView not initialized');
11
- }
12
- }
13
-
14
- function goUp() {
15
- console.log('[goUp] Called');
16
- if (window.simpleCodeView) {
17
- window.simpleCodeView.goUp();
18
- } else {
19
- console.error('[goUp] simpleCodeView not initialized');
20
- }
21
- }
22
-
23
- function analyzeFileFromPath(filePath) {
24
- console.log('[analyzeFileFromPath] Called with path:', filePath);
25
- if (window.simpleCodeView) {
26
- window.simpleCodeView.analyzeFileFromPath(filePath);
27
- } else {
28
- console.error('[analyzeFileFromPath] simpleCodeView not initialized');
29
- }
30
- }
31
-
32
- class SimpleCodeView {
33
- constructor() {
34
- console.log('[SimpleCodeView] Constructor called');
35
- // Try to get the current working directory from various sources
36
- this.currentPath = this.getInitialPath();
37
- this.container = null;
38
- this.apiBase = window.location.origin;
39
- console.log('[SimpleCodeView] API base:', this.apiBase);
40
-
41
- // Tree view properties
42
- this.currentView = 'directory';
43
- this.socket = null;
44
- this.svg = null;
45
- this.treeGroup = null;
46
- this.treeLayout = null;
47
- this.treeData = null;
48
- this.width = 800;
49
- this.height = 600;
50
- this.margin = {top: 20, right: 20, bottom: 20, left: 120};
51
- }
52
-
53
- init(container) {
54
- console.log('[SimpleCodeView.init] Starting with container:', container);
55
-
56
- if (!container) {
57
- console.error('[SimpleCodeView.init] No container provided!');
58
- document.body.innerHTML += '<div style="color:red;font-size:20px;">ERROR: No container for SimpleCodeView</div>';
59
- return;
60
- }
61
-
62
- this.container = container;
63
- this.render();
64
-
65
- // Load initial directory after a short delay to ensure DOM is ready
66
- setTimeout(() => {
67
- console.log('[SimpleCodeView.init] Loading initial directory after delay');
68
- this.loadDirectory(this.currentPath);
69
- }, 100);
70
- }
71
-
72
- render() {
73
- console.log('[SimpleCodeView.render] Rendering UI');
74
-
75
- const html = `
76
- <div class="simple-code-view" style="padding: 20px;">
77
- <h2>Simple Code Browser</h2>
78
-
79
- <div class="view-toggle" style="margin: 10px 0; padding: 10px; background: #f0f0f0; border-radius: 4px;">
80
- <button id="dir-view-btn" onclick="window.simpleCodeView.setView('directory')" class="active"
81
- style="margin-right: 10px; padding: 8px 16px; border: 1px solid #ccc; background: #007cba; color: white; border-radius: 4px; cursor: pointer;">
82
- 📁 Directory View
83
- </button>
84
- <button id="tree-view-btn" onclick="window.simpleCodeView.setView('tree')"
85
- style="padding: 8px 16px; border: 1px solid #ccc; background: #f9f9f9; color: #333; border-radius: 4px; cursor: pointer;">
86
- 🌳 Tree View
87
- </button>
88
- </div>
89
-
90
- <div id="status-bar" style="padding: 10px; background: #e0e0e0; border-radius: 4px; margin-bottom: 10px;">
91
- Status: Initializing...
92
- </div>
93
-
94
- <div class="path-bar" style="margin: 10px 0; padding: 10px; background: #f0f0f0; border-radius: 4px;">
95
- <strong>Current Path:</strong>
96
- <input type="text" id="path-input" value="${this.currentPath}" style="width: 50%; margin: 0 10px;">
97
- <button id="load-btn" onclick="loadDir(document.getElementById('path-input').value)">Load</button>
98
- <button id="up-btn" onclick="goUp()">Go Up</button>
99
- </div>
100
-
101
- <div id="error-display" style="display:none; padding: 10px; background: #fee; color: red; border: 1px solid #fcc; border-radius: 4px; margin: 10px 0;">
102
- </div>
103
-
104
- <div id="directory-contents" style="border: 1px solid #ccc; padding: 10px; min-height: 400px; background: white;">
105
- <div style="color: #666;">Waiting to load directory...</div>
106
- </div>
107
-
108
- <div id="tree-view-container" style="display: none;">
109
- <div class="file-selector" style="margin: 10px 0; padding: 10px; background: #f9f9f9; border-radius: 4px;">
110
- <input type="text" id="file-path-input" placeholder="Enter file path to analyze (e.g., ./src/claude_mpm/core/framework_loader.py)"
111
- style="width: 70%; padding: 8px; margin-right: 10px; border: 1px solid #ccc; border-radius: 4px;">
112
- <button onclick="window.simpleCodeView.analyzeFile()"
113
- style="padding: 8px 16px; border: 1px solid #ccc; background: #28a745; color: white; border-radius: 4px; cursor: pointer;">
114
- Analyze File
115
- </button>
116
- </div>
117
- <div id="tree-visualization" style="border: 1px solid #ccc; min-height: 500px; background: white; overflow: auto; position: relative;">
118
- <div style="padding: 20px; text-align: center; color: #666;">Enter a file path above and click "Analyze File" to view AST tree</div>
119
- </div>
120
- </div>
121
-
122
- <div id="debug-info" style="margin-top: 10px; padding: 10px; background: #f9f9f9; font-family: monospace; font-size: 12px;">
123
- <strong>Debug Info:</strong><br>
124
- API Base: ${this.apiBase}<br>
125
- Current Path: ${this.currentPath}<br>
126
- Status: Waiting for first load...
127
- </div>
128
- </div>
129
- `;
130
-
131
- this.container.innerHTML = html;
132
- console.log('[SimpleCodeView.render] UI rendered');
133
-
134
- this.updateStatus('UI Rendered - Ready to load directory', 'blue');
135
- }
136
-
137
- updateStatus(message, color = 'black') {
138
- console.log('[SimpleCodeView.updateStatus]', message);
139
- const statusBar = document.getElementById('status-bar');
140
- if (statusBar) {
141
- statusBar.innerHTML = `Status: ${message}`;
142
- statusBar.style.color = color;
143
- }
144
- }
145
-
146
- showError(message) {
147
- console.error('[SimpleCodeView.showError]', message);
148
- const errorDiv = document.getElementById('error-display');
149
- if (errorDiv) {
150
- errorDiv.style.display = 'block';
151
- errorDiv.innerHTML = `Error: ${message}`;
152
- }
153
- this.updateStatus('Error occurred', 'red');
154
- }
155
-
156
- hideError() {
157
- const errorDiv = document.getElementById('error-display');
158
- if (errorDiv) {
159
- errorDiv.style.display = 'none';
160
- }
161
- }
162
-
163
- async loadDirectory(path) {
164
- console.log('[SimpleCodeView.loadDirectory] Loading path:', path);
165
-
166
- this.currentPath = path;
167
- this.hideError();
168
- this.updateStatus(`Loading ${path}...`, 'blue');
169
-
170
- // Update path input
171
- const pathInput = document.getElementById('path-input');
172
- if (pathInput) {
173
- pathInput.value = path;
174
- }
175
-
176
- // Update debug info
177
- const debugDiv = document.getElementById('debug-info');
178
- const contentsDiv = document.getElementById('directory-contents');
179
-
180
- const apiUrl = `${this.apiBase}/api/directory/list?path=${encodeURIComponent(path)}`;
181
-
182
- if (debugDiv) {
183
- debugDiv.innerHTML = `
184
- <strong>Debug Info:</strong><br>
185
- API URL: ${apiUrl}<br>
186
- Timestamp: ${new Date().toISOString()}<br>
187
- Status: Fetching...
188
- `;
189
- }
190
-
191
- try {
192
- console.log('[SimpleCodeView.loadDirectory] Fetching:', apiUrl);
193
-
194
- const response = await fetch(apiUrl);
195
- console.log('[SimpleCodeView.loadDirectory] Response status:', response.status);
196
-
197
- if (!response.ok) {
198
- throw new Error(`HTTP ${response.status}: ${response.statusText}`);
199
- }
200
-
201
- const data = await response.json();
202
- console.log('[SimpleCodeView.loadDirectory] Data received:', data);
203
-
204
- // Update debug info with response and filtering status
205
- if (debugDiv) {
206
- let debugContent = `
207
- <strong>Debug Info:</strong><br>
208
- API URL: ${apiUrl}<br>
209
- Response Status: ${response.status}<br>
210
- Path Exists: ${data.exists}<br>
211
- Is Directory: ${data.is_directory}<br>
212
- Item Count: ${data.contents ? data.contents.length : 0}<br>
213
- `;
214
-
215
- // Add filtering information
216
- if (data.filtered) {
217
- debugContent += `<strong>Filtering:</strong> ${data.filter_info || 'Filtered view'}<br>`;
218
- if (data.summary) {
219
- debugContent += `<strong>Items:</strong> ${data.summary.directories} directories, ${data.summary.code_files} code files<br>`;
220
- }
221
- }
222
-
223
- debugContent += `
224
- <details>
225
- <summary>Raw Response (click to expand)</summary>
226
- <pre style="overflow-x: auto;">${JSON.stringify(data, null, 2)}</pre>
227
- </details>
228
- `;
229
-
230
- debugDiv.innerHTML = debugContent;
231
- }
232
-
233
- // Display contents
234
- if (!data.exists) {
235
- contentsDiv.innerHTML = '<p style="color: red;">❌ Path does not exist</p>';
236
- this.updateStatus('Path does not exist', 'red');
237
- } else if (!data.is_directory) {
238
- contentsDiv.innerHTML = '<p style="color: orange;">⚠️ Path is not a directory</p>';
239
- this.updateStatus('Not a directory', 'orange');
240
- } else if (data.error) {
241
- contentsDiv.innerHTML = `<p style="color: red;">❌ Error: ${data.error}</p>`;
242
- this.showError(data.error);
243
- } else if (!data.contents || data.contents.length === 0) {
244
- contentsDiv.innerHTML = '<p style="color: gray;">📭 No code files or subdirectories found (hidden files/folders not shown)</p>';
245
- this.updateStatus('No code content found', 'gray');
246
- } else {
247
- // Build the list with filtering indicator
248
- let headerText = `Found ${data.contents.length} items`;
249
- if (data.filtered && data.summary) {
250
- headerText += ` (${data.summary.directories} directories, ${data.summary.code_files} code files)`;
251
- }
252
- headerText += ':';
253
-
254
- let html = `<div style="margin-bottom: 10px; color: #666;">${headerText}</div>`;
255
-
256
- // Add filtering notice if applicable
257
- if (data.filtered) {
258
- html += `<div style="margin-bottom: 10px; padding: 8px; background: #e8f4fd; border-left: 3px solid #2196f3; color: #1565c0; font-size: 13px;">
259
- 🔍 Filtered view: ${data.filter_info || 'Showing only code-related files and directories'}
260
- </div>`;
261
- }
262
-
263
- html += '<ul style="list-style: none; padding: 0; margin: 0;">';
264
-
265
- // Sort: directories first, then files
266
- const sorted = data.contents.sort((a, b) => {
267
- if (a.is_directory !== b.is_directory) {
268
- return a.is_directory ? -1 : 1;
269
- }
270
- return a.name.localeCompare(b.name);
271
- });
272
-
273
- for (const item of sorted) {
274
- if (item.is_directory) {
275
- // Make directories clickable
276
- html += `<li style="padding: 5px 0;">
277
- 📁 <a href="#" onclick="loadDir('${item.path.replace(/'/g, "\\'")}'); return false;" style="color: blue; text-decoration: none; cursor: pointer;">
278
- ${item.name}/
279
- </a>
280
- </li>`;
281
- } else {
282
- // Check if it's a code file and make it clickable
283
- const isCodeFile = this.isCodeFile(item.name);
284
- const fileIcon = this.getFileIcon(item.name);
285
-
286
- if (isCodeFile) {
287
- // Code files - clickable to show AST
288
- html += `<li style="padding: 5px 0;">
289
- ${fileIcon} <a href="#" onclick="analyzeFileFromPath('${item.path.replace(/'/g, "\\'")}'); return false;" style="color: #0066cc; text-decoration: none; cursor: pointer; font-weight: 500;" title="Click to view AST">
290
- ${item.name}
291
- </a>
292
- </li>`;
293
- } else {
294
- // Non-code files - not clickable
295
- html += `<li style="padding: 5px 0;">
296
- 📄 <span style="color: #666;">${item.name}</span>
297
- </li>`;
298
- }
299
- }
300
- }
301
-
302
- html += '</ul>';
303
- contentsDiv.innerHTML = html;
304
- this.updateStatus(`Loaded ${data.contents.length} items`, 'green');
305
- }
306
-
307
- } catch (error) {
308
- console.error('[SimpleCodeView.loadDirectory] Error:', error);
309
-
310
- const errorMsg = `Failed to load directory: ${error.message}`;
311
- this.showError(errorMsg);
312
-
313
- if (contentsDiv) {
314
- contentsDiv.innerHTML = `
315
- <div style="color: red;">
316
- <p>❌ Failed to load directory</p>
317
- <p>Error: ${error.message}</p>
318
- <p style="font-size: 12px;">Check browser console for details</p>
319
- </div>
320
- `;
321
- }
322
-
323
- if (debugDiv) {
324
- debugDiv.innerHTML += `<br><span style="color:red;">ERROR: ${error.stack || error.message}</span>`;
325
- }
326
- }
327
- }
328
-
329
- goUp() {
330
- console.log('[SimpleCodeView.goUp] Current path:', this.currentPath);
331
- if (this.currentPath === '/' || this.currentPath === '') {
332
- console.log('[SimpleCodeView.goUp] Already at root');
333
- this.updateStatus('Already at root directory', 'orange');
334
- return;
335
- }
336
-
337
- const lastSlash = this.currentPath.lastIndexOf('/');
338
- const parent = lastSlash > 0 ? this.currentPath.substring(0, lastSlash) : '/';
339
- console.log('[SimpleCodeView.goUp] Going up to:', parent);
340
- this.loadDirectory(parent);
341
- }
342
-
343
- // Tree view methods
344
- setView(view) {
345
- console.log('[SimpleCodeView.setView] Switching to view:', view);
346
- this.currentView = view;
347
-
348
- const dirContents = document.getElementById('directory-contents');
349
- const treeContainer = document.getElementById('tree-view-container');
350
- const dirBtn = document.getElementById('dir-view-btn');
351
- const treeBtn = document.getElementById('tree-view-btn');
352
-
353
- if (view === 'tree') {
354
- dirContents.style.display = 'none';
355
- treeContainer.style.display = 'block';
356
-
357
- // Update button styles
358
- dirBtn.style.background = '#f9f9f9';
359
- dirBtn.style.color = '#333';
360
- dirBtn.classList.remove('active');
361
-
362
- treeBtn.style.background = '#007cba';
363
- treeBtn.style.color = 'white';
364
- treeBtn.classList.add('active');
365
-
366
- this.initializeTreeView();
367
- } else {
368
- dirContents.style.display = 'block';
369
- treeContainer.style.display = 'none';
370
-
371
- // Update button styles
372
- treeBtn.style.background = '#f9f9f9';
373
- treeBtn.style.color = '#333';
374
- treeBtn.classList.remove('active');
375
-
376
- dirBtn.style.background = '#007cba';
377
- dirBtn.style.color = 'white';
378
- dirBtn.classList.add('active');
379
- }
380
- }
381
-
382
- initializeTreeView() {
383
- console.log('[SimpleCodeView.initializeTreeView] Initializing tree view');
384
-
385
- if (!window.d3) {
386
- this.updateStatus('D3.js not loaded - cannot initialize tree view', 'red');
387
- return;
388
- }
389
-
390
- this.initializeSocket();
391
- }
392
-
393
- initializeSocket() {
394
- console.log('[SimpleCodeView.initializeSocket] Initializing Socket.IO connection');
395
-
396
- if (!window.io) {
397
- this.updateStatus('Socket.IO not loaded - using fallback mode', 'orange');
398
- return;
399
- }
400
-
401
- try {
402
- this.socket = io('/');
403
-
404
- this.socket.on('connect', () => {
405
- console.log('[SimpleCodeView] Socket connected');
406
- this.updateStatus('Connected to analysis server', 'green');
407
- });
408
-
409
- this.socket.on('disconnect', () => {
410
- console.log('[SimpleCodeView] Socket disconnected');
411
- this.updateStatus('Disconnected from analysis server', 'orange');
412
- });
413
-
414
- this.socket.on('analysis_progress', (data) => {
415
- console.log('[SimpleCodeView] Analysis progress:', data);
416
- this.updateStatus(`Analysis: ${data.message}`, 'blue');
417
- });
418
-
419
- this.socket.on('analysis_complete', (data) => {
420
- console.log('[SimpleCodeView] Analysis complete:', data);
421
- this.updateStatus('Analysis complete - rendering tree', 'green');
422
- this.renderTree(data);
423
- });
424
-
425
- this.socket.on('analysis_error', (error) => {
426
- console.error('[SimpleCodeView] Analysis error:', error);
427
- this.showError(`Analysis failed: ${error.message || error}`);
428
- });
429
-
430
- } catch (error) {
431
- console.error('[SimpleCodeView] Failed to initialize socket:', error);
432
- this.updateStatus('Socket connection failed - using fallback mode', 'orange');
433
- }
434
- }
435
-
436
- async analyzeFile() {
437
- console.log('[SimpleCodeView.analyzeFile] Starting file analysis');
438
-
439
- const fileInput = document.getElementById('file-path-input');
440
- const filePath = fileInput.value.trim();
441
-
442
- if (!filePath) {
443
- this.showError('Please enter a file path');
444
- return;
445
- }
446
-
447
- this.hideError();
448
- this.updateStatus(`Analyzing file: ${filePath}`, 'blue');
449
-
450
- // Clear previous tree
451
- const treeViz = document.getElementById('tree-visualization');
452
- if (treeViz) {
453
- treeViz.innerHTML = '<div style="padding: 20px; text-align: center; color: #666;">Analyzing file...</div>';
454
- }
455
-
456
- try {
457
- if (this.socket && this.socket.connected) {
458
- // Use Socket.IO for real-time analysis
459
- this.socket.emit('analyze_file', {
460
- path: filePath,
461
- working_directory: this.currentPath
462
- });
463
- } else {
464
- // Fallback to HTTP API (we'll create a simple endpoint)
465
- await this.analyzeFileHTTP(filePath);
466
- }
467
- } catch (error) {
468
- console.error('[SimpleCodeView.analyzeFile] Error:', error);
469
- this.showError(`Failed to analyze file: ${error.message}`);
470
- }
471
- }
472
-
473
- async analyzeFileHTTP(filePath) {
474
- console.log('[SimpleCodeView.analyzeFileHTTP] Using HTTP fallback for:', filePath);
475
-
476
- try {
477
- // Create a simple mock analysis for demonstration
478
- // In a real implementation, this would call a proper analysis endpoint
479
- setTimeout(() => {
480
- const mockData = this.createMockTreeData(filePath);
481
- this.renderTree(mockData);
482
- }, 1000);
483
-
484
- } catch (error) {
485
- throw new Error(`HTTP analysis failed: ${error.message}`);
486
- }
487
- }
488
-
489
- createMockTreeData(filePath) {
490
- // Create mock AST data for demonstration
491
- const fileName = filePath.split('/').pop() || 'file';
492
- const ext = fileName.split('.').pop()?.toLowerCase();
493
-
494
- let mockData = {
495
- name: fileName,
496
- type: 'module',
497
- children: []
498
- };
499
-
500
- if (ext === 'py') {
501
- mockData.children = [
502
- {
503
- name: 'imports',
504
- type: 'imports',
505
- children: [
506
- { name: 'import os', type: 'import' },
507
- { name: 'from pathlib import Path', type: 'import' }
508
- ]
509
- },
510
- {
511
- name: 'MyClass',
512
- type: 'class',
513
- children: [
514
- { name: '__init__', type: 'method' },
515
- { name: 'process_data', type: 'method' },
516
- { name: 'save_results', type: 'method' }
517
- ]
518
- },
519
- {
520
- name: 'helper_function',
521
- type: 'function',
522
- children: []
523
- }
524
- ];
525
- } else if (ext === 'js' || ext === 'ts') {
526
- mockData.children = [
527
- {
528
- name: 'imports',
529
- type: 'imports',
530
- children: [
531
- { name: "import React from 'react'", type: 'import' },
532
- { name: "import { useState } from 'react'", type: 'import' }
533
- ]
534
- },
535
- {
536
- name: 'MyComponent',
537
- type: 'function',
538
- children: [
539
- { name: 'useState', type: 'hook' },
540
- { name: 'useEffect', type: 'hook' },
541
- { name: 'handleClick', type: 'function' }
542
- ]
543
- }
544
- ];
545
- } else {
546
- mockData.children = [
547
- { name: 'Content Section 1', type: 'section' },
548
- { name: 'Content Section 2', type: 'section' },
549
- { name: 'Content Section 3', type: 'section' }
550
- ];
551
- }
552
-
553
- return mockData;
554
- }
555
-
556
- renderTree(data) {
557
- console.log('[SimpleCodeView.renderTree] Rendering tree with data:', data);
558
-
559
- if (!data || !window.d3) {
560
- this.showError('Cannot render tree: missing data or D3.js');
561
- return;
562
- }
563
-
564
- this.treeData = data;
565
-
566
- // Clear previous visualization
567
- const container = document.getElementById('tree-visualization');
568
- if (!container) {
569
- this.showError('Tree visualization container not found');
570
- return;
571
- }
572
-
573
- container.innerHTML = '';
574
-
575
- // Create SVG
576
- const margin = this.margin;
577
- const width = this.width - margin.left - margin.right;
578
- const height = this.height - margin.top - margin.bottom;
579
-
580
- this.svg = d3.select('#tree-visualization')
581
- .append('svg')
582
- .attr('width', this.width)
583
- .attr('height', this.height);
584
-
585
- // Add zoom behavior
586
- const zoom = d3.zoom()
587
- .scaleExtent([0.1, 4])
588
- .on('zoom', (event) => {
589
- this.treeGroup.attr('transform', event.transform);
590
- });
591
-
592
- this.svg.call(zoom);
593
-
594
- // Create main group
595
- this.treeGroup = this.svg.append('g')
596
- .attr('transform', `translate(${margin.left},${margin.top})`);
597
-
598
- // Create tree layout
599
- this.treeLayout = d3.tree()
600
- .size([height, width]);
601
-
602
- // Convert data to hierarchy
603
- const hierarchy = d3.hierarchy(data);
604
-
605
- // Generate tree layout
606
- const treeData = this.treeLayout(hierarchy);
607
-
608
- // Add links
609
- const links = this.treeGroup.selectAll('.link')
610
- .data(treeData.links())
611
- .enter()
612
- .append('path')
613
- .attr('class', 'link')
614
- .attr('d', d3.linkHorizontal()
615
- .x(d => d.y)
616
- .y(d => d.x)
617
- )
618
- .style('fill', 'none')
619
- .style('stroke', '#ccc')
620
- .style('stroke-width', '2px');
621
-
622
- // Add nodes
623
- const nodes = this.treeGroup.selectAll('.node')
624
- .data(treeData.descendants())
625
- .enter()
626
- .append('g')
627
- .attr('class', 'tree-node')
628
- .attr('transform', d => `translate(${d.y},${d.x})`);
629
-
630
- // Add circles for nodes
631
- nodes.append('circle')
632
- .attr('r', 6)
633
- .style('fill', d => this.getNodeColor(d.data.type))
634
- .style('stroke', '#333')
635
- .style('stroke-width', '2px');
636
-
637
- // Add labels
638
- nodes.append('text')
639
- .attr('dy', '.35em')
640
- .attr('x', d => d.children ? -13 : 13)
641
- .style('text-anchor', d => d.children ? 'end' : 'start')
642
- .style('font-size', '12px')
643
- .style('font-family', 'Arial, sans-serif')
644
- .text(d => d.data.name);
645
-
646
- // Add tooltips
647
- nodes.append('title')
648
- .text(d => `${d.data.type}: ${d.data.name}`);
649
-
650
- // Add legend
651
- this.addLegend(container);
652
-
653
- this.updateStatus('Tree visualization rendered successfully', 'green');
654
- }
655
-
656
- getNodeColor(type) {
657
- const colors = {
658
- 'module': '#1f77b4',
659
- 'class': '#ff7f0e',
660
- 'function': '#2ca02c',
661
- 'method': '#d62728',
662
- 'import': '#9467bd',
663
- 'imports': '#8c564b',
664
- 'section': '#e377c2',
665
- 'hook': '#7f7f7f'
666
- };
667
- return colors[type] || '#bcbd22';
668
- }
669
-
670
- // Check if file is a code file
671
- isCodeFile(filename) {
672
- const codeExtensions = [
673
- '.py', '.js', '.jsx', '.ts', '.tsx', '.java', '.cpp', '.c', '.h',
674
- '.cs', '.go', '.rs', '.rb', '.php', '.swift', '.kt', '.scala',
675
- '.r', '.m', '.mm', '.sh', '.bash', '.zsh', '.sql', '.html',
676
- '.css', '.scss', '.sass', '.less', '.xml', '.json', '.yaml', '.yml',
677
- '.md', '.rst', '.txt', '.log', '.conf', '.ini', '.toml'
678
- ];
679
- const ext = filename.toLowerCase().substring(filename.lastIndexOf('.'));
680
- return codeExtensions.includes(ext);
681
- }
682
-
683
- // Get appropriate icon for file type
684
- getFileIcon(filename) {
685
- const ext = filename.toLowerCase().substring(filename.lastIndexOf('.'));
686
- const iconMap = {
687
- '.py': '🐍',
688
- '.js': '📜',
689
- '.jsx': '⚛️',
690
- '.ts': '📘',
691
- '.tsx': '⚛️',
692
- '.json': '📋',
693
- '.html': '🌐',
694
- '.css': '🎨',
695
- '.md': '📝',
696
- '.yml': '⚙️',
697
- '.yaml': '⚙️',
698
- '.sh': '🔧',
699
- '.go': '🐹',
700
- '.rs': '🦀',
701
- '.java': '☕',
702
- '.rb': '💎',
703
- '.php': '🐘',
704
- '.cpp': '⚙️',
705
- '.c': '⚙️',
706
- '.h': '📄',
707
- '.cs': '💜',
708
- '.swift': '🦉',
709
- '.kt': '🚀',
710
- '.scala': '📈',
711
- '.r': '📊',
712
- '.sql': '🗃️',
713
- '.scss': '🎨',
714
- '.sass': '🎨',
715
- '.less': '🎨',
716
- '.xml': '📑',
717
- '.bash': '🔧',
718
- '.zsh': '🔧',
719
- '.md': '📝',
720
- '.rst': '📝',
721
- '.txt': '📄',
722
- '.log': '📋',
723
- '.conf': '⚙️',
724
- '.ini': '⚙️',
725
- '.toml': '⚙️'
726
- };
727
- return iconMap[ext] || '💻';
728
- }
729
-
730
- // New method to analyze file from directory click
731
- analyzeFileFromPath(filePath) {
732
- console.log('[SimpleCodeView.analyzeFileFromPath] Analyzing file:', filePath);
733
-
734
- // Switch to tree view
735
- this.setView('tree');
736
-
737
- // Set the file path in the input
738
- const fileInput = document.getElementById('file-path-input');
739
- if (fileInput) {
740
- fileInput.value = filePath;
741
- }
742
-
743
- // Trigger analysis
744
- this.analyzeFile();
745
- }
746
-
747
- addLegend(container) {
748
- // Remove existing legend
749
- const existingLegend = container.querySelector('.tree-legend');
750
- if (existingLegend) {
751
- existingLegend.remove();
752
- }
753
-
754
- const legend = document.createElement('div');
755
- legend.className = 'tree-legend';
756
-
757
- const legendTypes = [
758
- { type: 'module', label: 'Module/File' },
759
- { type: 'class', label: 'Class' },
760
- { type: 'function', label: 'Function' },
761
- { type: 'method', label: 'Method' },
762
- { type: 'imports', label: 'Imports' },
763
- { type: 'import', label: 'Import' },
764
- { type: 'section', label: 'Section' },
765
- { type: 'hook', label: 'Hook/Other' }
766
- ];
767
-
768
- legend.innerHTML = `
769
- <strong>Legend</strong><br>
770
- ${legendTypes.map(item => `
771
- <div class="tree-legend-item">
772
- <div class="tree-legend-color" style="background-color: ${this.getNodeColor(item.type)};"></div>
773
- ${item.label}
774
- </div>
775
- `).join('')}
776
- <hr style="margin: 8px 0;">
777
- <div style="font-size: 11px; color: #666;">
778
- Zoom: Mouse wheel<br>
779
- Pan: Click and drag
780
- </div>
781
- `;
782
-
783
- container.appendChild(legend);
784
- }
785
-
786
- /**
787
- * Get initial path from various sources
788
- * @returns {string} Initial path to use
789
- */
790
- getInitialPath() {
791
- // Try to get from working directory manager
792
- if (window.dashboard && window.dashboard.workingDirectoryManager) {
793
- const dir = window.dashboard.workingDirectoryManager.getCurrentWorkingDir();
794
- if (dir) return dir;
795
- }
796
-
797
- // Try to get from working directory element
798
- const workingDirPath = document.getElementById('working-dir-path');
799
- if (workingDirPath && workingDirPath.textContent && workingDirPath.textContent !== 'Loading...') {
800
- return workingDirPath.textContent.trim();
801
- }
802
-
803
- // Try to get from footer
804
- const footerDir = document.getElementById('footer-working-dir');
805
- if (footerDir && footerDir.textContent && footerDir.textContent !== 'Unknown') {
806
- return footerDir.textContent.trim();
807
- }
808
-
809
- // Try to get from recent events
810
- if (window.socketClient && window.socketClient.events) {
811
- const eventsWithDir = window.socketClient.events
812
- .filter(e => e.data && (e.data.working_directory || e.data.cwd || e.data.working_dir))
813
- .reverse();
814
-
815
- if (eventsWithDir.length > 0) {
816
- const recentEvent = eventsWithDir[0];
817
- const dir = recentEvent.data.working_directory ||
818
- recentEvent.data.cwd ||
819
- recentEvent.data.working_dir;
820
- if (dir) return dir;
821
- }
822
- }
823
-
824
- // Default fallback
825
- return '/';
826
- }
827
- }
828
-
829
- // Create global instance
830
- console.log('[code-simple.js] Creating global simpleCodeView instance');
831
- window.simpleCodeView = new SimpleCodeView();
832
-
833
- // Auto-initialize when DOM is ready
834
- if (document.readyState === 'loading') {
835
- console.log('[code-simple.js] DOM still loading, waiting for DOMContentLoaded');
836
- document.addEventListener('DOMContentLoaded', () => {
837
- console.log('[code-simple.js] DOMContentLoaded fired');
838
- const container = document.getElementById('code-container');
839
- if (container) {
840
- window.simpleCodeView.init(container);
841
- } else {
842
- console.error('[code-simple.js] No code-container element found!');
843
- }
844
- });
845
- } else {
846
- console.log('[code-simple.js] DOM already loaded, initializing immediately');
847
- setTimeout(() => {
848
- const container = document.getElementById('code-container');
849
- if (container) {
850
- window.simpleCodeView.init(container);
851
- } else {
852
- console.error('[code-simple.js] No code-container element found!');
853
- }
854
- }, 0);
855
- }
856
-
857
- console.log('[code-simple.js] Script setup complete');