about summary refs log tree commit diff
diff options
context:
space:
mode:
authorgennyble <gen@nyble.dev>2021-03-15 23:46:59 -0500
committergennyble <gen@nyble.dev>2021-03-15 23:46:59 -0500
commit1b3e0e2e216b07124a84c00434fb436b9764eced (patch)
tree9500905bd23f9e6c768230b5115725fd47f85353
parent81429b7a36e7d02a4c5310e1f63c9042105b0824 (diff)
downloadcolorsquash-1b3e0e2e216b07124a84c00434fb436b9764eced.tar.gz
colorsquash-1b3e0e2e216b07124a84c00434fb436b9764eced.zip
Improved algorithm by adjusting tolerance
-rw-r--r--src/main.rs56
1 files changed, 46 insertions, 10 deletions
diff --git a/src/main.rs b/src/main.rs
index aa0fee3..4b324c0 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -1,13 +1,15 @@
-use std::collections::HashMap;
+use std::{collections::HashMap, env::args, time::Instant};
 
 use image::{io::Reader as ImageReader};
 use image::Rgb;
 
 fn main() {
+	let before = Instant::now();
     let filename = std::env::args().skip(1).next().unwrap();
+	let outname = std::env::args().skip(2).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 tolerance = 0.3;
+	let rgb_tolerance = (256.0 * tolerance) as u16;
 	let max_colors = 256;
 
 	println!("File is {}", &filename);
@@ -16,6 +18,7 @@ fn main() {
 	let mut image = imageread.decode().expect("Failed to decode image!").into_rgb8();
 
 	println!("Decoded!");
+	let before_algo = Instant::now();
 
 	let mut colors: HashMap<Rgb<u8>, usize> = HashMap::new();
 
@@ -30,13 +33,16 @@ fn main() {
 		}
 	}
 
-	println!("{} has {} colors in it", filename, colors.len());
-	println!("Sorting...");
+	println!("{} has {} colors in it. Sorting most occuring to least...", filename, colors.len());
 
 	let mut sorted: Vec<(Rgb<u8>, usize)> = colors.into_iter().collect();
 	sorted.sort_by(|a, b| a.1.cmp(&b.1).reverse());
 
-	println!("Sorted!");
+	println!("Sorted! Selecting colors...");
+
+	for (color, count) in sorted.iter().take(10) {
+		println!("{:?} count {}", color, count);
+	}
 
 	let mut sorted_iter = sorted.iter();
 
@@ -56,6 +62,12 @@ fn main() {
 		}
 	}
 
+	for color in selected_colors.iter().take(10) {
+		println!("selected {:?}", color);
+	}
+
+	println!("Selected {} colors! Creating map...", selected_colors.len());
+
 	let mut color_map: HashMap<Rgb<u8>, Rgb<u8>> = HashMap::with_capacity(sorted.len());
 	// Selected colors are themselves
 	for color in selected_colors.iter() {
@@ -63,12 +75,15 @@ fn main() {
 	}
 
 	// Max complexity is O(n * max_colors)
-	for (key, _value) in sorted.iter() {
+	'sorted_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 == 0 {
+				continue 'sorted_colors;
+			}
 			if difference < min_difference {
 				min_difference = difference;
 				min_difference_color = unsafe {*selected_colors.get_unchecked(index) };
@@ -78,13 +93,34 @@ fn main() {
 		color_map.insert(*key, min_difference_color);
 	}
 
+	println!("Mapped! Filling in image...");
+
 	for pixel in image.pixels_mut() {
 		pixel.clone_from(color_map.get(pixel).unwrap());
 	}
 
-	image.save("out.png").expect("Failed to write out");
+	println!("Filled! Took {}ms. Recounting colors...", Instant::now().duration_since(before_algo).as_millis());
+
+	let mut recounted_colors = Vec::with_capacity(max_colors);
+	// Recount colors
+	for pixel in image.pixels() {
+		if !recounted_colors.contains(pixel) {
+			recounted_colors.push(*pixel);
+		}
+	}
+	
+	println!("Aiming for a max of {} colors, got {}", max_colors, recounted_colors.len());
+
+	image.save(outname).expect("Failed to write out");
+	println!("Took {}ms", Instant::now().duration_since(before).as_millis());
 }
 
 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
+	//((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
+	//(a.0[0] as i16 - b.0[0] as i16).abs().max((a.0[1] as i16 - b.0[1] as i16).abs().max(a.0[2] as i16 - b.0[2] as i16).abs()) as u16
+	(a.0[0] as i16 - b.0[0] as i16).abs().max((a.0[1] as i16 - b.0[1] as i16).abs()).max((a.0[2] as i16 - b.0[2] as i16).abs()) as u16
+}
+
+//rd.abs().max(gd.abs().max(bd).abs()) as u16
+//Diff0: Rgb([92, 77, 40]) Rgb([92, 77, 50])
+//0.max(0.max(10))
\ No newline at end of file