rlvgl-core 0.1.7

Core runtime, widget tree, renderer, style, and plugin interfaces for rlvgl.
Documentation
//! JPEG decoder for Color pixel arrays.
use crate::widget::Color;
use alloc::vec::Vec;
pub use jpeg_decoder::Error;
use jpeg_decoder::{Decoder, PixelFormat};
use std::io::Cursor;

/// Decode a JPEG image into a vector of RGB [`Color`]s.
pub fn decode(data: &[u8]) -> Result<(Vec<Color>, u16, u16), Error> {
    let mut decoder = Decoder::new(Cursor::new(data));
    let pixels_raw = decoder.decode()?;
    let info = decoder
        .info()
        .ok_or_else(|| Error::Format("missing image info".into()))?;
    let mut pixels = Vec::with_capacity(info.width as usize * info.height as usize);
    match info.pixel_format {
        PixelFormat::L8 => {
            for &v in &pixels_raw {
                pixels.push(Color(v, v, v, 255));
            }
        }
        PixelFormat::L16 => {
            for chunk in pixels_raw.chunks_exact(2) {
                let val = u16::from_be_bytes([chunk[0], chunk[1]]);
                let v = (val / 257) as u8;
                pixels.push(Color(v, v, v, 255));
            }
        }
        PixelFormat::RGB24 => {
            for chunk in pixels_raw.chunks_exact(3) {
                pixels.push(Color(chunk[0], chunk[1], chunk[2], 255));
            }
        }
        PixelFormat::CMYK32 => {
            for chunk in pixels_raw.chunks_exact(4) {
                let c = chunk[0] as f32 / 255.0;
                let m = chunk[1] as f32 / 255.0;
                let y = chunk[2] as f32 / 255.0;
                let k = chunk[3] as f32 / 255.0;
                let r = (1.0 - (c * (1.0 - k) + k)) * 255.0;
                let g = (1.0 - (m * (1.0 - k) + k)) * 255.0;
                let b = (1.0 - (y * (1.0 - k) + k)) * 255.0;
                pixels.push(Color(r as u8, g as u8, b as u8, 255));
            }
        }
    }
    Ok((pixels, info.width, info.height))
}

#[cfg(test)]
mod tests {
    use super::*;
    use base64::Engine;

    const RED_DOT_JPEG: &str = "/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAgGBgcGBQgHBwcJCQgKDBQNDAsLDBkSEw8UHRofHh0aHBwgJC4nICIsIxwcKDcpLDAxNDQ0Hyc5PTgyPC4zNDL/2wBDAQkJCQwLDBgNDRgyIRwhMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjL/wAARCAABAAEDASIAAhEBAxEB/8QAHwAAAQUBAQEBAQEAAAAAAAAAAAECAwQFBgcICQoL/8QAtRAAAgEDAwIEAwUFBAQAAAF9AQIDAAQRBRIhMUEGE1FhByJxFDKBkaEII0KxwRVS0fAkM2JyggkKFhcYGRolJicoKSo0NTY3ODk6Q0RFRkdISUpTVFVWV1hZWmNkZWZnaGlqc3R1dnd4eXqDhIWGh4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5usLDxMXGx8jJytLT1NXW19jZ2uHi4+Tl5ufo6erx8vP09fb3+Pn6/8QAHwEAAwEBAQEBAQEBAQAAAAAAAAECAwQFBgcICQoL/8QAtREAAgECBAQDBAcFBAQAAQJ3AAECAxEEBSExBhJBUQdhcRMiMoEIFEKRobHBCSMzUvAVYnLRChYkNOEl8RcYGRomJygpKjU2Nzg5OkNERUZHSElKU1RVVldYWVpjZGVmZ2hpanN0dXZ3eHl6goOEhYaHiImKkpOUlZaXmJmaoqOkpaanqKmqsrO0tba3uLm6wsPExcbHyMnK0tPU1dbX2Nna4uPk5ebn6Onq8vP09fb3+Pn6/9oADAMBAAIRAxEAPwDi6KKK+ZP3E//Z";

    #[test]
    fn decode_red_dot() {
        let data = base64::engine::general_purpose::STANDARD
            .decode(RED_DOT_JPEG)
            .unwrap();
        let (pixels, w, h) = decode(&data).unwrap();
        assert_eq!((w, h), (1, 1));
        assert!(pixels[0].0 >= 250 && pixels[0].1 == 0 && pixels[0].2 == 0);
    }
}