codepp 0.0.437__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 (288) hide show
  1. code_puppy/__init__.py +10 -0
  2. code_puppy/__main__.py +10 -0
  3. code_puppy/agents/__init__.py +31 -0
  4. code_puppy/agents/agent_c_reviewer.py +155 -0
  5. code_puppy/agents/agent_code_puppy.py +117 -0
  6. code_puppy/agents/agent_code_reviewer.py +90 -0
  7. code_puppy/agents/agent_cpp_reviewer.py +132 -0
  8. code_puppy/agents/agent_creator_agent.py +638 -0
  9. code_puppy/agents/agent_golang_reviewer.py +151 -0
  10. code_puppy/agents/agent_helios.py +124 -0
  11. code_puppy/agents/agent_javascript_reviewer.py +160 -0
  12. code_puppy/agents/agent_manager.py +742 -0
  13. code_puppy/agents/agent_pack_leader.py +385 -0
  14. code_puppy/agents/agent_planning.py +165 -0
  15. code_puppy/agents/agent_python_programmer.py +169 -0
  16. code_puppy/agents/agent_python_reviewer.py +90 -0
  17. code_puppy/agents/agent_qa_expert.py +163 -0
  18. code_puppy/agents/agent_qa_kitten.py +208 -0
  19. code_puppy/agents/agent_scheduler.py +121 -0
  20. code_puppy/agents/agent_security_auditor.py +181 -0
  21. code_puppy/agents/agent_terminal_qa.py +323 -0
  22. code_puppy/agents/agent_typescript_reviewer.py +166 -0
  23. code_puppy/agents/base_agent.py +2156 -0
  24. code_puppy/agents/event_stream_handler.py +348 -0
  25. code_puppy/agents/json_agent.py +202 -0
  26. code_puppy/agents/pack/__init__.py +34 -0
  27. code_puppy/agents/pack/bloodhound.py +304 -0
  28. code_puppy/agents/pack/husky.py +327 -0
  29. code_puppy/agents/pack/retriever.py +393 -0
  30. code_puppy/agents/pack/shepherd.py +348 -0
  31. code_puppy/agents/pack/terrier.py +287 -0
  32. code_puppy/agents/pack/watchdog.py +367 -0
  33. code_puppy/agents/prompt_reviewer.py +145 -0
  34. code_puppy/agents/subagent_stream_handler.py +276 -0
  35. code_puppy/api/__init__.py +13 -0
  36. code_puppy/api/app.py +169 -0
  37. code_puppy/api/main.py +21 -0
  38. code_puppy/api/pty_manager.py +453 -0
  39. code_puppy/api/routers/__init__.py +12 -0
  40. code_puppy/api/routers/agents.py +36 -0
  41. code_puppy/api/routers/commands.py +217 -0
  42. code_puppy/api/routers/config.py +75 -0
  43. code_puppy/api/routers/sessions.py +234 -0
  44. code_puppy/api/templates/terminal.html +361 -0
  45. code_puppy/api/websocket.py +154 -0
  46. code_puppy/callbacks.py +692 -0
  47. code_puppy/chatgpt_codex_client.py +338 -0
  48. code_puppy/claude_cache_client.py +672 -0
  49. code_puppy/cli_runner.py +1073 -0
  50. code_puppy/command_line/__init__.py +1 -0
  51. code_puppy/command_line/add_model_menu.py +1092 -0
  52. code_puppy/command_line/agent_menu.py +662 -0
  53. code_puppy/command_line/attachments.py +395 -0
  54. code_puppy/command_line/autosave_menu.py +704 -0
  55. code_puppy/command_line/clipboard.py +527 -0
  56. code_puppy/command_line/colors_menu.py +532 -0
  57. code_puppy/command_line/command_handler.py +293 -0
  58. code_puppy/command_line/command_registry.py +150 -0
  59. code_puppy/command_line/config_commands.py +719 -0
  60. code_puppy/command_line/core_commands.py +867 -0
  61. code_puppy/command_line/diff_menu.py +865 -0
  62. code_puppy/command_line/file_path_completion.py +73 -0
  63. code_puppy/command_line/load_context_completion.py +52 -0
  64. code_puppy/command_line/mcp/__init__.py +10 -0
  65. code_puppy/command_line/mcp/base.py +32 -0
  66. code_puppy/command_line/mcp/catalog_server_installer.py +175 -0
  67. code_puppy/command_line/mcp/custom_server_form.py +688 -0
  68. code_puppy/command_line/mcp/custom_server_installer.py +195 -0
  69. code_puppy/command_line/mcp/edit_command.py +148 -0
  70. code_puppy/command_line/mcp/handler.py +138 -0
  71. code_puppy/command_line/mcp/help_command.py +147 -0
  72. code_puppy/command_line/mcp/install_command.py +214 -0
  73. code_puppy/command_line/mcp/install_menu.py +705 -0
  74. code_puppy/command_line/mcp/list_command.py +94 -0
  75. code_puppy/command_line/mcp/logs_command.py +235 -0
  76. code_puppy/command_line/mcp/remove_command.py +82 -0
  77. code_puppy/command_line/mcp/restart_command.py +100 -0
  78. code_puppy/command_line/mcp/search_command.py +123 -0
  79. code_puppy/command_line/mcp/start_all_command.py +135 -0
  80. code_puppy/command_line/mcp/start_command.py +117 -0
  81. code_puppy/command_line/mcp/status_command.py +184 -0
  82. code_puppy/command_line/mcp/stop_all_command.py +112 -0
  83. code_puppy/command_line/mcp/stop_command.py +80 -0
  84. code_puppy/command_line/mcp/test_command.py +107 -0
  85. code_puppy/command_line/mcp/utils.py +129 -0
  86. code_puppy/command_line/mcp/wizard_utils.py +334 -0
  87. code_puppy/command_line/mcp_completion.py +174 -0
  88. code_puppy/command_line/model_picker_completion.py +197 -0
  89. code_puppy/command_line/model_settings_menu.py +932 -0
  90. code_puppy/command_line/motd.py +96 -0
  91. code_puppy/command_line/onboarding_slides.py +179 -0
  92. code_puppy/command_line/onboarding_wizard.py +342 -0
  93. code_puppy/command_line/pin_command_completion.py +329 -0
  94. code_puppy/command_line/prompt_toolkit_completion.py +846 -0
  95. code_puppy/command_line/session_commands.py +302 -0
  96. code_puppy/command_line/shell_passthrough.py +145 -0
  97. code_puppy/command_line/skills_completion.py +160 -0
  98. code_puppy/command_line/uc_menu.py +893 -0
  99. code_puppy/command_line/utils.py +93 -0
  100. code_puppy/command_line/wiggum_state.py +78 -0
  101. code_puppy/config.py +1770 -0
  102. code_puppy/error_logging.py +134 -0
  103. code_puppy/gemini_code_assist.py +385 -0
  104. code_puppy/gemini_model.py +754 -0
  105. code_puppy/hook_engine/README.md +105 -0
  106. code_puppy/hook_engine/__init__.py +21 -0
  107. code_puppy/hook_engine/aliases.py +155 -0
  108. code_puppy/hook_engine/engine.py +221 -0
  109. code_puppy/hook_engine/executor.py +296 -0
  110. code_puppy/hook_engine/matcher.py +156 -0
  111. code_puppy/hook_engine/models.py +240 -0
  112. code_puppy/hook_engine/registry.py +106 -0
  113. code_puppy/hook_engine/validator.py +144 -0
  114. code_puppy/http_utils.py +361 -0
  115. code_puppy/keymap.py +128 -0
  116. code_puppy/main.py +10 -0
  117. code_puppy/mcp_/__init__.py +66 -0
  118. code_puppy/mcp_/async_lifecycle.py +286 -0
  119. code_puppy/mcp_/blocking_startup.py +469 -0
  120. code_puppy/mcp_/captured_stdio_server.py +275 -0
  121. code_puppy/mcp_/circuit_breaker.py +290 -0
  122. code_puppy/mcp_/config_wizard.py +507 -0
  123. code_puppy/mcp_/dashboard.py +308 -0
  124. code_puppy/mcp_/error_isolation.py +407 -0
  125. code_puppy/mcp_/examples/retry_example.py +226 -0
  126. code_puppy/mcp_/health_monitor.py +589 -0
  127. code_puppy/mcp_/managed_server.py +428 -0
  128. code_puppy/mcp_/manager.py +807 -0
  129. code_puppy/mcp_/mcp_logs.py +224 -0
  130. code_puppy/mcp_/registry.py +451 -0
  131. code_puppy/mcp_/retry_manager.py +337 -0
  132. code_puppy/mcp_/server_registry_catalog.py +1126 -0
  133. code_puppy/mcp_/status_tracker.py +355 -0
  134. code_puppy/mcp_/system_tools.py +209 -0
  135. code_puppy/mcp_prompts/__init__.py +1 -0
  136. code_puppy/mcp_prompts/hook_creator.py +103 -0
  137. code_puppy/messaging/__init__.py +255 -0
  138. code_puppy/messaging/bus.py +613 -0
  139. code_puppy/messaging/commands.py +167 -0
  140. code_puppy/messaging/markdown_patches.py +57 -0
  141. code_puppy/messaging/message_queue.py +361 -0
  142. code_puppy/messaging/messages.py +569 -0
  143. code_puppy/messaging/queue_console.py +271 -0
  144. code_puppy/messaging/renderers.py +311 -0
  145. code_puppy/messaging/rich_renderer.py +1158 -0
  146. code_puppy/messaging/spinner/__init__.py +83 -0
  147. code_puppy/messaging/spinner/console_spinner.py +240 -0
  148. code_puppy/messaging/spinner/spinner_base.py +95 -0
  149. code_puppy/messaging/subagent_console.py +460 -0
  150. code_puppy/model_factory.py +848 -0
  151. code_puppy/model_switching.py +63 -0
  152. code_puppy/model_utils.py +168 -0
  153. code_puppy/models.json +174 -0
  154. code_puppy/models_dev_api.json +1 -0
  155. code_puppy/models_dev_parser.py +592 -0
  156. code_puppy/plugins/__init__.py +186 -0
  157. code_puppy/plugins/agent_skills/__init__.py +22 -0
  158. code_puppy/plugins/agent_skills/config.py +175 -0
  159. code_puppy/plugins/agent_skills/discovery.py +136 -0
  160. code_puppy/plugins/agent_skills/downloader.py +392 -0
  161. code_puppy/plugins/agent_skills/installer.py +22 -0
  162. code_puppy/plugins/agent_skills/metadata.py +219 -0
  163. code_puppy/plugins/agent_skills/prompt_builder.py +60 -0
  164. code_puppy/plugins/agent_skills/register_callbacks.py +241 -0
  165. code_puppy/plugins/agent_skills/remote_catalog.py +322 -0
  166. code_puppy/plugins/agent_skills/skill_catalog.py +257 -0
  167. code_puppy/plugins/agent_skills/skills_install_menu.py +664 -0
  168. code_puppy/plugins/agent_skills/skills_menu.py +781 -0
  169. code_puppy/plugins/antigravity_oauth/__init__.py +10 -0
  170. code_puppy/plugins/antigravity_oauth/accounts.py +406 -0
  171. code_puppy/plugins/antigravity_oauth/antigravity_model.py +706 -0
  172. code_puppy/plugins/antigravity_oauth/config.py +42 -0
  173. code_puppy/plugins/antigravity_oauth/constants.py +133 -0
  174. code_puppy/plugins/antigravity_oauth/oauth.py +478 -0
  175. code_puppy/plugins/antigravity_oauth/register_callbacks.py +518 -0
  176. code_puppy/plugins/antigravity_oauth/storage.py +288 -0
  177. code_puppy/plugins/antigravity_oauth/test_plugin.py +319 -0
  178. code_puppy/plugins/antigravity_oauth/token.py +167 -0
  179. code_puppy/plugins/antigravity_oauth/transport.py +863 -0
  180. code_puppy/plugins/antigravity_oauth/utils.py +168 -0
  181. code_puppy/plugins/chatgpt_oauth/__init__.py +8 -0
  182. code_puppy/plugins/chatgpt_oauth/config.py +52 -0
  183. code_puppy/plugins/chatgpt_oauth/oauth_flow.py +329 -0
  184. code_puppy/plugins/chatgpt_oauth/register_callbacks.py +176 -0
  185. code_puppy/plugins/chatgpt_oauth/test_plugin.py +301 -0
  186. code_puppy/plugins/chatgpt_oauth/utils.py +523 -0
  187. code_puppy/plugins/claude_code_hooks/__init__.py +1 -0
  188. code_puppy/plugins/claude_code_hooks/config.py +137 -0
  189. code_puppy/plugins/claude_code_hooks/register_callbacks.py +175 -0
  190. code_puppy/plugins/claude_code_oauth/README.md +167 -0
  191. code_puppy/plugins/claude_code_oauth/SETUP.md +93 -0
  192. code_puppy/plugins/claude_code_oauth/__init__.py +25 -0
  193. code_puppy/plugins/claude_code_oauth/config.py +52 -0
  194. code_puppy/plugins/claude_code_oauth/register_callbacks.py +453 -0
  195. code_puppy/plugins/claude_code_oauth/test_plugin.py +283 -0
  196. code_puppy/plugins/claude_code_oauth/token_refresh_heartbeat.py +241 -0
  197. code_puppy/plugins/claude_code_oauth/utils.py +640 -0
  198. code_puppy/plugins/customizable_commands/__init__.py +0 -0
  199. code_puppy/plugins/customizable_commands/register_callbacks.py +152 -0
  200. code_puppy/plugins/example_custom_command/README.md +280 -0
  201. code_puppy/plugins/example_custom_command/register_callbacks.py +51 -0
  202. code_puppy/plugins/file_permission_handler/__init__.py +4 -0
  203. code_puppy/plugins/file_permission_handler/register_callbacks.py +470 -0
  204. code_puppy/plugins/frontend_emitter/__init__.py +25 -0
  205. code_puppy/plugins/frontend_emitter/emitter.py +121 -0
  206. code_puppy/plugins/frontend_emitter/register_callbacks.py +261 -0
  207. code_puppy/plugins/hook_creator/__init__.py +1 -0
  208. code_puppy/plugins/hook_creator/register_callbacks.py +33 -0
  209. code_puppy/plugins/hook_manager/__init__.py +1 -0
  210. code_puppy/plugins/hook_manager/config.py +290 -0
  211. code_puppy/plugins/hook_manager/hooks_menu.py +564 -0
  212. code_puppy/plugins/hook_manager/register_callbacks.py +227 -0
  213. code_puppy/plugins/oauth_puppy_html.py +228 -0
  214. code_puppy/plugins/scheduler/__init__.py +1 -0
  215. code_puppy/plugins/scheduler/register_callbacks.py +88 -0
  216. code_puppy/plugins/scheduler/scheduler_menu.py +522 -0
  217. code_puppy/plugins/scheduler/scheduler_wizard.py +341 -0
  218. code_puppy/plugins/shell_safety/__init__.py +6 -0
  219. code_puppy/plugins/shell_safety/agent_shell_safety.py +69 -0
  220. code_puppy/plugins/shell_safety/command_cache.py +156 -0
  221. code_puppy/plugins/shell_safety/register_callbacks.py +202 -0
  222. code_puppy/plugins/synthetic_status/__init__.py +1 -0
  223. code_puppy/plugins/synthetic_status/register_callbacks.py +132 -0
  224. code_puppy/plugins/synthetic_status/status_api.py +147 -0
  225. code_puppy/plugins/universal_constructor/__init__.py +13 -0
  226. code_puppy/plugins/universal_constructor/models.py +138 -0
  227. code_puppy/plugins/universal_constructor/register_callbacks.py +47 -0
  228. code_puppy/plugins/universal_constructor/registry.py +302 -0
  229. code_puppy/plugins/universal_constructor/sandbox.py +584 -0
  230. code_puppy/prompts/antigravity_system_prompt.md +1 -0
  231. code_puppy/pydantic_patches.py +356 -0
  232. code_puppy/reopenable_async_client.py +232 -0
  233. code_puppy/round_robin_model.py +150 -0
  234. code_puppy/scheduler/__init__.py +41 -0
  235. code_puppy/scheduler/__main__.py +9 -0
  236. code_puppy/scheduler/cli.py +118 -0
  237. code_puppy/scheduler/config.py +126 -0
  238. code_puppy/scheduler/daemon.py +280 -0
  239. code_puppy/scheduler/executor.py +155 -0
  240. code_puppy/scheduler/platform.py +19 -0
  241. code_puppy/scheduler/platform_unix.py +22 -0
  242. code_puppy/scheduler/platform_win.py +32 -0
  243. code_puppy/session_storage.py +338 -0
  244. code_puppy/status_display.py +257 -0
  245. code_puppy/summarization_agent.py +176 -0
  246. code_puppy/terminal_utils.py +418 -0
  247. code_puppy/tools/__init__.py +501 -0
  248. code_puppy/tools/agent_tools.py +603 -0
  249. code_puppy/tools/ask_user_question/__init__.py +26 -0
  250. code_puppy/tools/ask_user_question/constants.py +73 -0
  251. code_puppy/tools/ask_user_question/demo_tui.py +55 -0
  252. code_puppy/tools/ask_user_question/handler.py +232 -0
  253. code_puppy/tools/ask_user_question/models.py +304 -0
  254. code_puppy/tools/ask_user_question/registration.py +26 -0
  255. code_puppy/tools/ask_user_question/renderers.py +309 -0
  256. code_puppy/tools/ask_user_question/terminal_ui.py +329 -0
  257. code_puppy/tools/ask_user_question/theme.py +155 -0
  258. code_puppy/tools/ask_user_question/tui_loop.py +423 -0
  259. code_puppy/tools/browser/__init__.py +37 -0
  260. code_puppy/tools/browser/browser_control.py +289 -0
  261. code_puppy/tools/browser/browser_interactions.py +545 -0
  262. code_puppy/tools/browser/browser_locators.py +640 -0
  263. code_puppy/tools/browser/browser_manager.py +378 -0
  264. code_puppy/tools/browser/browser_navigation.py +251 -0
  265. code_puppy/tools/browser/browser_screenshot.py +179 -0
  266. code_puppy/tools/browser/browser_scripts.py +462 -0
  267. code_puppy/tools/browser/browser_workflows.py +221 -0
  268. code_puppy/tools/browser/chromium_terminal_manager.py +259 -0
  269. code_puppy/tools/browser/terminal_command_tools.py +534 -0
  270. code_puppy/tools/browser/terminal_screenshot_tools.py +552 -0
  271. code_puppy/tools/browser/terminal_tools.py +525 -0
  272. code_puppy/tools/command_runner.py +1346 -0
  273. code_puppy/tools/common.py +1409 -0
  274. code_puppy/tools/display.py +84 -0
  275. code_puppy/tools/file_modifications.py +886 -0
  276. code_puppy/tools/file_operations.py +802 -0
  277. code_puppy/tools/scheduler_tools.py +412 -0
  278. code_puppy/tools/skills_tools.py +244 -0
  279. code_puppy/tools/subagent_context.py +158 -0
  280. code_puppy/tools/tools_content.py +51 -0
  281. code_puppy/tools/universal_constructor.py +889 -0
  282. code_puppy/uvx_detection.py +242 -0
  283. code_puppy/version_checker.py +82 -0
  284. codepp-0.0.437.dist-info/METADATA +766 -0
  285. codepp-0.0.437.dist-info/RECORD +288 -0
  286. codepp-0.0.437.dist-info/WHEEL +4 -0
  287. codepp-0.0.437.dist-info/entry_points.txt +3 -0
  288. codepp-0.0.437.dist-info/licenses/LICENSE +21 -0
