claude-mpm 5.0.9__py3-none-any.whl → 5.4.41__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 (263) 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/{PM_INSTRUCTIONS_TEACH.md → CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md} +721 -41
  5. claude_mpm/agents/MEMORY.md +1 -1
  6. claude_mpm/agents/PM_INSTRUCTIONS.md +468 -468
  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 +70 -2
  11. claude_mpm/agents/templates/circuit-breakers.md +431 -45
  12. claude_mpm/cli/__init__.py +0 -1
  13. claude_mpm/cli/__main__.py +4 -0
  14. claude_mpm/cli/chrome_devtools_installer.py +175 -0
  15. claude_mpm/cli/commands/agent_state_manager.py +18 -27
  16. claude_mpm/cli/commands/agents.py +175 -37
  17. claude_mpm/cli/commands/auto_configure.py +723 -236
  18. claude_mpm/cli/commands/config.py +88 -2
  19. claude_mpm/cli/commands/configure.py +1262 -157
  20. claude_mpm/cli/commands/configure_agent_display.py +25 -6
  21. claude_mpm/cli/commands/mpm_init/core.py +225 -46
  22. claude_mpm/cli/commands/mpm_init/knowledge_extractor.py +481 -0
  23. claude_mpm/cli/commands/mpm_init/prompts.py +280 -0
  24. claude_mpm/cli/commands/postmortem.py +1 -1
  25. claude_mpm/cli/commands/profile.py +277 -0
  26. claude_mpm/cli/commands/skills.py +214 -189
  27. claude_mpm/cli/commands/summarize.py +413 -0
  28. claude_mpm/cli/executor.py +21 -3
  29. claude_mpm/cli/interactive/agent_wizard.py +85 -10
  30. claude_mpm/cli/parsers/agents_parser.py +54 -9
  31. claude_mpm/cli/parsers/auto_configure_parser.py +13 -138
  32. claude_mpm/cli/parsers/base_parser.py +12 -0
  33. claude_mpm/cli/parsers/config_parser.py +153 -83
  34. claude_mpm/cli/parsers/profile_parser.py +148 -0
  35. claude_mpm/cli/parsers/skills_parser.py +3 -2
  36. claude_mpm/cli/startup.py +879 -149
  37. claude_mpm/commands/mpm-config.md +28 -0
  38. claude_mpm/commands/mpm-doctor.md +9 -22
  39. claude_mpm/commands/mpm-help.md +5 -287
  40. claude_mpm/commands/mpm-init.md +81 -507
  41. claude_mpm/commands/mpm-monitor.md +15 -402
  42. claude_mpm/commands/mpm-organize.md +120 -0
  43. claude_mpm/commands/mpm-postmortem.md +6 -108
  44. claude_mpm/commands/mpm-session-resume.md +12 -363
  45. claude_mpm/commands/mpm-status.md +5 -69
  46. claude_mpm/commands/mpm-ticket-view.md +52 -495
  47. claude_mpm/commands/mpm-version.md +5 -107
  48. claude_mpm/config/agent_sources.py +27 -0
  49. claude_mpm/core/config.py +2 -4
  50. claude_mpm/core/framework/formatters/content_formatter.py +3 -13
  51. claude_mpm/core/framework/loaders/agent_loader.py +8 -5
  52. claude_mpm/core/framework/loaders/instruction_loader.py +52 -11
  53. claude_mpm/core/framework_loader.py +4 -2
  54. claude_mpm/core/logger.py +13 -0
  55. claude_mpm/core/optimized_startup.py +59 -0
  56. claude_mpm/core/output_style_manager.py +173 -43
  57. claude_mpm/core/shared/config_loader.py +1 -1
  58. claude_mpm/core/socketio_pool.py +3 -3
  59. claude_mpm/core/unified_agent_registry.py +134 -16
  60. claude_mpm/core/unified_config.py +22 -0
  61. claude_mpm/dashboard/static/svelte-build/_app/env.js +1 -0
  62. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/0.B_FtCwCQ.css +1 -0
  63. claude_mpm/dashboard/static/svelte-build/_app/immutable/assets/2.Cl_eSA4x.css +1 -0
  64. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/BgChzWQ1.js +1 -0
  65. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CIXEwuWe.js +1 -0
  66. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/CWc5urbQ.js +1 -0
  67. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DMkZpdF2.js +2 -0
  68. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/DjhvlsAc.js +1 -0
  69. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/N4qtv3Hx.js +2 -0
  70. claude_mpm/dashboard/static/svelte-build/_app/immutable/chunks/uj46x2Wr.js +1 -0
  71. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/app.DTL5mJO-.js +2 -0
  72. claude_mpm/dashboard/static/svelte-build/_app/immutable/entry/start.DzuEhzqh.js +1 -0
  73. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/0.CAGBuiOw.js +1 -0
  74. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/1.DFLC8jdE.js +1 -0
  75. claude_mpm/dashboard/static/svelte-build/_app/immutable/nodes/2.DPvEihJJ.js +10 -0
  76. claude_mpm/dashboard/static/svelte-build/_app/version.json +1 -0
  77. claude_mpm/dashboard/static/svelte-build/favicon.svg +7 -0
  78. claude_mpm/dashboard/static/svelte-build/index.html +36 -0
  79. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  80. claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
  81. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
  82. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
  83. claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
  84. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
  85. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
  86. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
  87. claude_mpm/hooks/claude_hooks/correlation_manager.py +60 -0
  88. claude_mpm/hooks/claude_hooks/event_handlers.py +211 -78
  89. claude_mpm/hooks/claude_hooks/hook_handler.py +155 -1
  90. claude_mpm/hooks/claude_hooks/installer.py +33 -10
  91. claude_mpm/hooks/claude_hooks/memory_integration.py +28 -0
  92. claude_mpm/hooks/claude_hooks/response_tracking.py +2 -3
  93. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
  94. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager.cpython-311.pyc +0 -0
  95. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
  96. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
  97. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
  98. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
  99. claude_mpm/hooks/claude_hooks/services/connection_manager.py +30 -6
  100. claude_mpm/hooks/memory_integration_hook.py +46 -1
  101. claude_mpm/init.py +63 -19
  102. claude_mpm/models/agent_definition.py +7 -0
  103. claude_mpm/models/git_repository.py +3 -3
  104. claude_mpm/scripts/claude-hook-handler.sh +58 -18
  105. claude_mpm/scripts/launch_monitor.py +93 -13
  106. claude_mpm/scripts/start_activity_logging.py +0 -0
  107. claude_mpm/services/agents/agent_builder.py +3 -3
  108. claude_mpm/services/agents/agent_recommendation_service.py +278 -0
  109. claude_mpm/services/agents/agent_review_service.py +280 -0
  110. claude_mpm/services/agents/cache_git_manager.py +6 -6
  111. claude_mpm/services/agents/deployment/agent_deployment.py +29 -7
  112. claude_mpm/services/agents/deployment/agent_discovery_service.py +4 -5
  113. claude_mpm/services/agents/deployment/agent_template_builder.py +5 -3
  114. claude_mpm/services/agents/deployment/agents_directory_resolver.py +2 -2
  115. claude_mpm/services/agents/deployment/multi_source_deployment_service.py +320 -29
  116. claude_mpm/services/agents/deployment/remote_agent_discovery_service.py +546 -68
  117. claude_mpm/services/agents/git_source_manager.py +36 -2
  118. claude_mpm/services/agents/loading/base_agent_manager.py +1 -13
  119. claude_mpm/services/agents/recommender.py +5 -3
  120. claude_mpm/services/agents/single_tier_deployment_service.py +2 -2
  121. claude_mpm/services/agents/sources/git_source_sync_service.py +13 -6
  122. claude_mpm/services/agents/startup_sync.py +22 -2
  123. claude_mpm/services/agents/toolchain_detector.py +10 -6
  124. claude_mpm/services/analysis/__init__.py +11 -1
  125. claude_mpm/services/analysis/clone_detector.py +1030 -0
  126. claude_mpm/services/command_deployment_service.py +81 -10
  127. claude_mpm/services/diagnostics/checks/agent_check.py +2 -2
  128. claude_mpm/services/diagnostics/checks/agent_sources_check.py +1 -1
  129. claude_mpm/services/event_bus/config.py +3 -1
  130. claude_mpm/services/git/git_operations_service.py +101 -16
  131. claude_mpm/services/monitor/daemon.py +9 -2
  132. claude_mpm/services/monitor/daemon_manager.py +39 -3
  133. claude_mpm/services/monitor/management/lifecycle.py +8 -1
  134. claude_mpm/services/monitor/server.py +698 -22
  135. claude_mpm/services/pm_skills_deployer.py +676 -0
  136. claude_mpm/services/profile_manager.py +331 -0
  137. claude_mpm/services/project/project_organizer.py +4 -0
  138. claude_mpm/services/self_upgrade_service.py +120 -12
  139. claude_mpm/services/skills/__init__.py +3 -0
  140. claude_mpm/services/skills/git_skill_source_manager.py +130 -2
  141. claude_mpm/services/skills/selective_skill_deployer.py +704 -0
  142. claude_mpm/services/skills/skill_to_agent_mapper.py +406 -0
  143. claude_mpm/services/skills_deployer.py +126 -9
  144. claude_mpm/services/socketio/dashboard_server.py +1 -0
  145. claude_mpm/services/socketio/event_normalizer.py +51 -6
  146. claude_mpm/services/socketio/server/core.py +386 -108
  147. claude_mpm/services/version_control/git_operations.py +103 -0
  148. claude_mpm/skills/skill_manager.py +92 -3
  149. claude_mpm/utils/agent_dependency_loader.py +14 -2
  150. claude_mpm/utils/agent_filters.py +17 -44
  151. claude_mpm/utils/gitignore.py +3 -0
  152. claude_mpm/utils/migration.py +4 -4
  153. claude_mpm/utils/robust_installer.py +47 -3
  154. {claude_mpm-5.0.9.dist-info → claude_mpm-5.4.41.dist-info}/METADATA +57 -87
  155. {claude_mpm-5.0.9.dist-info → claude_mpm-5.4.41.dist-info}/RECORD +160 -211
  156. claude_mpm-5.4.41.dist-info/entry_points.txt +5 -0
  157. claude_mpm-5.4.41.dist-info/licenses/LICENSE +94 -0
  158. claude_mpm-5.4.41.dist-info/licenses/LICENSE-FAQ.md +153 -0
  159. claude_mpm/agents/BASE_AGENT_TEMPLATE.md +0 -292
  160. claude_mpm/agents/BASE_DOCUMENTATION.md +0 -53
  161. claude_mpm/agents/BASE_OPS.md +0 -219
  162. claude_mpm/agents/BASE_PM.md +0 -480
  163. claude_mpm/agents/BASE_PROMPT_ENGINEER.md +0 -787
  164. claude_mpm/agents/BASE_QA.md +0 -167
  165. claude_mpm/agents/BASE_RESEARCH.md +0 -53
  166. claude_mpm/agents/base_agent_loader.py +0 -601
  167. claude_mpm/cli/commands/agents_detect.py +0 -380
  168. claude_mpm/cli/commands/agents_recommend.py +0 -309
  169. claude_mpm/cli/ticket_cli.py +0 -35
  170. claude_mpm/commands/mpm-agents-auto-configure.md +0 -278
  171. claude_mpm/commands/mpm-agents-detect.md +0 -177
  172. claude_mpm/commands/mpm-agents-list.md +0 -131
  173. claude_mpm/commands/mpm-agents-recommend.md +0 -223
  174. claude_mpm/commands/mpm-config-view.md +0 -150
  175. claude_mpm/commands/mpm-ticket-organize.md +0 -304
  176. claude_mpm/dashboard/analysis_runner.py +0 -455
  177. claude_mpm/dashboard/index.html +0 -13
  178. claude_mpm/dashboard/open_dashboard.py +0 -66
  179. claude_mpm/dashboard/static/css/activity.css +0 -1958
  180. claude_mpm/dashboard/static/css/connection-status.css +0 -370
  181. claude_mpm/dashboard/static/css/dashboard.css +0 -4701
  182. claude_mpm/dashboard/static/js/components/activity-tree.js +0 -1871
  183. claude_mpm/dashboard/static/js/components/agent-hierarchy.js +0 -777
  184. claude_mpm/dashboard/static/js/components/agent-inference.js +0 -956
  185. claude_mpm/dashboard/static/js/components/build-tracker.js +0 -333
  186. claude_mpm/dashboard/static/js/components/code-simple.js +0 -857
  187. claude_mpm/dashboard/static/js/components/connection-debug.js +0 -654
  188. claude_mpm/dashboard/static/js/components/diff-viewer.js +0 -891
  189. claude_mpm/dashboard/static/js/components/event-processor.js +0 -542
  190. claude_mpm/dashboard/static/js/components/event-viewer.js +0 -1155
  191. claude_mpm/dashboard/static/js/components/export-manager.js +0 -368
  192. claude_mpm/dashboard/static/js/components/file-change-tracker.js +0 -443
  193. claude_mpm/dashboard/static/js/components/file-change-viewer.js +0 -690
  194. claude_mpm/dashboard/static/js/components/file-tool-tracker.js +0 -724
  195. claude_mpm/dashboard/static/js/components/file-viewer.js +0 -580
  196. claude_mpm/dashboard/static/js/components/hud-library-loader.js +0 -211
  197. claude_mpm/dashboard/static/js/components/hud-manager.js +0 -671
  198. claude_mpm/dashboard/static/js/components/hud-visualizer.js +0 -1718
  199. claude_mpm/dashboard/static/js/components/module-viewer.js +0 -2764
  200. claude_mpm/dashboard/static/js/components/session-manager.js +0 -579
  201. claude_mpm/dashboard/static/js/components/socket-manager.js +0 -368
  202. claude_mpm/dashboard/static/js/components/ui-state-manager.js +0 -749
  203. claude_mpm/dashboard/static/js/components/unified-data-viewer.js +0 -1824
  204. claude_mpm/dashboard/static/js/components/working-directory.js +0 -920
  205. claude_mpm/dashboard/static/js/connection-manager.js +0 -536
  206. claude_mpm/dashboard/static/js/dashboard.js +0 -1914
  207. claude_mpm/dashboard/static/js/extension-error-handler.js +0 -164
  208. claude_mpm/dashboard/static/js/socket-client.js +0 -1474
  209. claude_mpm/dashboard/static/js/tab-isolation-fix.js +0 -185
  210. claude_mpm/dashboard/static/socket.io.min.js +0 -7
  211. claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +0 -7
  212. claude_mpm/dashboard/templates/code_simple.html +0 -153
  213. claude_mpm/dashboard/templates/index.html +0 -606
  214. claude_mpm/dashboard/test_dashboard.html +0 -372
  215. claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-313.pyc +0 -0
  216. claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-313.pyc +0 -0
  217. claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-313.pyc +0 -0
  218. claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-313.pyc +0 -0
  219. claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-313.pyc +0 -0
  220. claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-313.pyc +0 -0
  221. claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-313.pyc +0 -0
  222. claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-313.pyc +0 -0
  223. claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-313.pyc +0 -0
  224. claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-313.pyc +0 -0
  225. claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-313.pyc +0 -0
  226. claude_mpm/scripts/mcp_server.py +0 -75
  227. claude_mpm/scripts/mcp_wrapper.py +0 -39
  228. claude_mpm/services/mcp_gateway/__init__.py +0 -159
  229. claude_mpm/services/mcp_gateway/auto_configure.py +0 -369
  230. claude_mpm/services/mcp_gateway/config/__init__.py +0 -17
  231. claude_mpm/services/mcp_gateway/config/config_loader.py +0 -296
  232. claude_mpm/services/mcp_gateway/config/config_schema.py +0 -243
  233. claude_mpm/services/mcp_gateway/config/configuration.py +0 -429
  234. claude_mpm/services/mcp_gateway/core/__init__.py +0 -43
  235. claude_mpm/services/mcp_gateway/core/base.py +0 -312
  236. claude_mpm/services/mcp_gateway/core/exceptions.py +0 -253
  237. claude_mpm/services/mcp_gateway/core/interfaces.py +0 -443
  238. claude_mpm/services/mcp_gateway/core/process_pool.py +0 -977
  239. claude_mpm/services/mcp_gateway/core/singleton_manager.py +0 -315
  240. claude_mpm/services/mcp_gateway/core/startup_verification.py +0 -316
  241. claude_mpm/services/mcp_gateway/main.py +0 -589
  242. claude_mpm/services/mcp_gateway/registry/__init__.py +0 -12
  243. claude_mpm/services/mcp_gateway/registry/service_registry.py +0 -412
  244. claude_mpm/services/mcp_gateway/registry/tool_registry.py +0 -489
  245. claude_mpm/services/mcp_gateway/server/__init__.py +0 -15
  246. claude_mpm/services/mcp_gateway/server/mcp_gateway.py +0 -414
  247. claude_mpm/services/mcp_gateway/server/stdio_handler.py +0 -372
  248. claude_mpm/services/mcp_gateway/server/stdio_server.py +0 -712
  249. claude_mpm/services/mcp_gateway/tools/__init__.py +0 -36
  250. claude_mpm/services/mcp_gateway/tools/base_adapter.py +0 -485
  251. claude_mpm/services/mcp_gateway/tools/document_summarizer.py +0 -789
  252. claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +0 -654
  253. claude_mpm/services/mcp_gateway/tools/health_check_tool.py +0 -456
  254. claude_mpm/services/mcp_gateway/tools/hello_world.py +0 -551
  255. claude_mpm/services/mcp_gateway/tools/kuzu_memory_service.py +0 -555
  256. claude_mpm/services/mcp_gateway/utils/__init__.py +0 -14
  257. claude_mpm/services/mcp_gateway/utils/package_version_checker.py +0 -160
  258. claude_mpm/services/mcp_gateway/utils/update_preferences.py +0 -170
  259. claude_mpm-5.0.9.dist-info/entry_points.txt +0 -10
  260. claude_mpm-5.0.9.dist-info/licenses/LICENSE +0 -21
  261. /claude_mpm/agents/{OUTPUT_STYLE.md → CLAUDE_MPM_OUTPUT_STYLE.md} +0 -0
  262. {claude_mpm-5.0.9.dist-info → claude_mpm-5.4.41.dist-info}/WHEEL +0 -0
  263. {claude_mpm-5.0.9.dist-info → claude_mpm-5.4.41.dist-info}/top_level.txt +0 -0
