claude-mpm 4.2.44__py3-none-any.whl → 4.2.51__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.
Files changed (148) hide show
  1. claude_mpm/VERSION +1 -1
  2. claude_mpm/agents/BASE_PM.md +43 -1
  3. claude_mpm/agents/INSTRUCTIONS.md +75 -1
  4. claude_mpm/agents/WORKFLOW.md +46 -1
  5. claude_mpm/agents/frontmatter_validator.py +20 -12
  6. claude_mpm/agents/templates/nextjs_engineer.json +277 -0
  7. claude_mpm/agents/templates/python_engineer.json +289 -0
  8. claude_mpm/agents/templates/react_engineer.json +11 -3
  9. claude_mpm/agents/templates/security.json +50 -9
  10. claude_mpm/cli/commands/agents.py +2 -2
  11. claude_mpm/cli/commands/uninstall.py +1 -2
  12. claude_mpm/cli/interactive/agent_wizard.py +3 -3
  13. claude_mpm/cli/parsers/agent_manager_parser.py +3 -3
  14. claude_mpm/cli/parsers/agents_parser.py +1 -1
  15. claude_mpm/constants.py +1 -1
  16. claude_mpm/core/error_handler.py +2 -4
  17. claude_mpm/core/file_utils.py +4 -12
  18. claude_mpm/core/log_manager.py +8 -5
  19. claude_mpm/core/logger.py +1 -1
  20. claude_mpm/core/logging_utils.py +6 -6
  21. claude_mpm/core/unified_agent_registry.py +18 -4
  22. claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +188 -0
  23. claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +156 -0
  24. claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +38 -0
  25. claude_mpm/dashboard/react/components/shared/FilterBar.module.css +92 -0
  26. claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +248 -0
  27. claude_mpm/dashboard/static/archive/activity_dashboard_test.html +61 -0
  28. claude_mpm/dashboard/static/archive/test_activity_connection.html +179 -0
  29. claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +68 -0
  30. claude_mpm/dashboard/static/archive/test_dashboard.html +409 -0
  31. claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +519 -0
  32. claude_mpm/dashboard/static/archive/test_dashboard_verification.html +181 -0
  33. claude_mpm/dashboard/static/archive/test_file_data.html +315 -0
  34. claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +243 -0
  35. claude_mpm/dashboard/static/archive/test_file_tree_fix.html +234 -0
  36. claude_mpm/dashboard/static/archive/test_file_tree_rename.html +117 -0
  37. claude_mpm/dashboard/static/archive/test_file_tree_tab.html +115 -0
  38. claude_mpm/dashboard/static/archive/test_file_viewer.html +224 -0
  39. claude_mpm/dashboard/static/archive/test_final_activity.html +220 -0
  40. claude_mpm/dashboard/static/archive/test_tab_fix.html +139 -0
  41. claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +1 -0
  42. claude_mpm/dashboard/static/built/components/activity-tree.js +1 -1
  43. claude_mpm/dashboard/static/built/components/agent-hierarchy.js +777 -0
  44. claude_mpm/dashboard/static/built/components/agent-inference.js +1 -1
  45. claude_mpm/dashboard/static/built/components/build-tracker.js +333 -0
  46. claude_mpm/dashboard/static/built/components/code-simple.js +857 -0
  47. claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +353 -0
  48. claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +235 -0
  49. claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +409 -0
  50. claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +435 -0
  51. claude_mpm/dashboard/static/built/components/code-viewer.js +2 -1076
  52. claude_mpm/dashboard/static/built/components/connection-debug.js +654 -0
  53. claude_mpm/dashboard/static/built/components/diff-viewer.js +891 -0
  54. claude_mpm/dashboard/static/built/components/event-processor.js +1 -1
  55. claude_mpm/dashboard/static/built/components/event-viewer.js +1 -1
  56. claude_mpm/dashboard/static/built/components/export-manager.js +1 -1
  57. claude_mpm/dashboard/static/built/components/file-change-tracker.js +443 -0
  58. claude_mpm/dashboard/static/built/components/file-change-viewer.js +690 -0
  59. claude_mpm/dashboard/static/built/components/file-tool-tracker.js +1 -1
  60. claude_mpm/dashboard/static/built/components/module-viewer.js +1 -1
  61. claude_mpm/dashboard/static/built/components/nav-bar.js +145 -0
  62. claude_mpm/dashboard/static/built/components/page-structure.js +429 -0
  63. claude_mpm/dashboard/static/built/components/session-manager.js +1 -1
  64. claude_mpm/dashboard/static/built/components/ui-state-manager.js +2 -465
  65. claude_mpm/dashboard/static/built/components/working-directory.js +1 -1
  66. claude_mpm/dashboard/static/built/connection-manager.js +536 -0
  67. claude_mpm/dashboard/static/built/dashboard.js +1 -1
  68. claude_mpm/dashboard/static/built/extension-error-handler.js +164 -0
  69. claude_mpm/dashboard/static/built/react/events.js +30 -0
  70. claude_mpm/dashboard/static/built/shared/dom-helpers.js +396 -0
  71. claude_mpm/dashboard/static/built/shared/event-bus.js +330 -0
  72. claude_mpm/dashboard/static/built/shared/event-filter-service.js +540 -0
  73. claude_mpm/dashboard/static/built/shared/logger.js +385 -0
  74. claude_mpm/dashboard/static/built/shared/page-structure.js +251 -0
  75. claude_mpm/dashboard/static/built/shared/tooltip-service.js +253 -0
  76. claude_mpm/dashboard/static/built/socket-client.js +1 -1
  77. claude_mpm/dashboard/static/built/tab-isolation-fix.js +185 -0
  78. claude_mpm/dashboard/static/css/dashboard.css +28 -5
  79. claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +1 -0
  80. claude_mpm/dashboard/static/dist/components/activity-tree.js +1 -1
  81. claude_mpm/dashboard/static/dist/components/agent-inference.js +1 -1
  82. claude_mpm/dashboard/static/dist/components/code-viewer.js +2 -0
  83. claude_mpm/dashboard/static/dist/components/event-processor.js +1 -1
  84. claude_mpm/dashboard/static/dist/components/event-viewer.js +1 -1
  85. claude_mpm/dashboard/static/dist/components/export-manager.js +1 -1
  86. claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +1 -1
  87. claude_mpm/dashboard/static/dist/components/module-viewer.js +1 -1
  88. claude_mpm/dashboard/static/dist/components/session-manager.js +1 -1
  89. claude_mpm/dashboard/static/dist/components/working-directory.js +1 -1
  90. claude_mpm/dashboard/static/dist/dashboard.js +1 -1
  91. claude_mpm/dashboard/static/dist/react/events.js +30 -0
  92. claude_mpm/dashboard/static/dist/socket-client.js +1 -1
  93. claude_mpm/dashboard/static/events.html +607 -0
  94. claude_mpm/dashboard/static/index.html +713 -0
  95. claude_mpm/dashboard/static/js/components/activity-tree.js +3 -17
  96. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +4 -1
  97. claude_mpm/dashboard/static/js/components/agent-inference.js +3 -0
  98. claude_mpm/dashboard/static/js/components/build-tracker.js +8 -0
  99. claude_mpm/dashboard/static/js/components/code-viewer.js +306 -66
  100. claude_mpm/dashboard/static/js/components/event-processor.js +3 -0
  101. claude_mpm/dashboard/static/js/components/event-viewer.js +39 -2
  102. claude_mpm/dashboard/static/js/components/export-manager.js +3 -0
  103. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +30 -10
  104. claude_mpm/dashboard/static/js/components/socket-manager.js +4 -0
  105. claude_mpm/dashboard/static/js/components/ui-state-manager.js +285 -85
  106. claude_mpm/dashboard/static/js/components/working-directory.js +3 -0
  107. claude_mpm/dashboard/static/js/dashboard.js +61 -33
  108. claude_mpm/dashboard/static/js/socket-client.js +12 -8
  109. claude_mpm/dashboard/static/js/stores/dashboard-store.js +562 -0
  110. claude_mpm/dashboard/static/js/tab-isolation-fix.js +185 -0
  111. claude_mpm/dashboard/static/legacy/activity.html +736 -0
  112. claude_mpm/dashboard/static/legacy/agents.html +786 -0
  113. claude_mpm/dashboard/static/legacy/files.html +747 -0
  114. claude_mpm/dashboard/static/legacy/tools.html +831 -0
  115. claude_mpm/dashboard/static/monitors-index.html +218 -0
  116. claude_mpm/dashboard/static/monitors.html +431 -0
  117. claude_mpm/dashboard/static/production/events.html +659 -0
  118. claude_mpm/dashboard/static/production/main.html +715 -0
  119. claude_mpm/dashboard/static/production/monitors.html +483 -0
  120. claude_mpm/dashboard/static/socket.io.min.js +7 -0
  121. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +7 -0
  122. claude_mpm/dashboard/static/test-archive/dashboard.html +635 -0
  123. claude_mpm/dashboard/static/test-archive/debug-events.html +147 -0
  124. claude_mpm/dashboard/static/test-archive/test-navigation.html +256 -0
  125. claude_mpm/dashboard/static/test-archive/test-react-exports.html +180 -0
  126. claude_mpm/dashboard/templates/index.html +79 -9
  127. claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +1 -1
  128. claude_mpm/services/agents/deployment/agent_discovery_service.py +3 -0
  129. claude_mpm/services/agents/deployment/agent_template_builder.py +25 -8
  130. claude_mpm/services/agents/deployment/agent_validator.py +3 -0
  131. claude_mpm/services/agents/deployment/validation/template_validator.py +13 -4
  132. claude_mpm/services/agents/local_template_manager.py +2 -6
  133. claude_mpm/services/monitor/daemon.py +1 -2
  134. claude_mpm/services/monitor/daemon_manager.py +2 -5
  135. claude_mpm/services/monitor/event_emitter.py +2 -2
  136. claude_mpm/services/monitor/handlers/code_analysis.py +4 -6
  137. claude_mpm/services/monitor/handlers/hooks.py +2 -4
  138. claude_mpm/services/monitor/server.py +27 -4
  139. claude_mpm/tools/code_tree_analyzer.py +2 -2
  140. {claude_mpm-4.2.44.dist-info → claude_mpm-4.2.51.dist-info}/METADATA +1 -1
  141. {claude_mpm-4.2.44.dist-info → claude_mpm-4.2.51.dist-info}/RECORD +146 -81
  142. claude_mpm/dashboard/static/test-browser-monitor.html +0 -470
  143. claude_mpm/dashboard/static/test-simple.html +0 -97
  144. /claude_mpm/dashboard/static/{test_debug.html → test-archive/test_debug.html} +0 -0
  145. {claude_mpm-4.2.44.dist-info → claude_mpm-4.2.51.dist-info}/WHEEL +0 -0
  146. {claude_mpm-4.2.44.dist-info → claude_mpm-4.2.51.dist-info}/entry_points.txt +0 -0
  147. {claude_mpm-4.2.44.dist-info → claude_mpm-4.2.51.dist-info}/licenses/LICENSE +0 -0
  148. {claude_mpm-4.2.44.dist-info → claude_mpm-4.2.51.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,857 @@
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');