feat(desktop): improve app icon handling for cross-platform support (#230)

- Add platform-specific icon path selection for Windows
- Enhance app command item rendering with dynamic icon resolution
- Modify icon loading to use applications-rs for Windows icon extraction
- Update Rust icon loading utility to provide more robust error handling
This commit is contained in:
Huakun 2025-03-03 05:22:08 -05:00 committed by GitHub
parent a42d4d97eb
commit 2cbe45f6d1
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
6 changed files with 24 additions and 58 deletions

View File

@ -65,7 +65,6 @@ cocoa = "0.24.1"
mac-security-rs = { workspace = true } mac-security-rs = { workspace = true }
objc = "0.2.7" objc = "0.2.7"
[target."cfg(not(any(target_os = \"android\", target_os = \"ios\")))".dependencies] [target."cfg(not(any(target_os = \"android\", target_os = \"ios\")))".dependencies]
tauri-plugin-autostart = "2" tauri-plugin-autostart = "2"
tauri-plugin-cli = "2" tauri-plugin-cli = "2"

View File

@ -15,7 +15,8 @@
</script> </script>
<DraggableCommandGroup heading="Apps"> <DraggableCommandGroup heading="Apps">
{#each apps.filter((app) => app.name) as app} {#each apps.filter((app) => app.name) as app, idx}
{@const iconPath = platform === "windows" ? (app.icon_path ?? app.app_path_exe) : app.icon_path}
<Command.Item <Command.Item
class="flex justify-between" class="flex justify-between"
onSelect={async () => { onSelect={async () => {
@ -39,14 +40,14 @@
await getCurrentWindow().hide() await getCurrentWindow().hide()
appState.clearSearchTerm() appState.clearSearchTerm()
}} }}
value={app.app_desktop_path} value={`app:${idx}:${app.app_desktop_path}`}
> >
<span class="flex gap-2"> <span class="flex gap-2">
<IconMultiplexer <IconMultiplexer
icon={app.icon_path icon={iconPath
? { ? {
type: IconEnum.RemoteUrl, type: IconEnum.RemoteUrl,
value: convertFileSrc(app.icon_path, "appicon") value: convertFileSrc(iconPath, "appicon")
} }
: { : {
type: IconEnum.Iconify, type: IconEnum.Iconify,
@ -55,6 +56,7 @@
class="!h-5 !w-5 shrink-0" class="!h-5 !w-5 shrink-0"
/> />
<span>{app.name}</span> <span>{app.name}</span>
<!-- <span>{app.app_path_exe}</span> -->
</span> </span>
</Command.Item> </Command.Item>
{/each} {/each}

View File

@ -22,7 +22,7 @@
"typescript": "^5.0.0", "typescript": "^5.0.0",
"verify-package-export": "^0.0.3" "verify-package-export": "^0.0.3"
}, },
"packageManager": "pnpm@10.4.1", "packageManager": "pnpm@10.5.2",
"engines": { "engines": {
"node": ">=22" "node": ">=22"
}, },

View File

@ -67,6 +67,7 @@ objc = "0.2.7"
tauri-winres = "0.1.1" tauri-winres = "0.1.1"
ico = "0.3.0" ico = "0.3.0"
[build-dependencies] [build-dependencies]
tauri-plugin = { version = "2.0.3", features = ["build"] } tauri-plugin = { version = "2.0.3", features = ["build"] }
tonic-build = "0.11" tonic-build = "0.11"

View File

@ -7,6 +7,7 @@ use std::{
path::PathBuf, path::PathBuf,
}; };
use uuid::Uuid; use uuid::Uuid;
// use windows_icons::get_icon_by_path;
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
use tauri_icns::{IconFamily, IconType}; use tauri_icns::{IconFamily, IconType};
@ -94,58 +95,21 @@ pub fn load_icon(path: PathBuf) -> tauri::http::Response<Vec<u8>> {
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
pub fn load_icon(path: PathBuf) -> tauri::http::Response<Vec<u8>> { pub fn load_icon(path: PathBuf) -> tauri::http::Response<Vec<u8>> {
// tauri::http::Response::builder().body(vec![]).unwrap() match applications::load_icon(&path) {
match path.exists() { Ok(icon) => {
true => { let bytes = icon.to_png().unwrap().get_bytes().to_vec();
let ico_loaded = load_ico(&path); println!("path: {:?} bytes: {:?}", path, bytes.len());
if ico_loaded.is_err() { tauri::http::Response::builder()
let res = tauri::http::Response::builder() .header("Access-Control-Allow-Origin", "*")
.status(tauri::http::StatusCode::INTERNAL_SERVER_ERROR) .header("Content-Type", "image/png")
.body("Error loading icon".as_bytes().to_vec()) .body(bytes)
.unwrap(); .unwrap()
return res;
} else {
let ico = ico_loaded.unwrap();
// write ico to random file name.png, read it and return
// Generate a random file name
let id = Uuid::new_v4();
let file_name = format!("{}.png", id);
// get temp folder
let temp_dir = std::env::temp_dir();
let file_path = temp_dir.join(file_name);
// Write the ico to the random file name.png
let file = File::create(&file_path).unwrap();
ico.write_png(file).unwrap();
// Read the file and return the bytes
let bytes = std::fs::read(&file_path).expect("Error reading file");
// Delete the file
std::fs::remove_file(&file_path).unwrap();
tauri::http::Response::builder()
.header("Content-Type", "image/png")
.body(bytes)
.unwrap()
}
} }
false => { Err(error) => tauri::http::Response::builder()
let res = tauri::http::Response::builder() .header("Access-Control-Allow-Origin", "*")
.status(tauri::http::StatusCode::NOT_FOUND) .status(tauri::http::StatusCode::INTERNAL_SERVER_ERROR)
.body("file not found".as_bytes().to_vec()) .body(error.to_string().as_bytes().to_vec())
.unwrap(); .unwrap(),
return res;
}
}
}
/// Load .ico image
#[cfg(target_os = "windows")]
pub fn load_ico(path: &Path) -> anyhow::Result<ico::IconImage> {
let file = std::fs::File::open(path)?;
let icon_dir = ico::IconDir::read(file)?;
let image = icon_dir.entries().first();
if let Some(image) = image {
Ok(image.decode()?)
} else {
Err(anyhow::anyhow!("No image found"))
} }
} }

@ -1 +1 @@
Subproject commit b8e85d01100d47cba226a3d65948a2ac4197fc69 Subproject commit bcfbebb93a57918aca4ba51257b2788be90892da