From 9b86f05b9bee61e788c5dca62cc9101c5a5be2b5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Leonardo=20Gibrowski=20Fa=C3=A9?= Date: Sat, 10 Feb 2024 15:10:34 -0300 Subject: [PATCH] fix single image webp and gif simply detect it ahead of time if the webp image has any frames --- daemon/src/wallpaper.rs | 2 +- src/cli.rs | 2 +- src/imgproc.rs | 68 +++++++++++++++++++++++++++++++---------- utils/src/ipc.rs | 2 +- 4 files changed, 55 insertions(+), 19 deletions(-) diff --git a/daemon/src/wallpaper.rs b/daemon/src/wallpaper.rs index 0548bdf3..80c221a8 100644 --- a/daemon/src/wallpaper.rs +++ b/daemon/src/wallpaper.rs @@ -120,7 +120,7 @@ impl Wallpaper { id: AtomicUsize::new(0), transition_finished: Arc::new(AtomicBool::new(false)), }, - configured: AtomicBool::new(false) + configured: AtomicBool::new(false), } } diff --git a/src/cli.rs b/src/cli.rs index b837738a..72d48088 100644 --- a/src/cli.rs +++ b/src/cli.rs @@ -269,7 +269,7 @@ pub struct Img { /// ///The 'left', 'right', 'top' and 'bottom' options make the transition happen from that ///position to its opposite in the screen. - /// + /// ///'none' is an alias to 'simple' that also sets the 'transition-step' to 255. This has the ///effect of the transition finishing instantly /// diff --git a/src/imgproc.rs b/src/imgproc.rs index aa90a37f..0c22e0c5 100644 --- a/src/imgproc.rs +++ b/src/imgproc.rs @@ -21,44 +21,77 @@ use crate::cli::ResizeStrategy; use super::cli; -pub enum ImgBuf { +enum ImgBufInner { Stdin(BufReader), File(image::io::Reader>), } +impl ImgBufInner { + /// Guess the format of the ImgBufInner + #[inline] + fn format(&self) -> Option { + match &self { + ImgBufInner::Stdin(_) => None, // not seekable + ImgBufInner::File(reader) => reader.format(), + } + } +} + +pub struct ImgBuf { + inner: ImgBufInner, + is_animated: bool, +} + impl ImgBuf { /// Create a new ImgBuf from a given path. Use - for Stdin pub fn new(path: &Path) -> Result { Ok(if let Some("-") = path.to_str() { let reader = BufReader::new(stdin()); - Self::Stdin(reader) + Self { + inner: ImgBufInner::Stdin(reader), + is_animated: false, + } } else { let reader = image::io::Reader::open(path) .map_err(|e| format!("failed to open image: {e}"))? .with_guessed_format() .map_err(|e| format!("failed to detect the image's format: {e}"))?; - Self::File(reader) + let is_animated = { + match reader.format() { + Some(ImageFormat::Gif) => true, + Some(ImageFormat::WebP) => { + // Note: unwraping is safe because we already openned the file once before this + WebPDecoder::new(BufReader::new(File::open(path).unwrap())) + .map_err(|e| format!("failed to decode Webp Image: {e}"))? + .has_animation() + } + + _ => false, + } + }; + + Self { + inner: ImgBufInner::File(reader), + is_animated, + } }) } /// Guess the format of the ImgBuf fn format(&self) -> Option { - match self { - ImgBuf::Stdin(_) => None, // not seekable - ImgBuf::File(reader) => reader.format(), - } + self.inner.format() } - /// Is this ImgBuf an animated image? + #[inline] pub fn is_animated(&self) -> bool { - matches!(self.format(), Some(ImageFormat::Gif | ImageFormat::WebP)) + self.is_animated } /// Decode the ImgBuf into am RgbImage pub fn decode(self) -> Result { - Ok(match self { - ImgBuf::Stdin(mut reader) => { + Ok(match self.inner { + ImgBufInner::Stdin(mut reader) => { let mut buffer = Vec::new(); reader .read_to_end(&mut buffer) @@ -66,7 +99,7 @@ impl ImgBuf { image::load_from_memory(&buffer) } - ImgBuf::File(reader) => reader.decode(), + ImgBufInner::File(reader) => reader.decode(), } .map_err(|e| format!("failed to decode image: {e}"))? .into_rgb8()) @@ -90,9 +123,9 @@ impl ImgBuf { } let img_format = self.format(); - match self { - ImgBuf::Stdin(reader) => create_decoder(img_format, reader), - ImgBuf::File(reader) => create_decoder(img_format, reader.into_inner()), + match self.inner { + ImgBufInner::Stdin(reader) => create_decoder(img_format, reader), + ImgBufInner::File(reader) => create_decoder(img_format, reader.into_inner()), } } } @@ -139,7 +172,10 @@ pub fn compress_frames( canvas = Some(img); } //Add the first frame we got earlier: - compressed_frames.push((BitPack::pack(&canvas.unwrap(), &first_img)?, first_duration)); + compressed_frames.push(( + BitPack::pack(canvas.as_ref().unwrap_or(&first_img), &first_img)?, + first_duration, + )); Ok(compressed_frames) } diff --git a/utils/src/ipc.rs b/utils/src/ipc.rs index 800c02ad..b6dbae8d 100644 --- a/utils/src/ipc.rs +++ b/utils/src/ipc.rs @@ -310,7 +310,7 @@ pub fn read_socket(stream: &UnixStream) -> Result, String> { if e.kind() == std::io::ErrorKind::WouldBlock && tries < 5 { std::thread::sleep(Duration::from_millis(1)); } else { - return Err(format!("failed to read serialized length: {e}")); + return Err(format!("failed to read serialized length: {e}")); } } }