2023-01-29 17:46:02 +00:00
|
|
|
use crate::desktop_file::get_desktop_icon_name;
|
2023-02-01 20:42:05 +00:00
|
|
|
use cfg_if::cfg_if;
|
|
|
|
use color_eyre::{Help, Report, Result};
|
2023-01-29 17:46:02 +00:00
|
|
|
use gtk::gdk_pixbuf::Pixbuf;
|
|
|
|
use gtk::prelude::*;
|
|
|
|
use gtk::{IconLookupFlags, IconTheme};
|
|
|
|
use std::path::{Path, PathBuf};
|
2023-02-01 20:42:05 +00:00
|
|
|
|
|
|
|
cfg_if!(
|
|
|
|
if #[cfg(feature = "http")] {
|
|
|
|
use crate::send;
|
|
|
|
use gtk::gio::{Cancellable, MemoryInputStream};
|
|
|
|
use tokio::spawn;
|
|
|
|
use tracing::error;
|
|
|
|
}
|
|
|
|
);
|
2023-01-29 17:46:02 +00:00
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
enum ImageLocation<'a> {
|
2023-02-01 20:42:05 +00:00
|
|
|
Icon {
|
|
|
|
name: String,
|
|
|
|
theme: &'a IconTheme,
|
|
|
|
},
|
2023-01-29 17:46:02 +00:00
|
|
|
Local(PathBuf),
|
|
|
|
Steam(String),
|
2023-02-01 20:42:05 +00:00
|
|
|
#[cfg(feature = "http")]
|
|
|
|
Remote(reqwest::Url),
|
2023-01-29 17:46:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct ImageProvider<'a> {
|
|
|
|
location: ImageLocation<'a>,
|
|
|
|
size: i32,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> ImageProvider<'a> {
|
|
|
|
/// Attempts to parse the image input to find its location.
|
|
|
|
/// Errors if no valid location type can be found.
|
|
|
|
///
|
|
|
|
/// Note this checks that icons exist in theme, or files exist on disk
|
|
|
|
/// but no other check is performed.
|
2023-01-29 22:48:42 +00:00
|
|
|
pub fn parse(input: &str, theme: &'a IconTheme, size: i32) -> Result<Self> {
|
2023-01-29 17:46:02 +00:00
|
|
|
let location = Self::get_location(input, theme, size)?;
|
|
|
|
Ok(Self { location, size })
|
|
|
|
}
|
|
|
|
|
2023-01-29 18:38:57 +00:00
|
|
|
/// Returns true if the input starts with a prefix
|
|
|
|
/// that is supported by the parser
|
|
|
|
/// (ie the parser would not fallback to checking the input).
|
2023-02-01 20:42:05 +00:00
|
|
|
#[cfg(any(feature = "music", feature = "workspaces"))]
|
2023-01-29 18:38:57 +00:00
|
|
|
pub fn is_definitely_image_input(input: &str) -> bool {
|
|
|
|
input.starts_with("icon:")
|
|
|
|
|| input.starts_with("file://")
|
|
|
|
|| input.starts_with("http://")
|
|
|
|
|| input.starts_with("https://")
|
|
|
|
}
|
|
|
|
|
2023-01-29 22:48:42 +00:00
|
|
|
fn get_location(input: &str, theme: &'a IconTheme, size: i32) -> Result<ImageLocation<'a>> {
|
2023-01-29 17:46:02 +00:00
|
|
|
let (input_type, input_name) = input
|
|
|
|
.split_once(':')
|
2023-01-29 22:48:42 +00:00
|
|
|
.map_or((None, input), |(t, n)| (Some(t), n));
|
2023-01-29 17:46:02 +00:00
|
|
|
|
|
|
|
match input_type {
|
|
|
|
Some(input_type) if input_type == "icon" => Ok(ImageLocation::Icon {
|
|
|
|
name: input_name.to_string(),
|
|
|
|
theme,
|
|
|
|
}),
|
|
|
|
Some(input_type) if input_type == "file" => Ok(ImageLocation::Local(PathBuf::from(
|
|
|
|
input_name[2..].to_string(),
|
|
|
|
))),
|
2023-02-01 20:42:05 +00:00
|
|
|
#[cfg(feature = "http")]
|
2023-01-29 17:46:02 +00:00
|
|
|
Some(input_type) if input_type == "http" || input_type == "https" => {
|
|
|
|
Ok(ImageLocation::Remote(input.parse()?))
|
|
|
|
}
|
|
|
|
None if input.starts_with("steam_app_") => Ok(ImageLocation::Steam(
|
|
|
|
input_name.chars().skip("steam_app_".len()).collect(),
|
|
|
|
)),
|
|
|
|
None if theme
|
2023-01-29 22:48:42 +00:00
|
|
|
.lookup_icon(input, size, IconLookupFlags::empty())
|
2023-01-29 17:46:02 +00:00
|
|
|
.is_some() =>
|
|
|
|
{
|
|
|
|
Ok(ImageLocation::Icon {
|
|
|
|
name: input_name.to_string(),
|
|
|
|
theme,
|
|
|
|
})
|
|
|
|
}
|
2023-02-01 20:42:05 +00:00
|
|
|
Some(input_type) => Err(Report::msg(format!("Unsupported image type: {input_type}"))
|
|
|
|
.note("You may need to recompile with support if available")),
|
2023-02-01 21:05:58 +00:00
|
|
|
None if PathBuf::from(input_name).is_file() => {
|
2023-01-29 17:46:02 +00:00
|
|
|
Ok(ImageLocation::Local(PathBuf::from(input_name)))
|
|
|
|
}
|
2023-01-29 22:48:42 +00:00
|
|
|
None => get_desktop_icon_name(input_name).map_or_else(
|
2023-03-04 23:13:35 +00:00
|
|
|
|| Err(Report::msg(format!("Unknown image type: '{input}'"))),
|
2023-01-29 22:48:42 +00:00
|
|
|
|input| Self::get_location(&input, theme, size),
|
|
|
|
),
|
2023-01-29 17:46:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Attempts to fetch the image from the location
|
|
|
|
/// and load it into the provided `GTK::Image` widget.
|
|
|
|
pub fn load_into_image(&self, image: gtk::Image) -> Result<()> {
|
|
|
|
// handle remote locations async to avoid blocking UI thread while downloading
|
2023-02-01 20:42:05 +00:00
|
|
|
#[cfg(feature = "http")]
|
2023-01-29 17:46:02 +00:00
|
|
|
if let ImageLocation::Remote(url) = &self.location {
|
|
|
|
let url = url.clone();
|
|
|
|
let (tx, rx) = glib::MainContext::channel(glib::PRIORITY_DEFAULT);
|
|
|
|
|
|
|
|
spawn(async move {
|
|
|
|
let bytes = Self::get_bytes_from_http(url).await;
|
|
|
|
if let Ok(bytes) = bytes {
|
|
|
|
send!(tx, bytes);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
{
|
|
|
|
let size = self.size;
|
|
|
|
rx.attach(None, move |bytes| {
|
|
|
|
let stream = MemoryInputStream::from_bytes(&bytes);
|
|
|
|
let pixbuf = Pixbuf::from_stream_at_scale(
|
|
|
|
&stream,
|
|
|
|
size,
|
|
|
|
size,
|
|
|
|
true,
|
|
|
|
Some(&Cancellable::new()),
|
|
|
|
);
|
|
|
|
|
|
|
|
match pixbuf {
|
|
|
|
Ok(pixbuf) => image.set_pixbuf(Some(&pixbuf)),
|
|
|
|
Err(err) => error!("{err:?}"),
|
|
|
|
}
|
|
|
|
|
|
|
|
Continue(false)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
2023-02-25 14:24:21 +00:00
|
|
|
self.load_into_image_sync(&image)?;
|
2023-01-29 22:48:42 +00:00
|
|
|
};
|
|
|
|
|
2023-02-01 20:42:05 +00:00
|
|
|
#[cfg(not(feature = "http"))]
|
2023-02-25 14:24:21 +00:00
|
|
|
self.load_into_image_sync(&image)?;
|
2023-02-01 20:42:05 +00:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-02-25 14:24:21 +00:00
|
|
|
fn load_into_image_sync(&self, image: >k::Image) -> Result<()> {
|
2023-02-01 20:42:05 +00:00
|
|
|
let pixbuf = match &self.location {
|
|
|
|
ImageLocation::Icon { name, theme } => self.get_from_icon(name, theme),
|
|
|
|
ImageLocation::Local(path) => self.get_from_file(path),
|
|
|
|
ImageLocation::Steam(steam_id) => self.get_from_steam_id(steam_id),
|
|
|
|
#[cfg(feature = "http")]
|
|
|
|
_ => unreachable!(), // handled above
|
|
|
|
}?;
|
|
|
|
|
|
|
|
image.set_pixbuf(Some(&pixbuf));
|
|
|
|
|
2023-01-29 22:48:42 +00:00
|
|
|
Ok(())
|
2023-01-29 17:46:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Attempts to get a `Pixbuf` from the GTK icon theme.
|
|
|
|
fn get_from_icon(&self, name: &str, theme: &IconTheme) -> Result<Pixbuf> {
|
|
|
|
let pixbuf = match theme.lookup_icon(name, self.size, IconLookupFlags::empty()) {
|
|
|
|
Some(_) => theme.load_icon(name, self.size, IconLookupFlags::FORCE_SIZE),
|
|
|
|
None => Ok(None),
|
|
|
|
}?;
|
|
|
|
|
2023-01-29 22:48:42 +00:00
|
|
|
pixbuf.map_or_else(
|
|
|
|
|| Err(Report::msg("Icon theme does not contain icon '{name}'")),
|
|
|
|
Ok,
|
|
|
|
)
|
2023-01-29 17:46:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Attempts to get a `Pixbuf` from a local file.
|
|
|
|
fn get_from_file(&self, path: &Path) -> Result<Pixbuf> {
|
|
|
|
let pixbuf = Pixbuf::from_file_at_scale(path, self.size, self.size, true)?;
|
|
|
|
Ok(pixbuf)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Attempts to get a `Pixbuf` from a local file,
|
|
|
|
/// using the Steam game ID to look it up.
|
|
|
|
fn get_from_steam_id(&self, steam_id: &str) -> Result<Pixbuf> {
|
|
|
|
// TODO: Can we load this from icon theme with app id `steam_icon_{}`?
|
2023-01-29 22:48:42 +00:00
|
|
|
let path = dirs::data_dir().map_or_else(
|
|
|
|
|| Err(Report::msg("Missing XDG data dir")),
|
|
|
|
|dir| {
|
|
|
|
Ok(dir.join(format!(
|
|
|
|
"icons/hicolor/32x32/apps/steam_icon_{steam_id}.png"
|
|
|
|
)))
|
|
|
|
},
|
|
|
|
)?;
|
2023-01-29 17:46:02 +00:00
|
|
|
|
|
|
|
self.get_from_file(&path)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Attempts to get `Bytes` from an HTTP resource asynchronously.
|
2023-02-01 20:42:05 +00:00
|
|
|
#[cfg(feature = "http")]
|
|
|
|
async fn get_bytes_from_http(url: reqwest::Url) -> Result<glib::Bytes> {
|
2023-01-29 17:46:02 +00:00
|
|
|
let bytes = reqwest::get(url).await?.bytes().await?;
|
2023-02-01 20:42:05 +00:00
|
|
|
Ok(glib::Bytes::from_owned(bytes))
|
2023-01-29 17:46:02 +00:00
|
|
|
}
|
|
|
|
}
|