use super::CmdResult; use crate::{ feat, logging, utils::{dirs, logging::Type}, wrap_err, }; use tauri::Manager; /// 打开应用程序所在目录 #[tauri::command] pub fn open_app_dir() -> CmdResult<()> { let app_dir = wrap_err!(dirs::app_home_dir())?; wrap_err!(open::that(app_dir)) } /// 打开核心所在目录 #[tauri::command] pub fn open_core_dir() -> CmdResult<()> { let core_dir = wrap_err!(tauri::utils::platform::current_exe())?; let core_dir = core_dir.parent().ok_or("failed to get core dir")?; wrap_err!(open::that(core_dir)) } /// 打开日志目录 #[tauri::command] pub fn open_logs_dir() -> CmdResult<()> { let log_dir = wrap_err!(dirs::app_logs_dir())?; wrap_err!(open::that(log_dir)) } /// 打开网页链接 #[tauri::command] pub fn open_web_url(url: String) -> CmdResult<()> { wrap_err!(open::that(url)) } /// 打开/关闭开发者工具 #[tauri::command] pub fn open_devtools(app_handle: tauri::AppHandle) { if let Some(window) = app_handle.get_webview_window("main") { if !window.is_devtools_open() { window.open_devtools(); } else { window.close_devtools(); } } } /// 退出应用 #[tauri::command] pub fn exit_app() { feat::quit(); } /// 重启应用 #[tauri::command] pub async fn restart_app() -> CmdResult<()> { feat::restart_app(); Ok(()) } /// 获取便携版标识 #[tauri::command] pub fn get_portable_flag() -> CmdResult { Ok(*dirs::PORTABLE_FLAG.get().unwrap_or(&false)) } /// 获取应用目录 #[tauri::command] pub fn get_app_dir() -> CmdResult { let app_home_dir = wrap_err!(dirs::app_home_dir())? .to_string_lossy() .to_string(); Ok(app_home_dir) } /// 获取当前自启动状态 #[tauri::command] pub fn get_auto_launch_status() -> CmdResult { use crate::core::sysopt::Sysopt; wrap_err!(Sysopt::global().get_launch_status()) } /// 下载图标缓存 #[tauri::command] pub async fn download_icon_cache(url: String, name: String) -> CmdResult { let icon_cache_dir = wrap_err!(dirs::app_home_dir())?.join("icons").join("cache"); let icon_path = icon_cache_dir.join(&name); if icon_path.exists() { return Ok(icon_path.to_string_lossy().to_string()); } if !icon_cache_dir.exists() { let _ = std::fs::create_dir_all(&icon_cache_dir); } let temp_path = icon_cache_dir.join(format!("{}.downloading", &name)); let response = wrap_err!(reqwest::get(&url).await)?; let content_type = response .headers() .get(reqwest::header::CONTENT_TYPE) .and_then(|v| v.to_str().ok()) .unwrap_or(""); let is_image = content_type.starts_with("image/"); let content = wrap_err!(response.bytes().await)?; let is_html = content.len() > 15 && (content.starts_with(b" file, Err(_) => { if icon_path.exists() { return Ok(icon_path.to_string_lossy().to_string()); } else { return Err("Failed to create temporary file".into()); } } }; wrap_err!(std::io::copy(&mut content.as_ref(), &mut file))?; } if !icon_path.exists() { match std::fs::rename(&temp_path, &icon_path) { Ok(_) => {} Err(_) => { let _ = std::fs::remove_file(&temp_path); if icon_path.exists() { return Ok(icon_path.to_string_lossy().to_string()); } } } } else { let _ = std::fs::remove_file(&temp_path); } Ok(icon_path.to_string_lossy().to_string()) } else { let _ = std::fs::remove_file(&temp_path); Err(format!("下载的内容不是有效图片: {}", url)) } } #[derive(Debug, serde::Serialize, serde::Deserialize)] pub struct IconInfo { name: String, previous_t: String, current_t: String, } /// 复制图标文件 #[tauri::command] pub fn copy_icon_file(path: String, icon_info: IconInfo) -> CmdResult { use std::{fs, path::Path}; let file_path = Path::new(&path); let icon_dir = wrap_err!(dirs::app_home_dir())?.join("icons"); if !icon_dir.exists() { let _ = fs::create_dir_all(&icon_dir); } let ext = match file_path.extension() { Some(e) => e.to_string_lossy().to_string(), None => "ico".to_string(), }; let dest_path = icon_dir.join(format!( "{0}-{1}.{ext}", icon_info.name, icon_info.current_t )); if file_path.exists() { if icon_info.previous_t.trim() != "" { fs::remove_file( icon_dir.join(format!("{0}-{1}.png", icon_info.name, icon_info.previous_t)), ) .unwrap_or_default(); fs::remove_file( icon_dir.join(format!("{0}-{1}.ico", icon_info.name, icon_info.previous_t)), ) .unwrap_or_default(); } logging!( info, Type::Cmd, true, "Copying icon file path: {:?} -> file dist: {:?}", path, dest_path ); match fs::copy(file_path, &dest_path) { Ok(_) => Ok(dest_path.to_string_lossy().to_string()), Err(err) => Err(err.to_string()), } } else { Err("file not found".to_string()) } } /// 通知UI已准备就绪 #[tauri::command] pub fn notify_ui_ready() -> CmdResult<()> { log::info!(target: "app", "前端UI已准备就绪"); crate::utils::resolve::mark_ui_ready(); Ok(()) } /// UI加载阶段 #[tauri::command] pub fn update_ui_stage(stage: String) -> CmdResult<()> { log::info!(target: "app", "UI加载阶段更新: {}", stage); use crate::utils::resolve::UiReadyStage; let stage_enum = match stage.as_str() { "NotStarted" => UiReadyStage::NotStarted, "Loading" => UiReadyStage::Loading, "DomReady" => UiReadyStage::DomReady, "ResourcesLoaded" => UiReadyStage::ResourcesLoaded, "Ready" => UiReadyStage::Ready, _ => { log::warn!(target: "app", "未知的UI加载阶段: {}", stage); return Err(format!("未知的UI加载阶段: {}", stage)); } }; crate::utils::resolve::update_ui_ready_stage(stage_enum); Ok(()) } /// 重置UI就绪状态 #[tauri::command] pub fn reset_ui_ready_state() -> CmdResult<()> { log::info!(target: "app", "重置UI就绪状态"); crate::utils::resolve::reset_ui_ready(); Ok(()) }