generated from nhcarrigan/template
f173892aaa
## Summary This PR includes major feature additions, bug fixes, comprehensive testing improvements, and responsive design enhancements! ## New Features โจ ### Plugin & MCP Management (#133, #134) - **Plugin Management Panel**: Install, uninstall, enable/disable, and update plugins - **MCP Server Management Panel**: Add/remove MCP servers, view detailed configuration - **Marketplace Management**: Add/remove plugin marketplaces from GitHub - Backend commands for full CLI integration (`list_plugins`, `install_plugin`, `add_mcp_server`, etc.) - Beautiful UI with proper loading states, error handling, and theme support ### Visual Todo List Panel (#132) - Real-time todo list display when Hikari uses the `TodoWrite` tool - Shows pending/in-progress/completed status with visual indicators - Progress bar and completion count - Automatically clears on disconnect - Theme-aware styling ### Clear Session History Button (#130) - "Clear All Sessions" button in Session History panel - Confirmation dialog with session count - Keyboard support and accessibility features - Gives users control over disk usage ### CLI Version Display (#131) - Displays Claude CLI version in status bar - Auto-polls every 30 seconds for updates - Useful for debugging and feature compatibility ## Bug Fixes ๐ ### Stats Panel Scrolling (#136) - **Fixed stats panel overflow**: Added scrollable container with `max-height` constraint - Stats panel now scrolls when content (Tools Used, Historical Costs, Budget sections) gets too long - Prevents content from overflowing off screen ### Agent Monitor Fixes (#122) - **Fixed agents stuck in "running" state**: Added `SubagentStop` hook parsing - **Fixed agents persisting after disconnect**: Call `clearConversation()` on disconnect - **Fixed "Kill All" button**: Now properly marks all agents as errored - **Fixed badge persisting after tab close**: Cleanup agents when conversation is deleted - Comprehensive tests for agent lifecycle management ### Discord RPC Cleanup (#129) - Removed file-based logging for Discord RPC - Replaced with proper `tracing` framework usage - Reduces disk usage and eliminates maintenance burden ### Close Modal Bug Fix (#128) - Fixed close confirmation modal not triggering after Discord RPC refactor - Removed frontend calls to deleted `log_discord_rpc` command - Modal now works correctly after all operations ### Responsive Design Fixes (#118) - Fixed top navigation icons getting cut off at small screen widths - Fixed Connect button disappearing on narrow screens - Fixed bottom status info (clock, CLI version) getting cut off - Added flex-wrap and mobile-optimised layouts - Icons-only mode on screens < 640px - Vertical stacking on screens < 768px ## Testing Improvements ๐งช ### Comprehensive Test Coverage (#114) - **417 backend tests** (up from 408) - **387 frontend tests** (up from 363) - **61%+ backend code coverage** - Added E2E integration tests for cross-platform notification commands - New test files: `agents.test.ts`, comprehensive CLI parsing tests - Tests for `debug_logger.rs`, `bridge_manager.rs`, `notifications.rs` - Console mocking for cleaner test output - Fixed flaky frontend tests ### Testing Documentation - Updated CLAUDE.md with comprehensive testing guidelines - Documented mocking approaches (console mocking, E2E command structure testing) - Added step-by-step guide for adding tests to new features - Goal to maintain ~100% test coverage documented ## Closes Closes #114 Closes #118 Closes #122 Closes #128 Closes #129 Closes #130 Closes #131 Closes #132 Closes #133 Closes #134 Closes #136 ## Technical Details - All new backend commands properly registered in `lib.rs` - CLI output parsing with comprehensive test coverage - Cross-platform compatibility verified through E2E tests (Linux CI can test Windows commands) - Theme-aware UI components using CSS variables throughout - Proper TypeScript types for all new stores and components - ESLint and Prettier compliant - All Clippy warnings addressed โจ This PR was created with help from Hikari~ ๐ธ Reviewed-on: #135 Co-authored-by: Hikari <hikari@nhcarrigan.com> Co-committed-by: Hikari <hikari@nhcarrigan.com>
216 lines
6.9 KiB
Rust
216 lines
6.9 KiB
Rust
mod achievements;
|
|
mod bridge_manager;
|
|
mod clipboard;
|
|
mod commands;
|
|
mod config;
|
|
mod cost_tracking;
|
|
mod debug_logger;
|
|
mod discord_rpc;
|
|
mod git;
|
|
mod notifications;
|
|
mod quick_actions;
|
|
mod sessions;
|
|
mod snippets;
|
|
mod stats;
|
|
mod temp_manager;
|
|
mod tool_cache;
|
|
mod tray;
|
|
mod types;
|
|
mod vbs_notification;
|
|
mod windows_toast;
|
|
mod wsl_bridge;
|
|
mod wsl_notifications;
|
|
|
|
use bridge_manager::create_shared_bridge_manager;
|
|
use clipboard::*;
|
|
use commands::load_saved_achievements;
|
|
use commands::*;
|
|
use debug_logger::TauriLogLayer;
|
|
use discord_rpc::DiscordRpcManager;
|
|
use git::*;
|
|
use notifications::*;
|
|
use quick_actions::*;
|
|
use sessions::*;
|
|
use snippets::*;
|
|
use std::sync::Arc;
|
|
use tauri::{Emitter, Manager};
|
|
use temp_manager::create_shared_temp_manager;
|
|
use tracing_subscriber::layer::SubscriberExt;
|
|
use tracing_subscriber::util::SubscriberInitExt;
|
|
use tray::setup_tray;
|
|
use vbs_notification::*;
|
|
use windows_toast::*;
|
|
use wsl_notifications::*;
|
|
|
|
#[cfg_attr(mobile, tauri::mobile_entry_point)]
|
|
pub fn run() {
|
|
let bridge_manager = create_shared_bridge_manager();
|
|
let temp_manager = create_shared_temp_manager().expect("Failed to create temp file manager");
|
|
let discord_rpc = Arc::new(DiscordRpcManager::new());
|
|
|
|
tauri::Builder::default()
|
|
.plugin(tauri_plugin_dialog::init())
|
|
.plugin(tauri_plugin_opener::init())
|
|
.plugin(tauri_plugin_shell::init())
|
|
.plugin(tauri_plugin_store::Builder::new().build())
|
|
.plugin(tauri_plugin_notification::init())
|
|
.plugin(tauri_plugin_os::init())
|
|
.plugin(tauri_plugin_http::init())
|
|
.plugin(tauri_plugin_clipboard_manager::init())
|
|
.plugin(tauri_plugin_fs::init())
|
|
.manage(bridge_manager.clone())
|
|
.manage(temp_manager.clone())
|
|
.manage(discord_rpc.clone())
|
|
.setup(move |app| {
|
|
// Initialize tracing with custom layer that emits to frontend
|
|
// NOTE: We don't use fmt::layer() because in production builds with windows_subsystem = "windows",
|
|
// stdout is hidden. Instead, all logs go through TauriLogLayer to the debug console.
|
|
let tauri_layer = TauriLogLayer::new(app.handle().clone());
|
|
tracing_subscriber::registry()
|
|
.with(tauri_layer)
|
|
.init();
|
|
|
|
// Initialize the app handle in the bridge manager
|
|
bridge_manager.lock().set_app_handle(app.handle().clone());
|
|
|
|
// Clean up any orphaned temp files from previous sessions
|
|
if let Ok(count) = temp_manager.lock().cleanup_orphaned_files() {
|
|
if count > 0 {
|
|
tracing::info!("Cleaned up {} orphaned temp files", count);
|
|
}
|
|
}
|
|
|
|
tracing::info!("Hikari Desktop started successfully");
|
|
|
|
// Set up system tray
|
|
if let Err(e) = setup_tray(app.handle()) {
|
|
tracing::error!("Failed to set up system tray: {}", e);
|
|
}
|
|
|
|
// Handle window close event for minimize to tray and close confirmation
|
|
let main_window = app.get_webview_window("main").unwrap();
|
|
main_window.on_window_event({
|
|
let app_handle = app.handle().clone();
|
|
move |event| {
|
|
if let tauri::WindowEvent::CloseRequested { api, .. } = event {
|
|
// Always prevent default close - let frontend handle it
|
|
api.prevent_close();
|
|
|
|
// Emit event to frontend to show confirmation modal
|
|
if let Some(window) = app_handle.get_webview_window("main") {
|
|
let _ = window.emit("window-close-requested", ());
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
Ok(())
|
|
})
|
|
.invoke_handler(tauri::generate_handler![
|
|
start_claude,
|
|
stop_claude,
|
|
interrupt_claude,
|
|
send_prompt,
|
|
is_claude_running,
|
|
get_working_directory,
|
|
select_wsl_directory,
|
|
get_config,
|
|
save_config,
|
|
get_usage_stats,
|
|
get_persisted_stats,
|
|
load_saved_achievements,
|
|
answer_question,
|
|
send_windows_notification,
|
|
send_simple_notification,
|
|
send_windows_toast,
|
|
send_notify_send,
|
|
send_wsl_notification,
|
|
send_vbs_notification,
|
|
validate_directory,
|
|
list_skills,
|
|
check_for_updates,
|
|
save_temp_file,
|
|
register_temp_file,
|
|
get_temp_files,
|
|
cleanup_temp_files,
|
|
cleanup_all_temp_files,
|
|
cleanup_orphaned_temp_files,
|
|
get_file_size,
|
|
list_sessions,
|
|
save_session,
|
|
load_session,
|
|
delete_session,
|
|
search_sessions,
|
|
clear_all_sessions,
|
|
list_snippets,
|
|
save_snippet,
|
|
delete_snippet,
|
|
get_snippet_categories,
|
|
reset_default_snippets,
|
|
list_quick_actions,
|
|
save_quick_action,
|
|
delete_quick_action,
|
|
reset_default_quick_actions,
|
|
git_status,
|
|
git_diff,
|
|
git_branches,
|
|
git_checkout,
|
|
git_stage,
|
|
git_unstage,
|
|
git_stage_all,
|
|
git_commit,
|
|
git_push,
|
|
git_pull,
|
|
git_fetch,
|
|
git_log,
|
|
git_discard,
|
|
git_create_branch,
|
|
list_clipboard_entries,
|
|
capture_clipboard,
|
|
delete_clipboard_entry,
|
|
toggle_pin_clipboard_entry,
|
|
clear_clipboard_history,
|
|
search_clipboard_entries,
|
|
get_clipboard_languages,
|
|
update_clipboard_language,
|
|
list_directory,
|
|
read_file_content,
|
|
write_file_content,
|
|
create_file,
|
|
create_directory,
|
|
delete_file,
|
|
delete_directory,
|
|
rename_path,
|
|
// Cost tracking commands
|
|
get_cost_summary,
|
|
get_cost_alerts,
|
|
set_cost_alert_thresholds,
|
|
export_cost_csv,
|
|
get_today_cost,
|
|
get_week_cost,
|
|
get_month_cost,
|
|
init_discord_rpc,
|
|
update_discord_rpc,
|
|
stop_discord_rpc,
|
|
close_application,
|
|
list_memory_files,
|
|
get_claude_version,
|
|
list_plugins,
|
|
install_plugin,
|
|
uninstall_plugin,
|
|
enable_plugin,
|
|
disable_plugin,
|
|
update_plugin,
|
|
list_marketplaces,
|
|
add_marketplace,
|
|
remove_marketplace,
|
|
list_mcp_servers,
|
|
get_mcp_server,
|
|
remove_mcp_server,
|
|
add_mcp_server,
|
|
get_mcp_server_details,
|
|
])
|
|
.run(tauri::generate_context!())
|
|
.expect("error while running tauri application");
|
|
}
|