diff options
author | gennyble <gen@nyble.dev> | 2021-03-15 21:04:22 -0500 |
---|---|---|
committer | gennyble <gen@nyble.dev> | 2021-03-15 21:10:18 -0500 |
commit | 81429b7a36e7d02a4c5310e1f63c9042105b0824 (patch) | |
tree | e1e54dbcaef9f88e3063ab3da9b679c04c7feb44 /src | |
download | colorsquash-81429b7a36e7d02a4c5310e1f63c9042105b0824.tar.gz colorsquash-81429b7a36e7d02a4c5310e1f63c9042105b0824.zip |
Initial commit
Algorithm works, roughly
Diffstat (limited to 'src')
-rw-r--r-- | src/main.rs | 90 |
1 files changed, 90 insertions, 0 deletions
diff --git a/src/main.rs b/src/main.rs new file mode 100644 index 0000000..aa0fee3 --- /dev/null +++ b/src/main.rs @@ -0,0 +1,90 @@ +use std::collections::HashMap; + +use image::{io::Reader as ImageReader}; +use image::Rgb; + +fn main() { + let filename = std::env::args().skip(1).next().unwrap(); + // The percent of RGB value difference a color has to surpass to be considere unique + let tolerance = 0.2; + let rgb_tolerance = (768.0 * tolerance) as u16; + let max_colors = 256; + + println!("File is {}", &filename); + + let imageread = ImageReader::open(&filename).expect("Failed to open image!"); + let mut image = imageread.decode().expect("Failed to decode image!").into_rgb8(); + + println!("Decoded!"); + + let mut colors: HashMap<Rgb<u8>, usize> = HashMap::new(); + + for pixel in image.pixels() { + match colors.get_mut(pixel) { + None => { + colors.insert(*pixel, 1); + }, + Some(n) => { + *n += 1 + } + } + } + + println!("{} has {} colors in it", filename, colors.len()); + println!("Sorting..."); + + let mut sorted: Vec<(Rgb<u8>, usize)> = colors.into_iter().collect(); + sorted.sort_by(|a, b| a.1.cmp(&b.1).reverse()); + + println!("Sorted!"); + + let mut sorted_iter = sorted.iter(); + + let mut selected_colors: Vec<Rgb<u8>> = Vec::with_capacity(max_colors); + selected_colors.push(sorted_iter.next().unwrap().0); + + for (key, _value) in sorted_iter { + if selected_colors.len() < max_colors { + for selected_color in selected_colors.iter() { + if rgb_difference(key, selected_color) > rgb_tolerance { + selected_colors.push(*key); + break; + } + } + } else { + break; + } + } + + let mut color_map: HashMap<Rgb<u8>, Rgb<u8>> = HashMap::with_capacity(sorted.len()); + // Selected colors are themselves + for color in selected_colors.iter() { + color_map.insert(*color, *color); + } + + // Max complexity is O(n * max_colors) + for (key, _value) in sorted.iter() { + let mut min_difference = 769; // One greater than the max difference + let mut min_difference_color = *key; + + for index in 0..selected_colors.len() { + let difference = rgb_difference(key, unsafe { selected_colors.get_unchecked(index) }); + if difference < min_difference { + min_difference = difference; + min_difference_color = unsafe {*selected_colors.get_unchecked(index) }; + } + } + + color_map.insert(*key, min_difference_color); + } + + for pixel in image.pixels_mut() { + pixel.clone_from(color_map.get(pixel).unwrap()); + } + + image.save("out.png").expect("Failed to write out"); +} + +fn rgb_difference(a: &Rgb<u8>, b: &Rgb<u8>) -> u16 { + ((a.0[0] as i16 - b.0[0] as i16).abs() + (a.0[1] as i16 - b.0[1] as i16).abs() +(a.0[2] as i16 - b.0[2] as i16).abs()) as u16 +} \ No newline at end of file |