use crate::{cmds, config::Config, feat, utils::resolve}; use anyhow::Result; use tauri::{ api, AppHandle, CustomMenuItem, Manager, SystemTrayEvent, SystemTrayMenu, SystemTrayMenuItem, SystemTraySubmenu, }; pub struct Tray {} impl Tray { pub fn tray_menu(app_handle: &AppHandle) -> SystemTrayMenu { let zh = { Config::verge().latest().language == Some("zh".into()) }; let version = app_handle.package_info().version.to_string(); macro_rules! t { ($en: expr, $zh: expr) => { if zh { $zh } else { $en } }; } SystemTrayMenu::new() .add_item(CustomMenuItem::new( "open_window", t!("Dashboard", "打开面板"), )) .add_native_item(SystemTrayMenuItem::Separator) .add_item(CustomMenuItem::new( "rule_mode", t!("Rule Mode", "规则模式"), )) .add_item(CustomMenuItem::new( "global_mode", t!("Global Mode", "全局模式"), )) .add_item(CustomMenuItem::new( "direct_mode", t!("Direct Mode", "直连模式"), )) .add_native_item(SystemTrayMenuItem::Separator) .add_item(CustomMenuItem::new( "system_proxy", t!("System Proxy", "系统代理"), )) .add_item(CustomMenuItem::new("tun_mode", t!("TUN Mode", "Tun 模式"))) .add_item(CustomMenuItem::new( "copy_env", t!("Copy Env", "复制环境变量"), )) .add_submenu(SystemTraySubmenu::new( t!("Open Dir", "打开目录"), SystemTrayMenu::new() .add_item(CustomMenuItem::new( "open_app_dir", t!("App Dir", "应用目录"), )) .add_item(CustomMenuItem::new( "open_core_dir", t!("Core Dir", "内核目录"), )) .add_item(CustomMenuItem::new( "open_logs_dir", t!("Logs Dir", "日志目录"), )), )) .add_submenu(SystemTraySubmenu::new( t!("More", "更多"), SystemTrayMenu::new() .add_item(CustomMenuItem::new( "restart_clash", t!("Restart Clash", "重启 Clash"), )) .add_item(CustomMenuItem::new( "restart_app", t!("Restart App", "重启应用"), )) .add_item( CustomMenuItem::new("app_version", format!("Version {version}")).disabled(), ), )) .add_native_item(SystemTrayMenuItem::Separator) .add_item(CustomMenuItem::new("quit", t!("Quit", "退出")).accelerator("CmdOrControl+Q")) } pub fn update_systray(app_handle: &AppHandle) -> Result<()> { app_handle .tray_handle() .set_menu(Tray::tray_menu(app_handle))?; Tray::update_part(app_handle)?; Ok(()) } pub fn update_part(app_handle: &AppHandle) -> Result<()> { let zh = { Config::verge().latest().language == Some("zh".into()) }; let version = app_handle.package_info().version.to_string(); macro_rules! t { ($en: expr, $zh: expr) => { if zh { $zh } else { $en } }; } let mode = { Config::clash() .latest() .0 .get("mode") .map(|val| val.as_str().unwrap_or("rule")) .unwrap_or("rule") .to_owned() }; let tray = app_handle.tray_handle(); let _ = tray.get_item("rule_mode").set_selected(mode == "rule"); let _ = tray.get_item("global_mode").set_selected(mode == "global"); let _ = tray.get_item("direct_mode").set_selected(mode == "direct"); let verge = Config::verge(); let verge = verge.latest(); let system_proxy = verge.enable_system_proxy.as_ref().unwrap_or(&false); let tun_mode = verge.enable_tun_mode.as_ref().unwrap_or(&false); let mut indication_icon = if *system_proxy { #[cfg(not(target_os = "macos"))] let icon = include_bytes!("../../icons/tray-icon-sys.png").to_vec(); #[cfg(target_os = "macos")] let icon = include_bytes!("../../icons/mac-tray-icon-sys.png").to_vec(); icon } else { #[cfg(not(target_os = "macos"))] let icon = include_bytes!("../../icons/tray-icon.png").to_vec(); #[cfg(target_os = "macos")] let icon = include_bytes!("../../icons/icon.png").to_vec(); icon }; if *tun_mode { #[cfg(not(target_os = "macos"))] let icon = include_bytes!("../../icons/tray-icon-tun.png").to_vec(); #[cfg(target_os = "macos")] let icon = include_bytes!("../../icons/mac-tray-icon-tun.png").to_vec(); indication_icon = icon } let _ = tray.set_icon(tauri::Icon::Raw(indication_icon)); let _ = tray.get_item("system_proxy").set_selected(*system_proxy); let _ = tray.get_item("tun_mode").set_selected(*tun_mode); let switch_map = { let mut map = std::collections::HashMap::new(); map.insert(true, "on"); map.insert(false, "off"); map }; let _ = tray.set_tooltip(&format!( "Clash Verge {version}\n{}: {}\n{}: {}", t!("System Proxy", "系统代理"), switch_map[system_proxy], t!("TUN Mode", "Tun 模式"), switch_map[tun_mode] )); Ok(()) } pub fn on_left_click(app_handle: &AppHandle) { let tray_event = { Config::verge().latest().tray_event.clone() }; let tray_event = tray_event.unwrap_or("main_window".into()); match tray_event.as_str() { "system_proxy" => feat::toggle_system_proxy(), "tun_mode" => feat::toggle_tun_mode(), "main_window" => resolve::create_window(app_handle), _ => {} } } pub fn on_system_tray_event(app_handle: &AppHandle, event: SystemTrayEvent) { match event { SystemTrayEvent::LeftClick { .. } => Tray::on_left_click(app_handle), SystemTrayEvent::MenuItemClick { id, .. } => match id.as_str() { mode @ ("rule_mode" | "global_mode" | "direct_mode") => { let mode = &mode[0..mode.len() - 5]; feat::change_clash_mode(mode.into()); } "open_window" => resolve::create_window(app_handle), "system_proxy" => feat::toggle_system_proxy(), "tun_mode" => feat::toggle_tun_mode(), "copy_env" => feat::copy_clash_env(app_handle), "open_app_dir" => crate::log_err!(cmds::open_app_dir()), "open_core_dir" => crate::log_err!(cmds::open_core_dir()), "open_logs_dir" => crate::log_err!(cmds::open_logs_dir()), "restart_clash" => feat::restart_clash_core(), "restart_app" => api::process::restart(&app_handle.env()), "quit" => { let _ = resolve::save_window_size_position(app_handle, true); resolve::resolve_reset(); api::process::kill_children(); app_handle.exit(0); std::process::exit(0); } _ => {} }, _ => {} } } }