Skip to content

Commit 0a73e5b

Browse files
committedOct 26, 2022
Allow loading palette files
1 parent 9f595c5 commit 0a73e5b

File tree

4 files changed

+55
-5
lines changed

4 files changed

+55
-5
lines changed
 

‎README.md

+9
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,14 @@ cargo run --release -- find -i .\gfx\tokyo.png -p "#050505,#ffffff,#ff0000" -m d
4444

4545
![Tokyo with looked up colors](gfx/tokyo-find-dither-dark-white-red.png)
4646

47+
### Find colors by loading a palette to dither the image
48+
49+
```sh
50+
cargo run --release -- find -i .\gfx\tokyo.png -p .\gfx\apollo-1x.png -m dither
51+
```
52+
53+
![Tokyo with looked up colors](gfx/tokyo-find-dither-apollo.png)
54+
4755
### Output the palette:
4856

4957
```sh
@@ -69,6 +77,7 @@ I had to read a bunch of stuff to even start to make sense of it all.
6977
+ http://alex-charlton.com/posts/Dithering_on_the_GPU/
7078
* Resources:
7179
+ Resurrect 64 color palette by [Kerrie Lake](https://lospec.com/kerrielake).
80+
+ Apollo palette by [Adam C Younis](https://lospec.com/adamcyounis)
7281

7382
## License
7483

‎cli/src/args.rs

+46-5
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ pub enum Commands {
3838
/// Optional output file
3939
#[clap(short, long, value_parser)]
4040
output: Option<PathBuf>,
41-
/// List of RGB replacement colors formatted as "#RRGGBB,#RRGGBB"
41+
/// List of RGB replacement colors formatted as "#RRGGBB,#RRGGBB" or path to a palette image
4242
#[clap(short, long, value_parser = validate_palette)]
4343
palette: Palette,
4444
/// Mix function to apply on the result
@@ -146,7 +146,7 @@ fn validate_k(s: &str) -> Result<u32> {
146146
}
147147

148148
fn validate_filenames(s: &str) -> Result<PathBuf> {
149-
if s.len() > 4 && s.ends_with(".png") || s.ends_with(".jpg") {
149+
if s.len() > 4 && (s.ends_with(".png") || s.ends_with(".jpg")) {
150150
Ok(PathBuf::from(s))
151151
} else {
152152
Err(anyhow!("Only support png or jpg files."))
@@ -158,10 +158,36 @@ fn validate_palette(s: &str) -> Result<Palette> {
158158
if re.is_match(s) {
159159
parse_colors(s)
160160
} else {
161-
Err(anyhow!(
162-
"The palette should be define as \"#ffaa12,#fe7845,#aabbff\""
163-
))
161+
let path = PathBuf::from(s);
162+
if s.len() > 4 && (s.ends_with(".png") || s.ends_with(".jpg")) && path.exists() {
163+
parse_palette(&path)
164+
} else {
165+
Err(anyhow!(
166+
"The palette should be a path to an image file, or defined as \"#RRGGBB,#RRGGBB,#RRGGBB\""
167+
))
168+
}
169+
}
170+
}
171+
172+
fn parse_palette(path: &PathBuf) -> Result<Palette> {
173+
let image = image::open(&path)?.to_rgba8();
174+
let (width, height) = image.dimensions();
175+
let pixel_count = width as usize * height as usize;
176+
177+
if pixel_count > 512 {
178+
return Err(anyhow!(
179+
"Trying to load a palette with more than 512 colors"
180+
));
164181
}
182+
183+
let mut colors: Vec<[u8; 4]> = bytemuck::cast_slice(image.as_raw()).to_vec();
184+
colors.sort();
185+
colors.dedup();
186+
187+
if colors.len() < pixel_count {
188+
return Err(anyhow!("Trying to load a palette with recuring colors"));
189+
}
190+
Ok(Palette { colors })
165191
}
166192

167193
fn parse_colors(colors: &str) -> Result<Palette> {
@@ -224,4 +250,19 @@ mod tests {
224250
let expected: Vec<[u8; 4]> = vec![[255, 255, 255, 255], [0, 0, 0, 255]];
225251
assert_eq!(colors, expected);
226252
}
253+
254+
#[test]
255+
fn test_parse_palette() {
256+
let manifest_dir = env!("CARGO_MANIFEST_DIR");
257+
let mut palette_path = PathBuf::from(manifest_dir);
258+
palette_path.push("../gfx/resurrect_64.png");
259+
260+
let parsed = parse_palette(&palette_path);
261+
262+
assert!(parsed.is_ok());
263+
264+
let parsed = parsed.unwrap();
265+
266+
assert_eq!(64, parsed.colors.len());
267+
}
227268
}

‎gfx/apollo-1x.png

277 Bytes
Loading

‎gfx/tokyo-find-dither-apollo.png

76 KB
Loading

0 commit comments

Comments
 (0)
Please sign in to comment.