@@ -1,891 +0,0 @@
1
- /**
2
- * Diff Viewer Component
3
- *
4
- * Shows side-by-side or unified diffs for file changes.
5
- * Supports syntax highlighting, navigation between changes,
6
- * and displays timestamps and operation types.
7
- *
8
- * Features:
9
- * - Side-by-side and unified diff views
10
- * - Syntax highlighting for code
11
- * - Navigation between multiple changes
12
- * - Operation history timeline
13
- */
14
- class DiffViewer {
15
- constructor() {
16
- this.modal = null;
17
- this.currentFile = null;
18
- this.currentMode = 'side-by-side'; // 'side-by-side' or 'unified'
19
- this.initialized = false;
20
-
21
- // Diff computation cache
22
- this.diffCache = new Map();
23
- }
24
-
25
- /**
26
- * Initialize the diff viewer
27
- */
28
- initialize() {
29
- if (this.initialized) return;
30
-
31
- this.createModal();
32
- this.setupEventHandlers();
33
- this.initialized = true;
34
- console.log('DiffViewer initialized');
35
- }
36
-
37
- /**
38
- * Create the modal HTML structure
39
- */
40
- createModal() {
41
- const modalHTML = `
42
- <div class="diff-viewer-modal" id="diff-viewer-modal">
43
- <div class="diff-viewer-content">
44
- <div class="diff-viewer-header">
45
- <div class="diff-viewer-title">
46
- <span class="diff-file-icon">📄</span>
47
- <span class="diff-file-path" id="diff-file-path">Loading...</span>
48
- </div>
49
- <div class="diff-viewer-controls">
50
- <div class="diff-mode-toggle">
51
- <button class="diff-mode-btn active" data-mode="side-by-side">
52
- Side by Side
53
- </button>
54
- <button class="diff-mode-btn" data-mode="unified">
55
- Unified
56
- </button>
57
- </div>
58
- <div class="diff-stats" id="diff-stats">
59
- <span class="additions">+0</span>
60
- <span class="deletions">-0</span>
61
- </div>
62
- <button class="diff-viewer-close" id="diff-viewer-close">×</button>
63
- </div>
64
- </div>
65
-
66
- <div class="diff-viewer-subheader">
67
- <div class="diff-operations-summary" id="diff-operations-summary">
68
- <span class="op-count">0 operations</span>
69
- <span class="op-timeline">Timeline</span>
70
- </div>
71
- <div class="diff-navigation">
72
- <button class="diff-nav-btn" id="diff-prev-change" disabled>
73
- ← Previous
74
- </button>
75
- <span class="diff-nav-info" id="diff-nav-info">Change 1 of 1</span>
76
- <button class="diff-nav-btn" id="diff-next-change" disabled>
77
- Next →
78
- </button>
79
- </div>
80
- </div>
81
-
82
- <div class="diff-viewer-body" id="diff-viewer-body">
83
- <!-- Diff content will be inserted here -->
84
- </div>
85
-
86
- <div class="diff-viewer-footer">
87
- <div class="diff-timeline" id="diff-timeline">
88
- <!-- Operation timeline will be inserted here -->
89
- </div>
90
- </div>
91
- </div>
92
- </div>
93
- `;
94
-
95
- document.body.insertAdjacentHTML('beforeend', modalHTML);
96
- this.modal = document.getElementById('diff-viewer-modal');
97
-
98
- // Add CSS for the diff viewer
99
- this.injectStyles();
100
- }
101
-
102
- /**
103
- * Inject CSS styles for the diff viewer
104
- */
105
- injectStyles() {
106
- const styleId = 'diff-viewer-styles';
107
- if (document.getElementById(styleId)) return;
108
-
109
- const styles = `
110
- <style id="${styleId}">
111
- .diff-viewer-modal {
112
- display: none;
113
- position: fixed;
114
- top: 0;
115
- left: 0;
116
- right: 0;
117
- bottom: 0;
118
- background: rgba(0, 0, 0, 0.5);
119
- z-index: 10000;
120
- padding: 20px;
121
- overflow: auto;
122
- }
123
-
124
- .diff-viewer-modal.show {
125
- display: flex;
126
- align-items: center;
127
- justify-content: center;
128
- }
129
-
130
- .diff-viewer-content {
131
- background: white;
132
- border-radius: 8px;
133
- width: 90%;
134
- max-width: 1400px;
135
- height: 90%;
136
- display: flex;
137
- flex-direction: column;
138
- box-shadow: 0 4px 20px rgba(0, 0, 0, 0.15);
139
- }
140
-
141
- .diff-viewer-header {
142
- display: flex;
143
- justify-content: space-between;
144
- align-items: center;
145
- padding: 16px 20px;
146
- border-bottom: 1px solid #e2e8f0;
147
- background: #f8fafc;
148
- border-radius: 8px 8px 0 0;
149
- }
150
-
151
- .diff-viewer-title {
152
- display: flex;
153
- align-items: center;
154
- gap: 8px;
155
- font-size: 14px;
156
- font-weight: 600;
157
- color: #2d3748;
158
- }
159
-
160
- .diff-file-icon {
161
- font-size: 18px;
162
- }
163
-
164
- .diff-file-path {
165
- font-family: 'SF Mono', Monaco, monospace;
166
- font-size: 13px;
167
- }
168
-
169
- .diff-viewer-controls {
170
- display: flex;
171
- align-items: center;
172
- gap: 16px;
173
- }
174
-
175
- .diff-mode-toggle {
176
- display: flex;
177
- gap: 4px;
178
- background: #e2e8f0;
179
- padding: 2px;
180
- border-radius: 4px;
181
- }
182
-
183
- .diff-mode-btn {
184
- padding: 4px 12px;
185
- border: none;
186
- background: transparent;
187
- cursor: pointer;
188
- font-size: 12px;
189
- border-radius: 3px;
190
- transition: all 0.2s;
191
- }
192
-
193
- .diff-mode-btn.active {
194
- background: white;
195
- box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
196
- }
197
-
198
- .diff-stats {
199
- display: flex;
200
- gap: 8px;
201
- font-size: 12px;
202
- font-family: monospace;
203
- }
204
-
205
- .diff-stats .additions {
206
- color: #22c55e;
207
- }
208
-
209
- .diff-stats .deletions {
210
- color: #ef4444;
211
- }
212
-
213
- .diff-viewer-close {
214
- width: 32px;
215
- height: 32px;
216
- border: none;
217
- background: transparent;
218
- font-size: 24px;
219
- cursor: pointer;
220
- color: #718096;
221
- display: flex;
222
- align-items: center;
223
- justify-content: center;
224
- border-radius: 4px;
225
- transition: all 0.2s;
226
- }
227
-
228
- .diff-viewer-close:hover {
229
- background: #e2e8f0;
230
- color: #2d3748;
231
- }
232
-
233
- .diff-viewer-subheader {
234
- display: flex;
235
- justify-content: space-between;
236
- align-items: center;
237
- padding: 12px 20px;
238
- border-bottom: 1px solid #e2e8f0;
239
- background: #fafbfc;
240
- }
241
-
242
- .diff-operations-summary {
243
- display: flex;
244
- gap: 16px;
245
- font-size: 13px;
246
- color: #4a5568;
247
- }
248
-
249
- .diff-navigation {
250
- display: flex;
251
- align-items: center;
252
- gap: 8px;
253
- }
254
-
255
- .diff-nav-btn {
256
- padding: 4px 12px;
257
- border: 1px solid #cbd5e0;
258
- background: white;
259
- cursor: pointer;
260
- font-size: 12px;
261
- border-radius: 4px;
262
- transition: all 0.2s;
263
- }
264
-
265
- .diff-nav-btn:hover:not(:disabled) {
266
- background: #f8fafc;
267
- border-color: #4299e1;
268
- }
269
-
270
- .diff-nav-btn:disabled {
271
- opacity: 0.5;
272
- cursor: not-allowed;
273
- }
274
-
275
- .diff-nav-info {
276
- font-size: 12px;
277
- color: #718096;
278
- padding: 0 8px;
279
- }
280
-
281
- .diff-viewer-body {
282
- flex: 1;
283
- overflow: auto;
284
- padding: 20px;
285
- background: #fafbfc;
286
- }
287
-
288
- /* Side-by-side diff styles */
289
- .diff-side-by-side {
290
- display: flex;
291
- gap: 16px;
292
- height: 100%;
293
- }
294
-
295
- .diff-panel {
296
- flex: 1;
297
- background: white;
298
- border: 1px solid #e2e8f0;
299
- border-radius: 4px;
300
- overflow: hidden;
301
- }
302
-
303
- .diff-panel-header {
304
- padding: 8px 12px;
305
- background: #f8fafc;
306
- border-bottom: 1px solid #e2e8f0;
307
- font-size: 12px;
308
- font-weight: 600;
309
- color: #4a5568;
310
- }
311
-
312
- .diff-panel-content {
313
- overflow: auto;
314
- height: calc(100% - 36px);
315
- }
316
-
317
- .diff-line {
318
- display: flex;
319
- font-family: 'SF Mono', Monaco, monospace;
320
- font-size: 12px;
321
- line-height: 1.5;
322
- white-space: pre;
323
- }
324
-
325
- .diff-line-number {
326
- width: 50px;
327
- padding: 0 8px;
328
- text-align: right;
329
- color: #a0aec0;
330
- background: #f8fafc;
331
- border-right: 1px solid #e2e8f0;
332
- user-select: none;
333
- }
334
-
335
- .diff-line-content {
336
- flex: 1;
337
- padding: 0 12px;
338
- overflow-x: auto;
339
- }
340
-
341
- .diff-line-added {
342
- background: #d4f4dd;
343
- }
344
-
345
- .diff-line-added .diff-line-content {
346
- background: #e7fced;
347
- }
348
-
349
- .diff-line-removed {
350
- background: #ffd4d4;
351
- }
352
-
353
- .diff-line-removed .diff-line-content {
354
- background: #ffeaea;
355
- }
356
-
357
- .diff-line-context {
358
- color: #4a5568;
359
- }
360
-
361
- /* Unified diff styles */
362
- .diff-unified {
363
- background: white;
364
- border: 1px solid #e2e8f0;
365
- border-radius: 4px;
366
- overflow: auto;
367
- }
368
-
369
- .diff-hunk-header {
370
- padding: 8px 12px;
371
- background: #f1f5f9;
372
- color: #475569;
373
- font-family: monospace;
374
- font-size: 12px;
375
- border-bottom: 1px solid #e2e8f0;
376
- }
377
-
378
- /* Timeline styles */
379
- .diff-viewer-footer {
380
- padding: 12px 20px;
381
- border-top: 1px solid #e2e8f0;
382
- background: #f8fafc;
383
- border-radius: 0 0 8px 8px;
384
- }
385
-
386
- .diff-timeline {
387
- display: flex;
388
- gap: 8px;
389
- overflow-x: auto;
390
- padding: 8px 0;
391
- }
392
-
393
- .timeline-item {
394
- display: flex;
395
- flex-direction: column;
396
- align-items: center;
397
- gap: 4px;
398
- padding: 4px 8px;
399
- background: white;
400
- border: 1px solid #cbd5e0;
401
- border-radius: 4px;
402
- cursor: pointer;
403
- transition: all 0.2s;
404
- min-width: 80px;
405
- font-size: 11px;
406
- }
407
-
408
- .timeline-item:hover {
409
- background: #edf2f7;
410
- border-color: #4299e1;
411
- }
412
-
413
- .timeline-item.active {
414
- background: #4299e1;
415
- color: white;
416
- border-color: #3182ce;
417
- }
418
-
419
- .timeline-operation {
420
- font-weight: 600;
421
- }
422
-
423
- .timeline-time {
424
- color: #718096;
425
- font-size: 10px;
426
- }
427
-
428
- .timeline-item.active .timeline-time {
429
- color: rgba(255, 255, 255, 0.9);
430
- }
431
- </style>
432
- `;
433
-
434
- document.head.insertAdjacentHTML('beforeend', styles);
435
- }
436
-
437
- /**
438
- * Setup event handlers
439
- */
440
- setupEventHandlers() {
441
- // Close button
442
- document.getElementById('diff-viewer-close').addEventListener('click', () => {
443
- this.hide();
444
- });
445
-
446
- // Modal background click
447
- this.modal.addEventListener('click', (e) => {
448
- if (e.target === this.modal) {
449
- this.hide();
450
- }
451
- });
452
-
453
- // Escape key
454
- document.addEventListener('keydown', (e) => {
455
- if (e.key === 'Escape' && this.modal.classList.contains('show')) {
456
- this.hide();
457
- }
458
- });
459
-
460
- // Mode toggle
461
- document.querySelectorAll('.diff-mode-btn').forEach(btn => {
462
- btn.addEventListener('click', (e) => {
463
- this.setMode(e.target.dataset.mode);
464
- });
465
- });
466
-
467
- // Navigation
468
- document.getElementById('diff-prev-change').addEventListener('click', () => {
469
- this.navigateToPreviousChange();
470
- });
471
-
472
- document.getElementById('diff-next-change').addEventListener('click', () => {
473
- this.navigateToNextChange();
474
- });
475
- }
476
-
477
- /**
478
- * Show the diff viewer for a file
479
- * @param {Object} diffData - Diff data from FileChangeTracker
480
- */
481
- show(diffData) {
482
- if (!this.initialized) {
483
- this.initialize();
484
- }
485
-
486
- this.currentFile = diffData;
487
- this.modal.classList.add('show');
488
-
489
- // Update header
490
- document.getElementById('diff-file-path').textContent = diffData.filePath;
491
-
492
- // Generate and display diff
493
- this.generateDiff(diffData);
494
-
495
- // Update operations timeline
496
- this.updateTimeline(diffData.operations);
497
-
498
- // Update statistics
499
- this.updateStatistics(diffData);
500
- }
501
-
502
- /**
503
- * Hide the diff viewer
504
- */
505
- hide() {
506
- this.modal.classList.remove('show');
507
- this.currentFile = null;
508
- }
509
-
510
- /**
511
- * Set diff display mode
512
- * @param {string} mode - 'side-by-side' or 'unified'
513
- */
514
- setMode(mode) {
515
- this.currentMode = mode;
516
-
517
- // Update button states
518
- document.querySelectorAll('.diff-mode-btn').forEach(btn => {
519
- btn.classList.toggle('active', btn.dataset.mode === mode);
520
- });
521
-
522
- // Regenerate diff display
523
- if (this.currentFile) {
524
- this.generateDiff(this.currentFile);
525
- }
526
- }
527
-
528
- /**
529
- * Generate diff display
530
- * @param {Object} diffData - Diff data
531
- */
532
- generateDiff(diffData) {
533
- const { initialContent, currentContent } = diffData;
534
-
535
- // Compute diff
536
- const diff = this.computeDiff(initialContent || '', currentContent || '');
537
-
538
- // Display based on mode
539
- const body = document.getElementById('diff-viewer-body');
540
- if (this.currentMode === 'side-by-side') {
541
- body.innerHTML = this.renderSideBySideDiff(diff, initialContent, currentContent);
542
- } else {
543
- body.innerHTML = this.renderUnifiedDiff(diff);
544
- }
545
- }
546
-
547
- /**
548
- * Compute diff between two strings
549
- * @param {string} oldText - Original text
550
- * @param {string} newText - New text
551
- * @returns {Array} Array of diff chunks
552
- */
553
- computeDiff(oldText, newText) {
554
- const oldLines = oldText.split('\n');
555
- const newLines = newText.split('\n');
556
-
557
- // Simple line-by-line diff algorithm
558
- const diff = [];
559
- const maxLines = Math.max(oldLines.length, newLines.length);
560
-
561
- let oldIndex = 0;
562
- let newIndex = 0;
563
-
564
- while (oldIndex < oldLines.length || newIndex < newLines.length) {
565
- const oldLine = oldIndex < oldLines.length ? oldLines[oldIndex] : null;
566
- const newLine = newIndex < newLines.length ? newLines[newIndex] : null;
567
-
568
- if (oldLine === newLine) {
569
- // Unchanged line
570
- diff.push({
571
- type: 'unchanged',
572
- oldLine: oldLine,
573
- newLine: newLine,
574
- oldLineNumber: oldIndex + 1,
575
- newLineNumber: newIndex + 1
576
- });
577
- oldIndex++;
578
- newIndex++;
579
- } else if (oldLine === null) {
580
- // Added line
581
- diff.push({
582
- type: 'added',
583
- newLine: newLine,
584
- newLineNumber: newIndex + 1
585
- });
586
- newIndex++;
587
- } else if (newLine === null) {
588
- // Removed line
589
- diff.push({
590
- type: 'removed',
591
- oldLine: oldLine,
592
- oldLineNumber: oldIndex + 1
593
- });
594
- oldIndex++;
595
- } else {
596
- // Changed line - try to find matching lines ahead
597
- let found = false;
598
-
599
- // Look ahead in new lines for old line
600
- for (let i = newIndex + 1; i < Math.min(newIndex + 5, newLines.length); i++) {
601
- if (newLines[i] === oldLine) {
602
- // Found old line ahead - mark intervening as added
603
- for (let j = newIndex; j < i; j++) {
604
- diff.push({
605
- type: 'added',
606
- newLine: newLines[j],
607
- newLineNumber: j + 1
608
- });
609
- }
610
- newIndex = i;
611
- found = true;
612
- break;
613
- }
614
- }
615
-
616
- if (!found) {
617
- // Look ahead in old lines for new line
618
- for (let i = oldIndex + 1; i < Math.min(oldIndex + 5, oldLines.length); i++) {
619
- if (oldLines[i] === newLine) {
620
- // Found new line ahead - mark intervening as removed
621
- for (let j = oldIndex; j < i; j++) {
622
- diff.push({
623
- type: 'removed',
624
- oldLine: oldLines[j],
625
- oldLineNumber: j + 1
626
- });
627
- }
628
- oldIndex = i;
629
- found = true;
630
- break;
631
- }
632
- }
633
- }
634
-
635
- if (!found) {
636
- // True change - show as remove + add
637
- diff.push({
638
- type: 'removed',
639
- oldLine: oldLine,
640
- oldLineNumber: oldIndex + 1
641
- });
642
- diff.push({
643
- type: 'added',
644
- newLine: newLine,
645
- newLineNumber: newIndex + 1
646
- });
647
- oldIndex++;
648
- newIndex++;
649
- }
650
- }
651
- }
652
-
653
- return diff;
654
- }
655
-
656
- /**
657
- * Render side-by-side diff
658
- * @param {Array} diff - Diff chunks
659
- * @param {string} oldContent - Original content
660
- * @param {string} newContent - New content
661
- * @returns {string} HTML
662
- */
663
- renderSideBySideDiff(diff, oldContent, newContent) {
664
- const oldLines = [];
665
- const newLines = [];
666
-
667
- for (const chunk of diff) {
668
- if (chunk.type === 'unchanged') {
669
- oldLines.push({
670
- number: chunk.oldLineNumber,
671
- content: chunk.oldLine,
672
- type: 'context'
673
- });
674
- newLines.push({
675
- number: chunk.newLineNumber,
676
- content: chunk.newLine,
677
- type: 'context'
678
- });
679
- } else if (chunk.type === 'removed') {
680
- oldLines.push({
681
- number: chunk.oldLineNumber,
682
- content: chunk.oldLine,
683
- type: 'removed'
684
- });
685
- newLines.push({
686
- number: '',
687
- content: '',
688
- type: 'empty'
689
- });
690
- } else if (chunk.type === 'added') {
691
- oldLines.push({
692
- number: '',
693
- content: '',
694
- type: 'empty'
695
- });
696
- newLines.push({
697
- number: chunk.newLineNumber,
698
- content: chunk.newLine,
699
- type: 'added'
700
- });
701
- }
702
- }
703
-
704
- return `
705
- <div class="diff-side-by-side">
706
- <div class="diff-panel">
707
- <div class="diff-panel-header">Original</div>
708
- <div class="diff-panel-content">
709
- ${this.renderDiffLines(oldLines)}
710
- </div>
711
- </div>
712
- <div class="diff-panel">
713
- <div class="diff-panel-header">Modified</div>
714
- <div class="diff-panel-content">
715
- ${this.renderDiffLines(newLines)}
716
- </div>
717
- </div>
718
- </div>
719
- `;
720
- }
721
-
722
- /**
723
- * Render diff lines
724
- * @param {Array} lines - Lines to render
725
- * @returns {string} HTML
726
- */
727
- renderDiffLines(lines) {
728
- return lines.map(line => {
729
- const lineClass = line.type === 'removed' ? 'diff-line-removed' :
730
- line.type === 'added' ? 'diff-line-added' :
731
- line.type === 'empty' ? 'diff-line-empty' :
732
- 'diff-line-context';
733
-
734
- return `
735
- <div class="diff-line ${lineClass}">
736
- <span class="diff-line-number">${line.number}</span>
737
- <span class="diff-line-content">${this.escapeHtml(line.content)}</span>
738
- </div>
739
- `;
740
- }).join('');
741
- }
742
-
743
- /**
744
- * Render unified diff
745
- * @param {Array} diff - Diff chunks
746
- * @returns {string} HTML
747
- */
748
- renderUnifiedDiff(diff) {
749
- const lines = [];
750
-
751
- for (const chunk of diff) {
752
- if (chunk.type === 'unchanged') {
753
- lines.push(`
754
- <div class="diff-line diff-line-context">
755
- <span class="diff-line-number">${chunk.oldLineNumber}</span>
756
- <span class="diff-line-number">${chunk.newLineNumber}</span>
757
- <span class="diff-line-content"> ${this.escapeHtml(chunk.oldLine)}</span>
758
- </div>
759
- `);
760
- } else if (chunk.type === 'removed') {
761
- lines.push(`
762
- <div class="diff-line diff-line-removed">
763
- <span class="diff-line-number">${chunk.oldLineNumber}</span>
764
- <span class="diff-line-number">-</span>
765
- <span class="diff-line-content">-${this.escapeHtml(chunk.oldLine)}</span>
766
- </div>
767
- `);
768
- } else if (chunk.type === 'added') {
769
- lines.push(`
770
- <div class="diff-line diff-line-added">
771
- <span class="diff-line-number">-</span>
772
- <span class="diff-line-number">${chunk.newLineNumber}</span>
773
- <span class="diff-line-content">+${this.escapeHtml(chunk.newLine)}</span>
774
- </div>
775
- `);
776
- }
777
- }
778
-
779
- return `
780
- <div class="diff-unified">
781
- ${lines.join('')}
782
- </div>
783
- `;
784
- }
785
-
786
- /**
787
- * Update operations timeline
788
- * @param {Array} operations - File operations
789
- */
790
- updateTimeline(operations) {
791
- const timeline = document.getElementById('diff-timeline');
792
-
793
- if (!operations || operations.length === 0) {
794
- timeline.innerHTML = '<div class="timeline-empty">No operations recorded</div>';
795
- return;
796
- }
797
-
798
- // Sort operations by timestamp
799
- const sortedOps = [...operations].sort((a, b) =>
800
- new Date(a.timestamp) - new Date(b.timestamp)
801
- );
802
-
803
- timeline.innerHTML = sortedOps.map((op, index) => {
804
- const time = new Date(op.timestamp);
805
- const timeStr = time.toLocaleTimeString();
806
-
807
- return `
808
- <div class="timeline-item ${index === 0 ? 'active' : ''}"
809
- data-index="${index}">
810
- <div class="timeline-operation">${op.operation}</div>
811
- <div class="timeline-time">${timeStr}</div>
812
- </div>
813
- `;
814
- }).join('');
815
-
816
- // Add click handlers
817
- timeline.querySelectorAll('.timeline-item').forEach(item => {
818
- item.addEventListener('click', (e) => {
819
- const index = parseInt(e.currentTarget.dataset.index);
820
- this.selectOperation(index);
821
- });
822
- });
823
- }
824
-
825
- /**
826
- * Select an operation in the timeline
827
- * @param {number} index - Operation index
828
- */
829
- selectOperation(index) {
830
- // Update active state
831
- document.querySelectorAll('.timeline-item').forEach((item, i) => {
832
- item.classList.toggle('active', i === index);
833
- });
834
-
835
- // Could implement showing diff up to this operation
836
- console.log('Selected operation:', index);
837
- }
838
-
839
- /**
840
- * Update statistics
841
- * @param {Object} diffData - Diff data
842
- */
843
- updateStatistics(diffData) {
844
- const diff = this.computeDiff(
845
- diffData.initialContent || '',
846
- diffData.currentContent || ''
847
- );
848
-
849
- const additions = diff.filter(d => d.type === 'added').length;
850
- const deletions = diff.filter(d => d.type === 'removed').length;
851
-
852
- document.querySelector('.diff-stats .additions').textContent = `+${additions}`;
853
- document.querySelector('.diff-stats .deletions').textContent = `-${deletions}`;
854
-
855
- const opCount = diffData.operations ? diffData.operations.length : 0;
856
- document.querySelector('.op-count').textContent =
857
- `${opCount} operation${opCount !== 1 ? 's' : ''}`;
858
- }
859
-
860
- /**
861
- * Navigate to previous change
862
- */
863
- navigateToPreviousChange() {
864
- console.log('Navigate to previous change');
865
- // Implementation would scroll to previous diff chunk
866
- }
867
-
868
- /**
869
- * Navigate to next change
870
- */
871
- navigateToNextChange() {
872
- console.log('Navigate to next change');
873
- // Implementation would scroll to next diff chunk
874
- }
875
-
876
- /**
877
- * Escape HTML for safe display
878
- * @param {string} text - Text to escape
879
- * @returns {string} Escaped text
880
- */
881
- escapeHtml(text) {
882
- const div = document.createElement('div');
883
- div.textContent = text || '';
884
- return div.innerHTML;
885
- }
886
- }
887
-
888
- // Export for use in other modules
889
- if (typeof window !== 'undefined') {
890
- window.DiffViewer = DiffViewer;
891
- }