claude-mpm 4.2.43__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.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/BASE_PM.md +117 -12
- claude_mpm/agents/INSTRUCTIONS.md +154 -10
- claude_mpm/agents/WORKFLOW.md +46 -1
- claude_mpm/agents/frontmatter_validator.py +20 -12
- claude_mpm/agents/templates/nextjs_engineer.json +277 -0
- claude_mpm/agents/templates/python_engineer.json +289 -0
- claude_mpm/agents/templates/react_engineer.json +11 -3
- claude_mpm/agents/templates/security.json +50 -9
- claude_mpm/cli/commands/agents.py +2 -2
- claude_mpm/cli/commands/uninstall.py +1 -2
- claude_mpm/cli/interactive/agent_wizard.py +3 -3
- claude_mpm/cli/parsers/agent_manager_parser.py +3 -3
- claude_mpm/cli/parsers/agents_parser.py +1 -1
- claude_mpm/constants.py +1 -1
- claude_mpm/core/api_validator.py +330 -0
- claude_mpm/core/error_handler.py +2 -4
- claude_mpm/core/file_utils.py +4 -12
- claude_mpm/core/framework_loader.py +22 -0
- claude_mpm/core/log_manager.py +8 -5
- claude_mpm/core/logger.py +1 -1
- claude_mpm/core/logging_utils.py +6 -6
- claude_mpm/core/unified_agent_registry.py +18 -4
- claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +188 -0
- claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +156 -0
- claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +38 -0
- claude_mpm/dashboard/react/components/shared/FilterBar.module.css +92 -0
- claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +248 -0
- claude_mpm/dashboard/static/archive/activity_dashboard_test.html +61 -0
- claude_mpm/dashboard/static/archive/test_activity_connection.html +179 -0
- claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +68 -0
- claude_mpm/dashboard/static/archive/test_dashboard.html +409 -0
- claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +519 -0
- claude_mpm/dashboard/static/archive/test_dashboard_verification.html +181 -0
- claude_mpm/dashboard/static/archive/test_file_data.html +315 -0
- claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +243 -0
- claude_mpm/dashboard/static/archive/test_file_tree_fix.html +234 -0
- claude_mpm/dashboard/static/archive/test_file_tree_rename.html +117 -0
- claude_mpm/dashboard/static/archive/test_file_tree_tab.html +115 -0
- claude_mpm/dashboard/static/archive/test_file_viewer.html +224 -0
- claude_mpm/dashboard/static/archive/test_final_activity.html +220 -0
- claude_mpm/dashboard/static/archive/test_tab_fix.html +139 -0
- claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +1 -0
- claude_mpm/dashboard/static/built/components/activity-tree.js +1 -1
- claude_mpm/dashboard/static/built/components/agent-hierarchy.js +777 -0
- claude_mpm/dashboard/static/built/components/agent-inference.js +1 -1
- claude_mpm/dashboard/static/built/components/build-tracker.js +333 -0
- claude_mpm/dashboard/static/built/components/code-simple.js +857 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +353 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +235 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +409 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +435 -0
- claude_mpm/dashboard/static/built/components/code-viewer.js +2 -1076
- claude_mpm/dashboard/static/built/components/connection-debug.js +654 -0
- claude_mpm/dashboard/static/built/components/diff-viewer.js +891 -0
- claude_mpm/dashboard/static/built/components/event-processor.js +1 -1
- claude_mpm/dashboard/static/built/components/event-viewer.js +1 -1
- claude_mpm/dashboard/static/built/components/export-manager.js +1 -1
- claude_mpm/dashboard/static/built/components/file-change-tracker.js +443 -0
- claude_mpm/dashboard/static/built/components/file-change-viewer.js +690 -0
- claude_mpm/dashboard/static/built/components/file-tool-tracker.js +1 -1
- claude_mpm/dashboard/static/built/components/module-viewer.js +1 -1
- claude_mpm/dashboard/static/built/components/nav-bar.js +145 -0
- claude_mpm/dashboard/static/built/components/page-structure.js +429 -0
- claude_mpm/dashboard/static/built/components/session-manager.js +1 -1
- claude_mpm/dashboard/static/built/components/ui-state-manager.js +2 -465
- claude_mpm/dashboard/static/built/components/working-directory.js +1 -1
- claude_mpm/dashboard/static/built/connection-manager.js +536 -0
- claude_mpm/dashboard/static/built/dashboard.js +1 -1
- claude_mpm/dashboard/static/built/extension-error-handler.js +164 -0
- claude_mpm/dashboard/static/built/react/events.js +30 -0
- claude_mpm/dashboard/static/built/shared/dom-helpers.js +396 -0
- claude_mpm/dashboard/static/built/shared/event-bus.js +330 -0
- claude_mpm/dashboard/static/built/shared/event-filter-service.js +540 -0
- claude_mpm/dashboard/static/built/shared/logger.js +385 -0
- claude_mpm/dashboard/static/built/shared/page-structure.js +251 -0
- claude_mpm/dashboard/static/built/shared/tooltip-service.js +253 -0
- claude_mpm/dashboard/static/built/socket-client.js +1 -1
- claude_mpm/dashboard/static/built/tab-isolation-fix.js +185 -0
- claude_mpm/dashboard/static/css/dashboard.css +28 -5
- claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +1 -0
- claude_mpm/dashboard/static/dist/components/activity-tree.js +1 -1
- claude_mpm/dashboard/static/dist/components/agent-inference.js +1 -1
- claude_mpm/dashboard/static/dist/components/code-viewer.js +2 -0
- claude_mpm/dashboard/static/dist/components/event-processor.js +1 -1
- claude_mpm/dashboard/static/dist/components/event-viewer.js +1 -1
- claude_mpm/dashboard/static/dist/components/export-manager.js +1 -1
- claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +1 -1
- claude_mpm/dashboard/static/dist/components/module-viewer.js +1 -1
- claude_mpm/dashboard/static/dist/components/session-manager.js +1 -1
- claude_mpm/dashboard/static/dist/components/working-directory.js +1 -1
- claude_mpm/dashboard/static/dist/dashboard.js +1 -1
- claude_mpm/dashboard/static/dist/react/events.js +30 -0
- claude_mpm/dashboard/static/dist/socket-client.js +1 -1
- claude_mpm/dashboard/static/events.html +607 -0
- claude_mpm/dashboard/static/index.html +713 -0
- claude_mpm/dashboard/static/js/components/activity-tree.js +3 -17
- claude_mpm/dashboard/static/js/components/agent-hierarchy.js +4 -1
- claude_mpm/dashboard/static/js/components/agent-inference.js +3 -0
- claude_mpm/dashboard/static/js/components/build-tracker.js +8 -0
- claude_mpm/dashboard/static/js/components/code-viewer.js +387 -72
- claude_mpm/dashboard/static/js/components/event-processor.js +3 -0
- claude_mpm/dashboard/static/js/components/event-viewer.js +39 -2
- claude_mpm/dashboard/static/js/components/export-manager.js +3 -0
- claude_mpm/dashboard/static/js/components/file-tool-tracker.js +30 -10
- claude_mpm/dashboard/static/js/components/socket-manager.js +4 -0
- claude_mpm/dashboard/static/js/components/ui-state-manager.js +286 -108
- claude_mpm/dashboard/static/js/components/working-directory.js +3 -0
- claude_mpm/dashboard/static/js/dashboard.js +61 -49
- claude_mpm/dashboard/static/js/socket-client.js +12 -8
- claude_mpm/dashboard/static/js/stores/dashboard-store.js +562 -0
- claude_mpm/dashboard/static/js/tab-isolation-fix.js +185 -0
- claude_mpm/dashboard/static/legacy/activity.html +736 -0
- claude_mpm/dashboard/static/legacy/agents.html +786 -0
- claude_mpm/dashboard/static/legacy/files.html +747 -0
- claude_mpm/dashboard/static/legacy/tools.html +831 -0
- claude_mpm/dashboard/static/monitors-index.html +218 -0
- claude_mpm/dashboard/static/monitors.html +431 -0
- claude_mpm/dashboard/static/production/events.html +659 -0
- claude_mpm/dashboard/static/production/main.html +715 -0
- claude_mpm/dashboard/static/production/monitors.html +483 -0
- claude_mpm/dashboard/static/socket.io.min.js +7 -0
- claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +7 -0
- claude_mpm/dashboard/static/test-archive/dashboard.html +635 -0
- claude_mpm/dashboard/static/test-archive/debug-events.html +147 -0
- claude_mpm/dashboard/static/test-archive/test-navigation.html +256 -0
- claude_mpm/dashboard/static/test-archive/test-react-exports.html +180 -0
- claude_mpm/dashboard/templates/index.html +82 -38
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +1 -1
- claude_mpm/services/agents/deployment/agent_discovery_service.py +3 -0
- claude_mpm/services/agents/deployment/agent_template_builder.py +25 -8
- claude_mpm/services/agents/deployment/agent_validator.py +3 -0
- claude_mpm/services/agents/deployment/validation/template_validator.py +13 -4
- claude_mpm/services/agents/local_template_manager.py +2 -6
- claude_mpm/services/monitor/daemon.py +1 -2
- claude_mpm/services/monitor/daemon_manager.py +2 -5
- claude_mpm/services/monitor/event_emitter.py +2 -2
- claude_mpm/services/monitor/handlers/code_analysis.py +4 -6
- claude_mpm/services/monitor/handlers/hooks.py +2 -4
- claude_mpm/services/monitor/server.py +23 -226
- claude_mpm/tools/code_tree_analyzer.py +2 -2
- {claude_mpm-4.2.43.dist-info → claude_mpm-4.2.51.dist-info}/METADATA +1 -1
- {claude_mpm-4.2.43.dist-info → claude_mpm-4.2.51.dist-info}/RECORD +148 -87
- claude_mpm/commands/mpm-browser-monitor.md +0 -370
- claude_mpm/commands/mpm-monitor.md +0 -177
- claude_mpm/dashboard/static/js/browser-console-monitor.js +0 -495
- claude_mpm/dashboard/static/js/components/browser-log-viewer.js +0 -763
- claude_mpm/dashboard/static/test-browser-monitor.html +0 -470
- claude_mpm/dashboard/static/test-simple.html +0 -97
- claude_mpm/services/monitor/handlers/browser.py +0 -451
- /claude_mpm/dashboard/static/{test_debug.html → test-archive/test_debug.html} +0 -0
- {claude_mpm-4.2.43.dist-info → claude_mpm-4.2.51.dist-info}/WHEEL +0 -0
- {claude_mpm-4.2.43.dist-info → claude_mpm-4.2.51.dist-info}/entry_points.txt +0 -0
- {claude_mpm-4.2.43.dist-info → claude_mpm-4.2.51.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-4.2.43.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');
|