mirror of
https://github.com/G2-Games/lbee-utils.git
synced 2025-04-19 07:12:55 -05:00
Improved modularity of tool
This commit is contained in:
parent
d31917ae12
commit
42be12392a
3 changed files with 98 additions and 144 deletions
|
@ -107,13 +107,9 @@ pub fn indexed_gen_palette(
|
||||||
.map(|c| Rgba([c.r, c.g, c.b, c.a]))
|
.map(|c| Rgba([c.r, c.g, c.b, c.a]))
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
dbg!(gen_palette.len());
|
|
||||||
|
|
||||||
let mut output_palette = vec![Rgba([0, 0, 0, 0]); 256];
|
let mut output_palette = vec![Rgba([0, 0, 0, 0]); 256];
|
||||||
output_palette[0..gen_palette.len()].copy_from_slice(&gen_palette);
|
output_palette[0..gen_palette.len()].copy_from_slice(&gen_palette);
|
||||||
|
|
||||||
dbg!(output_palette.len());
|
|
||||||
|
|
||||||
Ok((indicies, output_palette))
|
Ok((indicies, output_palette))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,6 +16,9 @@ pub enum CzError {
|
||||||
#[error("Bitmap size does not match image size")]
|
#[error("Bitmap size does not match image size")]
|
||||||
BitmapFormat,
|
BitmapFormat,
|
||||||
|
|
||||||
|
#[error("CZ version is invalid: {}", 0)]
|
||||||
|
InvalidVersion(u32),
|
||||||
|
|
||||||
#[error("File data is incorrect, it might be corrupt: {0}")]
|
#[error("File data is incorrect, it might be corrupt: {0}")]
|
||||||
Corrupt(String),
|
Corrupt(String),
|
||||||
|
|
||||||
|
@ -156,10 +159,10 @@ impl CommonHeader {
|
||||||
self.version
|
self.version
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_version<I: TryInto<CzVersion>>(&mut self, version: I) -> Result<(), ()> {
|
pub fn set_version<I: TryInto<CzVersion> + Into<u32> + Clone>(&mut self, version: I) -> Result<(), CzError> {
|
||||||
self.version = match version.try_into() {
|
self.version = match version.clone().try_into() {
|
||||||
Ok(val) => val,
|
Ok(val) => val,
|
||||||
Err(_) => return Err(()),
|
Err(_) => return Err(CzError::InvalidVersion(version.into())),
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
use cz::DynamicCz;
|
use cz::DynamicCz;
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
use std::path::PathBuf;
|
|
||||||
|
|
||||||
use clap::{error::ErrorKind, Error, Parser, Subcommand};
|
use clap::{error::ErrorKind, Error, Parser, Subcommand};
|
||||||
|
|
||||||
#[derive(Parser)]
|
#[derive(Parser)]
|
||||||
|
@ -54,7 +52,7 @@ enum Commands {
|
||||||
|
|
||||||
/// Output CZ file bit depth
|
/// Output CZ file bit depth
|
||||||
#[arg(short, long, value_name = "BIT DEPTH")]
|
#[arg(short, long, value_name = "BIT DEPTH")]
|
||||||
depth: Option<u8>,
|
depth: Option<u16>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -65,25 +63,16 @@ fn main() {
|
||||||
match &cli.command {
|
match &cli.command {
|
||||||
Commands::Decode { input, output, batch } => {
|
Commands::Decode { input, output, batch } => {
|
||||||
if !input.exists() {
|
if !input.exists() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "The input file/folder provided does not exist\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("The input file/folder provided does not exist\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if *batch {
|
if *batch {
|
||||||
if input.is_file() {
|
if input.is_file() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "Batch input must be a directory\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Batch input must be a directory\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if output.is_none() || output.as_ref().unwrap().is_file() {
|
if output.is_none() || output.as_ref().unwrap().is_file() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "Batch output must be a directory\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Batch output must be a directory\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for entry in walkdir::WalkDir::new(input).max_depth(1) {
|
for entry in walkdir::WalkDir::new(input).max_depth(1) {
|
||||||
|
@ -124,154 +113,120 @@ fn main() {
|
||||||
}
|
}
|
||||||
Commands::Replace { batch, input, replacement, output, version, depth } => {
|
Commands::Replace { batch, input, replacement, output, version, depth } => {
|
||||||
if !input.exists() {
|
if !input.exists() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "The original file provided does not exist\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("The original file provided does not exist\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if !replacement.exists() {
|
if !replacement.exists() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "The replacement file provided does not exist\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("The replacement file provided does not exist\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If it's a batch replacement, we want directories to search
|
||||||
if *batch {
|
if *batch {
|
||||||
if input.is_file() {
|
if !input.is_dir() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "Batch input location must be a directory\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Batch input must be a directory\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if replacement.is_file() {
|
if !replacement.is_dir() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "Batch replacement location must be a directory\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Batch replacement location must be a directory\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if output.is_file() {
|
if !output.is_dir() {
|
||||||
Error::raw(
|
Error::raw(ErrorKind::ValueValidation, "Batch output location must be a directory\n").exit()
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Batch output must be a directory\n")
|
|
||||||
).exit()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for entry in walkdir::WalkDir::new(input).max_depth(1) {
|
// Replace all the files within the directory and print errors for them
|
||||||
|
for entry in walkdir::WalkDir::new(input)
|
||||||
|
.max_depth(1)
|
||||||
|
.same_file_system(true)
|
||||||
|
{
|
||||||
let path = entry.unwrap().into_path();
|
let path = entry.unwrap().into_path();
|
||||||
if !path.is_file() {
|
if !path.is_file() {
|
||||||
continue;
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
let filename = PathBuf::from(path.file_name().unwrap());
|
// Set the replacement image to the same name as the original file
|
||||||
|
let mut final_replacement = replacement.to_path_buf();
|
||||||
|
final_replacement.push(PathBuf::from(path.file_name().unwrap()).with_extension("png"));
|
||||||
|
|
||||||
let mut final_path = output.clone();
|
// Set the replacement image to the same name as the original file
|
||||||
final_path.push(&filename);
|
let mut final_output = output.to_path_buf();
|
||||||
|
final_output.push(path.file_name().unwrap());
|
||||||
|
|
||||||
let mut final_replacement = replacement.clone();
|
if let Err(error) = replace_cz(
|
||||||
final_replacement.push(filename.with_extension("png"));
|
&path,
|
||||||
|
&final_output,
|
||||||
let repl_img = match image::open(&final_replacement) {
|
&final_replacement,
|
||||||
Ok(img) => img,
|
version,
|
||||||
Err(_) => {
|
depth
|
||||||
Error::raw(
|
) {
|
||||||
ErrorKind::ValueValidation,
|
Error::raw(ErrorKind::ValueValidation, format!("{:?} - {}\n", path, error)).print().unwrap();
|
||||||
format!("Could not open replacement file as an image: {}\n", final_replacement.into_os_string().to_str().unwrap())
|
|
||||||
).print().unwrap();
|
|
||||||
continue;
|
|
||||||
},
|
|
||||||
};
|
|
||||||
let repl_img = repl_img.to_rgba8();
|
|
||||||
|
|
||||||
let mut cz = match DynamicCz::open(&path) {
|
|
||||||
Ok(cz) => cz,
|
|
||||||
Err(_) => {
|
|
||||||
Error::raw(
|
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Could not open input as a CZ file: {}\n", path.into_os_string().to_str().unwrap())
|
|
||||||
).print().unwrap();
|
|
||||||
continue;
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
cz.header_mut().set_width(repl_img.width() as u16);
|
|
||||||
cz.header_mut().set_height(repl_img.height() as u16);
|
|
||||||
cz.set_bitmap(repl_img.into_raw());
|
|
||||||
cz.remove_palette();
|
|
||||||
|
|
||||||
if let Some(depth) = depth {
|
|
||||||
cz.header_mut().set_depth(*depth as u16)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(ver) = version {
|
|
||||||
match cz.header_mut().set_version(*ver) {
|
|
||||||
Ok(_) => (),
|
|
||||||
Err(_) => {
|
|
||||||
Error::raw(
|
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Invalid CZ Version {}; expected 0, 1, 2, 3, or 4\n", ver)
|
|
||||||
).exit()
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
cz.save_as_cz(&final_path).unwrap();
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
let mut cz = match DynamicCz::open(input) {
|
if !input.is_file() {
|
||||||
Ok(cz) => cz,
|
Error::raw(ErrorKind::ValueValidation, "Input must be a file\n").exit()
|
||||||
Err(err) => {
|
|
||||||
Error::raw(
|
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Could not open input as a CZ file: {}\n", err)
|
|
||||||
).exit()
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
let repl_img = match image::open(replacement) {
|
|
||||||
Ok(img) => img,
|
|
||||||
Err(err) => {
|
|
||||||
Error::raw(
|
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Could not open replacement file as an image: {}\n", err)
|
|
||||||
).exit()
|
|
||||||
},
|
|
||||||
};
|
|
||||||
let repl_img = repl_img.to_rgba8();
|
|
||||||
|
|
||||||
cz.header_mut().set_width(repl_img.width() as u16);
|
|
||||||
cz.header_mut().set_height(repl_img.height() as u16);
|
|
||||||
cz.set_bitmap(repl_img.into_raw());
|
|
||||||
cz.remove_palette();
|
|
||||||
|
|
||||||
if let Some(depth) = depth {
|
|
||||||
cz.header_mut().set_depth(*depth as u16)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(ver) = version {
|
if !replacement.is_file() {
|
||||||
match cz.header_mut().set_version(*ver) {
|
Error::raw(ErrorKind::ValueValidation, "Replacement must be a file\n").exit()
|
||||||
Ok(_) => (),
|
|
||||||
Err(_) => {
|
|
||||||
Error::raw(
|
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Invalid CZ Version {}; expected 0, 1, 2, 3, or 4\n", ver)
|
|
||||||
).exit()
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
match cz.save_as_cz(output) {
|
if !output.is_file() {
|
||||||
Ok(_) => (),
|
Error::raw(ErrorKind::ValueValidation, "Replacement output must be a file\n").exit()
|
||||||
Err(err) => {
|
|
||||||
Error::raw(
|
|
||||||
ErrorKind::ValueValidation,
|
|
||||||
format!("Failed to save CZ file: {}\n", err)
|
|
||||||
).exit()
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Replace the input file with the new image
|
||||||
|
replace_cz(
|
||||||
|
&input,
|
||||||
|
&output,
|
||||||
|
&replacement,
|
||||||
|
version,
|
||||||
|
depth
|
||||||
|
).unwrap();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Replace a CZ file with the bitmap of a PNG file
|
||||||
|
fn replace_cz<P: ?Sized + AsRef<Path>>(
|
||||||
|
input_path: &P,
|
||||||
|
output_path: &P,
|
||||||
|
replacement_path: &P,
|
||||||
|
version: &Option<u8>,
|
||||||
|
depth: &Option<u16>,
|
||||||
|
) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
let path = input_path.as_ref();
|
||||||
|
if !path.is_file() {
|
||||||
|
return Err("Input path is not a file".into())
|
||||||
|
}
|
||||||
|
|
||||||
|
if !replacement_path.as_ref().exists() || !replacement_path.as_ref().is_file() {
|
||||||
|
return Err("Replacement path does not exist or is not a file".into())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Open the replacement image and convert it to RGBA8
|
||||||
|
let repl_img = image::open(&replacement_path)?.to_rgba8();
|
||||||
|
|
||||||
|
// Open the original CZ file
|
||||||
|
let mut cz = DynamicCz::open(&path)?;
|
||||||
|
|
||||||
|
// Set CZ header parameters and the new bitmap
|
||||||
|
cz.header_mut().set_width(repl_img.width() as u16);
|
||||||
|
cz.header_mut().set_height(repl_img.height() as u16);
|
||||||
|
cz.set_bitmap(repl_img.into_raw());
|
||||||
|
cz.remove_palette();
|
||||||
|
|
||||||
|
if let Some(depth) = depth {
|
||||||
|
cz.header_mut().set_depth(*depth)
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(ver) = version {
|
||||||
|
cz.header_mut().set_version(*ver)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save the file to the proper output location
|
||||||
|
cz.save_as_cz(&output_path.as_ref()).unwrap();
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue