diff options
author | gennyble <gen@nyble.dev> | 2023-10-09 02:28:37 -0500 |
---|---|---|
committer | gennyble <gen@nyble.dev> | 2023-10-09 02:28:37 -0500 |
commit | d4589d9600b5bddd3482c9f35414d4f843255d13 (patch) | |
tree | 800a007a5d62e53d15016ef1cb84b319a2bba086 /squash/src/main.rs | |
parent | 560de837d38b9dd9d3c07ed4ea7fec61ed24ddeb (diff) | |
download | colorsquash-d4589d9600b5bddd3482c9f35414d4f843255d13.tar.gz colorsquash-d4589d9600b5bddd3482c9f35414d4f843255d13.zip |
accept jpeg input with zune_jpeg
Diffstat (limited to 'squash/src/main.rs')
-rw-r--r-- | squash/src/main.rs | 52 |
1 files changed, 48 insertions, 4 deletions
diff --git a/squash/src/main.rs b/squash/src/main.rs index b6eb699..da3cfbf 100644 --- a/squash/src/main.rs +++ b/squash/src/main.rs @@ -1,9 +1,10 @@ use std::{fs::File, io::BufWriter}; -use anyhow::bail; +use anyhow::{anyhow, bail}; use camino::{Utf8Path, Utf8PathBuf}; use colorsquash::Squasher; use png::{ColorType, Decoder, Encoder}; +use zune_jpeg::{zune_core::colorspace::ColorSpace, JpegDecoder}; fn main() -> Result<(), anyhow::Error> { // I should use clap or at least getopt, but this is fine. It's 20LOC. @@ -13,8 +14,13 @@ fn main() -> Result<(), anyhow::Error> { }; let mut argv = std::env::args().skip(1); - let color_count: u8 = if let Some(Ok(count)) = argv.next().map(|r| r.parse()) { - count + let color_count: u8 = if let Some(Ok(count)) = argv.next().map(|r| r.parse::<usize>()) { + if count > 256 { + eprintln!("max colour count must be 256 or below"); + std::process::exit(1); + } else { + (count - 1) as u8 + } } else { usage() }; @@ -31,12 +37,29 @@ fn main() -> Result<(), anyhow::Error> { usage(); }; - let mut image = get_png(input_path)?; + let mut image = match input_path.extension() { + None => { + eprintln!("can't determine input filetype!\nSupported input types: PNG, JPG"); + std::process::exit(1); + } + Some("png") => get_png(input_path)?, + Some("jpg") | Some("jpeg") => get_jpg(input_path)?, + Some(ext) => { + eprintln!("unknown filetype '{ext}'!\nSupported input types: PNG, JPG"); + std::process::exit(1); + } + }; let squasher = Squasher::new(color_count, &image.data); let size = squasher.map_over(&mut image.data); image.data.resize(size, 0); + println!( + "selected {} colours of max {}", + squasher.palette().len(), + color_count + ); + // PNG Output let file = File::create(output_path)?; let bufw = BufWriter::new(file); @@ -80,6 +103,27 @@ fn get_png<P: AsRef<Utf8Path>>(path: P) -> Result<Image, anyhow::Error> { } } +fn get_jpg<P: AsRef<Utf8Path>>(path: P) -> Result<Image, anyhow::Error> { + let content = std::fs::read(path.as_ref())?; + let mut dec = JpegDecoder::new(&content); + let pixels = dec.decode()?; + let info = dec + .info() + .ok_or(anyhow!("image had no info; this should be impossible"))?; + + let colorspace = dec.get_output_colorspace(); + match colorspace { + Some(ColorSpace::RGB) => (), + _ => bail!("colorspace {colorspace:?} not supported"), + } + + Ok(Image { + width: info.width as usize, + height: info.height as usize, + data: pixels, + }) +} + struct Image { width: usize, height: usize, |