main.rs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. #![cfg_attr(
  2. all(not(debug_assertions), target_os = "windows"),
  3. windows_subsystem = "windows"
  4. )]
  5. mod cmds;
  6. mod core;
  7. mod states;
  8. mod utils;
  9. use crate::{
  10. core::VergeConfig,
  11. utils::{dirs, resolve, server},
  12. };
  13. use tauri::{
  14. api, CustomMenuItem, Manager, SystemTray, SystemTrayEvent, SystemTrayMenu, SystemTrayMenuItem,
  15. };
  16. fn main() -> std::io::Result<()> {
  17. if server::check_singleton().is_err() {
  18. println!("app exists");
  19. return Ok(());
  20. }
  21. #[cfg(target_os = "windows")]
  22. unsafe {
  23. dirs::init_portable_flag();
  24. }
  25. let tray_menu = SystemTrayMenu::new()
  26. .add_item(CustomMenuItem::new("open_window", "Show"))
  27. .add_item(CustomMenuItem::new("system_proxy", "System Proxy"))
  28. .add_item(CustomMenuItem::new("restart_clash", "Restart Clash"))
  29. .add_native_item(SystemTrayMenuItem::Separator)
  30. .add_item(CustomMenuItem::new("quit", "Quit").accelerator("CmdOrControl+Q"));
  31. #[allow(unused_mut)]
  32. let mut builder = tauri::Builder::default()
  33. .manage(states::VergeState::default())
  34. .manage(states::ClashState::default())
  35. .manage(states::ProfilesState::default())
  36. .setup(|app| Ok(resolve::resolve_setup(app)))
  37. .system_tray(SystemTray::new().with_menu(tray_menu))
  38. .on_system_tray_event(move |app_handle, event| match event {
  39. SystemTrayEvent::MenuItemClick { id, .. } => match id.as_str() {
  40. "open_window" => {
  41. let window = app_handle.get_window("main").unwrap();
  42. window.unminimize().unwrap();
  43. window.show().unwrap();
  44. window.set_focus().unwrap();
  45. }
  46. "system_proxy" => {
  47. let verge_state = app_handle.state::<states::VergeState>();
  48. let mut verge = verge_state.0.lock().unwrap();
  49. let old_value = verge.config.enable_system_proxy.clone().unwrap_or(false);
  50. let new_value = !old_value;
  51. match verge.patch_config(VergeConfig {
  52. enable_system_proxy: Some(new_value),
  53. ..VergeConfig::default()
  54. }) {
  55. Ok(_) => {
  56. app_handle
  57. .tray_handle()
  58. .get_item(id.as_str())
  59. .set_selected(new_value)
  60. .unwrap();
  61. // update verge config
  62. let window = app_handle.get_window("main").unwrap();
  63. window.emit("verge://refresh-verge-config", "yes").unwrap();
  64. }
  65. Err(err) => log::error!("{err}"),
  66. }
  67. }
  68. "restart_clash" => {
  69. let clash_state = app_handle.state::<states::ClashState>();
  70. let profiles_state = app_handle.state::<states::ProfilesState>();
  71. let mut clash = clash_state.0.lock().unwrap();
  72. let mut profiles = profiles_state.0.lock().unwrap();
  73. crate::log_if_err!(clash.restart_sidecar(&mut profiles));
  74. }
  75. "quit" => {
  76. resolve::resolve_reset(app_handle);
  77. api::process::kill_children();
  78. std::process::exit(0);
  79. }
  80. _ => {}
  81. },
  82. #[cfg(target_os = "windows")]
  83. SystemTrayEvent::LeftClick { .. } => {
  84. let window = app_handle.get_window("main").unwrap();
  85. window.unminimize().unwrap();
  86. window.show().unwrap();
  87. window.set_focus().unwrap();
  88. }
  89. _ => {}
  90. })
  91. .invoke_handler(tauri::generate_handler![
  92. // common
  93. cmds::restart_sidecar,
  94. cmds::get_sys_proxy,
  95. cmds::get_cur_proxy,
  96. cmds::kill_sidecars,
  97. cmds::open_app_dir,
  98. cmds::open_logs_dir,
  99. // clash
  100. cmds::get_clash_info,
  101. cmds::patch_clash_config,
  102. // verge
  103. cmds::get_verge_config,
  104. cmds::patch_verge_config,
  105. // profile
  106. cmds::view_profile,
  107. cmds::patch_profile,
  108. cmds::create_profile,
  109. cmds::import_profile,
  110. cmds::update_profile,
  111. cmds::delete_profile,
  112. cmds::select_profile,
  113. cmds::get_profiles,
  114. cmds::sync_profiles,
  115. cmds::enhance_profiles,
  116. cmds::change_profile_chain,
  117. cmds::change_profile_valid,
  118. cmds::read_profile_file,
  119. cmds::save_profile_file
  120. ]);
  121. #[cfg(target_os = "macos")]
  122. {
  123. use tauri::{Menu, MenuItem, Submenu};
  124. let submenu_file = Submenu::new(
  125. "File",
  126. Menu::new()
  127. .add_native_item(MenuItem::Undo)
  128. .add_native_item(MenuItem::Redo)
  129. .add_native_item(MenuItem::Copy)
  130. .add_native_item(MenuItem::Paste)
  131. .add_native_item(MenuItem::Cut)
  132. .add_native_item(MenuItem::SelectAll),
  133. );
  134. builder = builder.menu(Menu::new().add_submenu(submenu_file));
  135. }
  136. builder
  137. .build(tauri::generate_context!())
  138. .expect("error while running tauri application")
  139. .run(|app_handle, e| match e {
  140. tauri::RunEvent::WindowEvent { label, event, .. } => match event {
  141. tauri::WindowEvent::CloseRequested { api, .. } => {
  142. let app_handle = app_handle.clone();
  143. api.prevent_close();
  144. app_handle.get_window(&label).unwrap().hide().unwrap();
  145. }
  146. _ => {}
  147. },
  148. tauri::RunEvent::ExitRequested { .. } => {
  149. resolve::resolve_reset(app_handle);
  150. api::process::kill_children();
  151. }
  152. _ => {}
  153. });
  154. Ok(())
  155. }