@@ -0,0 +1,532 @@
1
+ """Interactive TUI for configuring banner colors.
2
+
3
+ Similar to diff_menu.py but for customizing the banner background colors
4
+ for different tool outputs (THINKING, SHELL COMMAND, READ FILE, etc.).
5
+
6
+ Use /colors to launch the TUI and customize your banners!
7
+ """
8
+
9
+ import asyncio
10
+ import io
11
+ import sys
12
+ from typing import Callable, Optional
13
+
14
+ from prompt_toolkit import Application
15
+ from prompt_toolkit.formatted_text import ANSI, FormattedText
16
+ from prompt_toolkit.key_binding import KeyBindings
17
+ from prompt_toolkit.layout import Layout, VSplit, Window
18
+ from prompt_toolkit.layout.controls import FormattedTextControl
19
+ from prompt_toolkit.widgets import Frame
20
+ from rich.console import Console
21
+
22
+ # Banner display names with icons
23
+ BANNER_DISPLAY_INFO = {
24
+ "thinking": ("THINKING", "⚡"),
25
+ "agent_response": ("AGENT RESPONSE", ""),
26
+ "shell_command": ("SHELL COMMAND", "🚀"),
27
+ "read_file": ("READ FILE", "📂"),
28
+ "edit_file": ("EDIT FILE", "✏️"),
29
+ "create_file": ("CREATE FILE", "📝"),
30
+ "replace_in_file": ("REPLACE IN FILE", "✏️"),
31
+ "delete_snippet": ("DELETE SNIPPET", "✂️"),
32
+ "grep": ("GREP", "📂"),
33
+ "directory_listing": ("DIRECTORY LISTING", "📂"),
34
+ "agent_reasoning": ("AGENT REASONING", ""),
35
+ "invoke_agent": ("🤖 INVOKE AGENT", ""),
36
+ "subagent_response": ("✓ AGENT RESPONSE", ""),
37
+ "list_agents": ("LIST AGENTS", ""),
38
+ "universal_constructor": ("UNIVERSAL CONSTRUCTOR", "🔧"),
39
+ "terminal_tool": ("TERMINAL TOOL", "🖥️"),
40
+ }
41
+
42
+ # Sample content to show after each banner
43
+ BANNER_SAMPLE_CONTENT = {
44
+ "thinking": "Let me analyze this code structure and figure out the best approach...",
45
+ "agent_response": "I've implemented the feature you requested. The changes include...",
46
+ "shell_command": "$ npm run test -- --silent\n⏱ Timeout: 60s",
47
+ "read_file": "/path/to/file.py (lines 1-50)",
48
+ "edit_file": "MODIFY /path/to/file.py\n--- a/file.py\n+++ b/file.py",
49
+ "create_file": "CREATE /path/to/new_file.py\nFile created successfully.",
50
+ "replace_in_file": "MODIFY /path/to/file.py\n--- a/file.py\n+++ b/file.py",
51
+ "delete_snippet": "MODIFY /path/to/file.py\nSnippet deleted from file.",
52
+ "grep": "/src for 'handleClick'\n📄 Button.tsx (3 matches)",
53
+ "directory_listing": "/src (recursive=True)\n📁 components/\n └── Button.tsx",
54
+ "agent_reasoning": "Current reasoning:\nI need to refactor this function...",
55
+ "invoke_agent": "code-reviewer (New session)\nSession: review-auth-abc123",
56
+ "subagent_response": "code-reviewer\nThe code looks good overall...",
57
+ "list_agents": "- code-puppy: Code Puppy 🐶\n- planning-agent: Planning Agent",
58
+ "universal_constructor": "action=create tool_name=api.weather\n✅ Created successfully",
59
+ "terminal_tool": "$ chromium --headless\nBrowser terminal session started",
60
+ }
61
+
62
+ # Available background colors grouped by theme
63
+ BANNER_COLORS = {
64
+ # Cool colors
65
+ "blue": "blue",
66
+ "dark blue": "dark_blue",
67
+ "navy blue": "navy_blue",
68
+ "deep sky blue": "deep_sky_blue4",
69
+ "steel blue": "steel_blue",
70
+ "dodger blue": "dodger_blue3",
71
+ # Cyans & Teals
72
+ "dark cyan": "dark_cyan",
73
+ "cyan": "cyan4",
74
+ "teal": "dark_turquoise",
75
+ "aquamarine": "aquamarine1",
76
+ # Greens
77
+ "green": "green4",
78
+ "dark green": "dark_green",
79
+ "sea green": "dark_sea_green4",
80
+ "spring green": "spring_green4",
81
+ "chartreuse": "chartreuse4",
82
+ # Purples & Magentas
83
+ "purple": "purple",
84
+ "dark magenta": "dark_magenta",
85
+ "medium purple": "medium_purple4",
86
+ "dark violet": "dark_violet",
87
+ "plum": "plum4",
88
+ "orchid": "dark_orchid",
89
+ # Reds & Oranges
90
+ "red": "red3",
91
+ "dark red": "dark_red",
92
+ "indian red": "indian_red",
93
+ "orange red": "orange_red1",
94
+ "orange": "dark_orange3",
95
+ # Yellows & Golds
96
+ "gold": "gold3",
97
+ "dark goldenrod": "dark_goldenrod",
98
+ "olive": "dark_olive_green3",
99
+ # Grays
100
+ "grey30": "grey30",
101
+ "grey37": "grey37",
102
+ "grey42": "grey42",
103
+ "grey50": "grey50",
104
+ "grey58": "grey58",
105
+ "dark slate gray": "dark_slate_gray3",
106
+ # Pink tones
107
+ "hot pink": "hot_pink3",
108
+ "deep pink": "deep_pink4",
109
+ "pale violet red": "pale_violet_red1",
110
+ }
111
+
112
+
113
+ class ColorConfiguration:
114
+ """Holds the current banner color configuration state."""
115
+
116
+ def __init__(self):
117
+ """Initialize configuration from current settings."""
118
+ from code_puppy.config import get_all_banner_colors
119
+
120
+ self.current_colors = get_all_banner_colors()
121
+ self.original_colors = self.current_colors.copy()
122
+ self.selected_banner_index = 0
123
+ self.banner_keys = list(BANNER_DISPLAY_INFO.keys())
124
+
125
+ def has_changes(self) -> bool:
126
+ """Check if any changes have been made."""
127
+ return self.current_colors != self.original_colors
128
+
129
+ def get_current_banner_key(self) -> str:
130
+ """Get the currently selected banner key."""
131
+ return self.banner_keys[self.selected_banner_index]
132
+
133
+ def get_current_banner_color(self) -> str:
134
+ """Get the color of the currently selected banner."""
135
+ return self.current_colors[self.get_current_banner_key()]
136
+
137
+ def set_current_banner_color(self, color: str):
138
+ """Set the color of the currently selected banner."""
139
+ self.current_colors[self.get_current_banner_key()] = color
140
+
141
+ def next_banner(self):
142
+ """Cycle to the next banner."""
143
+ self.selected_banner_index = (self.selected_banner_index + 1) % len(
144
+ self.banner_keys
145
+ )
146
+
147
+ def prev_banner(self):
148
+ """Cycle to the previous banner."""
149
+ self.selected_banner_index = (self.selected_banner_index - 1) % len(
150
+ self.banner_keys
151
+ )
152
+
153
+
154
+ async def interactive_colors_picker() -> Optional[dict]:
155
+ """Show an interactive full-screen terminal UI to configure banner colors.
156
+
157
+ Returns:
158
+ A dict with changes or None if cancelled
159
+ """
160
+ from code_puppy.tools.command_runner import set_awaiting_user_input
161
+
162
+ config = ColorConfiguration()
163
+
164
+ set_awaiting_user_input(True)
165
+
166
+ # Enter alternate screen buffer once for entire session
167
+ sys.stdout.write("\033[?1049h") # Enter alternate buffer
168
+ sys.stdout.write("\033[2J\033[H") # Clear and home
169
+ sys.stdout.flush()
170
+ await asyncio.sleep(0.1) # Minimal delay for state sync
171
+
172
+ try:
173
+ # Main menu loop
174
+ while True:
175
+ choices = []
176
+ for key in config.banner_keys:
177
+ display_name, icon = BANNER_DISPLAY_INFO[key]
178
+ current_color = config.current_colors[key]
179
+ choices.append(f"{display_name} [{current_color}]")
180
+
181
+ # Add action items
182
+ if config.has_changes():
183
+ choices.append("─── Actions ───")
184
+ choices.append("💾 Save & Exit")
185
+ choices.append("🔄 Reset All to Defaults")
186
+ choices.append("❌ Discard & Exit")
187
+ else:
188
+ choices.append("─── Actions ───")
189
+ choices.append("🔄 Reset All to Defaults")
190
+ choices.append("❌ Exit")
191
+
192
+ def dummy_update(choice: str):
193
+ pass
194
+
195
+ def get_main_preview():
196
+ return _get_preview_text_for_prompt_toolkit(config)
197
+
198
+ try:
199
+ selected = await _split_panel_selector(
200
+ "Banner Color Configuration",
201
+ choices,
202
+ dummy_update,
203
+ get_preview=get_main_preview,
204
+ config=config,
205
+ )
206
+ except KeyboardInterrupt:
207
+ break
208
+
209
+ # Handle selection
210
+ if selected is None:
211
+ break
212
+ elif "Save & Exit" in selected:
213
+ break
214
+ elif "Reset All" in selected:
215
+ from code_puppy.config import DEFAULT_BANNER_COLORS
216
+
217
+ config.current_colors = DEFAULT_BANNER_COLORS.copy()
218
+ elif "Discard" in selected or selected == "❌ Exit":
219
+ config.current_colors = config.original_colors.copy()
220
+ break
221
+ elif "───" in selected:
222
+ # Separator - do nothing
223
+ pass
224
+ else:
225
+ # A banner was selected - show color picker
226
+ # Find which banner was selected
227
+ for i, key in enumerate(config.banner_keys):
228
+ display_name, _ = BANNER_DISPLAY_INFO[key]
229
+ if selected.startswith(display_name):
230
+ config.selected_banner_index = i
231
+ await _handle_color_menu(config)
232
+ break
233
+
234
+ except Exception:
235
+ # Silent error - just exit cleanly
236
+ return None
237
+ finally:
238
+ set_awaiting_user_input(False)
239
+ # Exit alternate screen buffer once at end
240
+ sys.stdout.write("\033[?1049l") # Exit alternate buffer
241
+ sys.stdout.flush()
242
+
243
+ # Clear exit message
244
+ from code_puppy.messaging import emit_info
245
+
246
+ emit_info("✓ Exited banner color configuration")
247
+
248
+ # Return changes if any
249
+ if config.has_changes():
250
+ return config.current_colors
251
+
252
+ return None
253
+
254
+
255
+ async def _split_panel_selector(
256
+ title: str,
257
+ choices: list[str],
258
+ on_change: Callable[[str], None],
259
+ get_preview: Callable[[], ANSI],
260
+ config: Optional[ColorConfiguration] = None,
261
+ ) -> Optional[str]:
262
+ """Split-panel selector with menu on left and live preview on right."""
263
+ selected_index = [0]
264
+ result = [None]
265
+
266
+ def get_left_panel_text():
267
+ """Generate the selector menu text."""
268
+ try:
269
+ lines = []
270
+ lines.append(("bold cyan", title))
271
+ lines.append(("", "\n\n"))
272
+
273
+ if not choices:
274
+ lines.append(("fg:ansiyellow", "No choices available"))
275
+ lines.append(("", "\n"))
276
+ else:
277
+ for i, choice in enumerate(choices):
278
+ # Skip separator lines for selection highlighting
279
+ if "───" in choice:
280
+ lines.append(("fg:ansigray", f" {choice}"))
281
+ lines.append(("", "\n"))
282
+ elif i == selected_index[0]:
283
+ lines.append(("fg:ansigreen", "▶ "))
284
+ lines.append(("fg:ansigreen bold", choice))
285
+ lines.append(("", "\n"))
286
+ else:
287
+ lines.append(("", " "))
288
+ lines.append(("", choice))
289
+ lines.append(("", "\n"))
290
+
291
+ lines.append(("", "\n"))
292
+ lines.append(
293
+ ("fg:ansicyan", "↑↓ Navigate │ Enter Select │ Ctrl-C Cancel")
294
+ )
295
+ return FormattedText(lines)
296
+ except Exception as e:
297
+ return FormattedText([("fg:ansired", f"Error: {e}")])
298
+
299
+ def get_right_panel_text():
300
+ """Generate the preview panel text."""
301
+ try:
302
+ preview = get_preview()
303
+ return preview
304
+ except Exception as e:
305
+ return FormattedText([("fg:ansired", f"Preview error: {e}")])
306
+
307
+ kb = KeyBindings()
308
+
309
+ @kb.add("up")
310
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
311
+ def move_up(event):
312
+ if choices:
313
+ # Skip separator lines
314
+ new_idx = (selected_index[0] - 1) % len(choices)
315
+ while "───" in choices[new_idx]:
316
+ new_idx = (new_idx - 1) % len(choices)
317
+ selected_index[0] = new_idx
318
+ on_change(choices[selected_index[0]])
319
+ event.app.invalidate()
320
+
321
+ @kb.add("down")
322
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
323
+ def move_down(event):
324
+ if choices:
325
+ # Skip separator lines
326
+ new_idx = (selected_index[0] + 1) % len(choices)
327
+ while "───" in choices[new_idx]:
328
+ new_idx = (new_idx + 1) % len(choices)
329
+ selected_index[0] = new_idx
330
+ on_change(choices[selected_index[0]])
331
+ event.app.invalidate()
332
+
333
+ @kb.add("enter")
334
+ def accept(event):
335
+ if choices:
336
+ result[0] = choices[selected_index[0]]
337
+ else:
338
+ result[0] = None
339
+ event.app.exit()
340
+
341
+ @kb.add("c-c")
342
+ def cancel(event):
343
+ result[0] = None
344
+ event.app.exit()
345
+
346
+ # Create split layout with left (selector) and right (preview) panels
347
+ left_panel = Window(
348
+ content=FormattedTextControl(lambda: get_left_panel_text()),
349
+ width=45,
350
+ )
351
+
352
+ right_panel = Window(
353
+ content=FormattedTextControl(lambda: get_right_panel_text()),
354
+ )
355
+
356
+ # Create vertical split (side-by-side panels)
357
+ root_container = VSplit(
358
+ [
359
+ Frame(left_panel, title="Menu"),
360
+ Frame(right_panel, title="Preview"),
361
+ ]
362
+ )
363
+
364
+ layout = Layout(root_container)
365
+ app = Application(
366
+ layout=layout,
367
+ key_bindings=kb,
368
+ full_screen=False,
369
+ mouse_support=False,
370
+ color_depth="DEPTH_24_BIT",
371
+ )
372
+
373
+ sys.stdout.flush()
374
+
375
+ # Trigger initial update only if choices is not empty
376
+ if choices:
377
+ on_change(choices[selected_index[0]])
378
+
379
+ # Clear the current buffer
380
+ sys.stdout.write("\033[2J\033[H")
381
+ sys.stdout.flush()
382
+
383
+ # Run application
384
+ await app.run_async()
385
+
386
+ if result[0] is None:
387
+ raise KeyboardInterrupt()
388
+
389
+ return result[0]
390
+
391
+
392
+ def _get_preview_text_for_prompt_toolkit(config: ColorConfiguration) -> ANSI:
393
+ """Get preview as ANSI for embedding in selector with live colors.
394
+
395
+ Returns ANSI-formatted text that prompt_toolkit can render with full colors.
396
+ """
397
+ # Build preview showing all banners with their current colors
398
+ buffer = io.StringIO()
399
+ console = Console(
400
+ file=buffer,
401
+ force_terminal=True,
402
+ width=70,
403
+ legacy_windows=False,
404
+ color_system="truecolor",
405
+ no_color=False,
406
+ force_interactive=True,
407
+ )
408
+
409
+ # Header
410
+ console.print("[bold]═" * 60 + "[/bold]")
411
+ console.print("[bold cyan] LIVE PREVIEW - Banner Colors[/bold cyan]")
412
+ console.print("[bold]═" * 60 + "[/bold]")
413
+ console.print()
414
+
415
+ # Show each banner with its current color
416
+ for key in config.banner_keys:
417
+ display_name, icon = BANNER_DISPLAY_INFO[key]
418
+ color = config.current_colors[key]
419
+ sample = BANNER_SAMPLE_CONTENT[key]
420
+
421
+ # Highlight the currently selected banner
422
+ is_selected = key == config.get_current_banner_key()
423
+ if is_selected:
424
+ console.print("[bold yellow]▶[/bold yellow] ", end="")
425
+ else:
426
+ console.print(" ", end="")
427
+
428
+ # Print the banner with its configured color
429
+ icon_str = f" {icon}" if icon else ""
430
+ banner_text = (
431
+ f"[bold white on {color}] {display_name} [/bold white on {color}]{icon_str}"
432
+ )
433
+ console.print(banner_text)
434
+
435
+ # Print sample content (dimmed)
436
+ sample_lines = sample.split("\n")
437
+ for line in sample_lines[:2]: # Only show first 2 lines
438
+ if is_selected:
439
+ console.print(f" [dim]{line}[/dim]")
440
+ else:
441
+ console.print(f" [dim]{line}[/dim]")
442
+ console.print()
443
+
444
+ console.print("[bold]═" * 60 + "[/bold]")
445
+
446
+ ansi_output = buffer.getvalue()
447
+ return ANSI(ansi_output)
448
+
449
+
450
+ async def _handle_color_menu(config: ColorConfiguration) -> None:
451
+ """Handle color selection for the current banner."""
452
+ banner_key = config.get_current_banner_key()
453
+ display_name, _ = BANNER_DISPLAY_INFO[banner_key]
454
+ current_color = config.get_current_banner_color()
455
+ title = f"Select color for {display_name}:"
456
+
457
+ # Build choices with color names
458
+ choices = []
459
+ for name, color_value in BANNER_COLORS.items():
460
+ marker = " ← current" if color_value == current_color else ""
461
+ choices.append(f"{name}{marker}")
462
+
463
+ # Store original color for potential cancellation
464
+ original_color = current_color
465
+
466
+ # Callback for live preview updates
467
+ def update_preview(selected_choice: str):
468
+ color_name = selected_choice.replace(" ← current", "").strip()
469
+ selected_color = BANNER_COLORS.get(color_name, "blue")
470
+ config.set_current_banner_color(selected_color)
471
+
472
+ def get_preview_header():
473
+ return _get_single_banner_preview(config)
474
+
475
+ try:
476
+ await _split_panel_selector(
477
+ title,
478
+ choices,
479
+ update_preview,
480
+ get_preview=get_preview_header,
481
+ config=config,
482
+ )
483
+ except KeyboardInterrupt:
484
+ # Restore original color on cancel
485
+ config.set_current_banner_color(original_color)
486
+ except Exception:
487
+ pass # Silent error handling
488
+
489
+
490
+ def _get_single_banner_preview(config: ColorConfiguration) -> ANSI:
491
+ """Get preview for a single banner being edited."""
492
+ buffer = io.StringIO()
493
+ console = Console(
494
+ file=buffer,
495
+ force_terminal=True,
496
+ width=70,
497
+ legacy_windows=False,
498
+ color_system="truecolor",
499
+ no_color=False,
500
+ force_interactive=True,
501
+ )
502
+
503
+ banner_key = config.get_current_banner_key()
504
+ display_name, icon = BANNER_DISPLAY_INFO[banner_key]
505
+ color = config.get_current_banner_color()
506
+ sample = BANNER_SAMPLE_CONTENT[banner_key]
507
+
508
+ # Header
509
+ console.print("[bold]═" * 60 + "[/bold]")
510
+ console.print(f"[bold cyan] Editing: {display_name}[/bold cyan]")
511
+ console.print(f" Current Color: [bold]{color}[/bold]")
512
+ console.print("[bold]═" * 60 + "[/bold]")
513
+ console.print()
514
+
515
+ # Show the banner large
516
+ icon_str = f" {icon}" if icon else ""
517
+ banner_text = (
518
+ f"[bold white on {color}] {display_name} [/bold white on {color}]{icon_str}"
519
+ )
520
+ console.print(banner_text)
521
+ console.print()
522
+
523
+ # Show sample content
524
+ console.print("[dim]Sample output:[/dim]")
525
+ for line in sample.split("\n"):
526
+ console.print(f"[dim]{line}[/dim]")
527
+
528
+ console.print()
529
+ console.print("[bold]═" * 60 + "[/bold]")
530
+
531
+ ansi_output = buffer.getvalue()
532
+ return ANSI(ansi_output)