@@ -38,7 +38,7 @@ pub enum Commands {
38
38
/// Optional output file
39
39
#[ clap( short, long, value_parser) ]
40
40
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
42
42
#[ clap( short, long, value_parser = validate_palette) ]
43
43
palette : Palette ,
44
44
/// Mix function to apply on the result
@@ -146,7 +146,7 @@ fn validate_k(s: &str) -> Result<u32> {
146
146
}
147
147
148
148
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" ) ) {
150
150
Ok ( PathBuf :: from ( s) )
151
151
} else {
152
152
Err ( anyhow ! ( "Only support png or jpg files." ) )
@@ -158,10 +158,36 @@ fn validate_palette(s: &str) -> Result<Palette> {
158
158
if re. is_match ( s) {
159
159
parse_colors ( s)
160
160
} 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
+ ) ) ;
164
181
}
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 } )
165
191
}
166
192
167
193
fn parse_colors ( colors : & str ) -> Result < Palette > {
@@ -224,4 +250,19 @@ mod tests {
224
250
let expected: Vec < [ u8 ; 4 ] > = vec ! [ [ 255 , 255 , 255 , 255 ] , [ 0 , 0 , 0 , 255 ] ] ;
225
251
assert_eq ! ( colors, expected) ;
226
252
}
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
+ }
227
268
}
0 commit comments