agb/agb-image-converter/src/lib.rs

125 lines
3 KiB
Rust
Raw Normal View History

2021-07-21 22:46:22 +01:00
use proc_macro::TokenStream;
2021-07-22 18:43:27 +01:00
use syn::parse_macro_input;
2021-07-21 22:46:22 +01:00
use std::path::Path;
2021-06-05 17:45:21 +01:00
use quote::{format_ident, quote};
2022-02-15 21:28:21 +00:00
mod aseprite;
2021-04-19 23:15:03 +01:00
mod colour;
2021-07-21 22:46:22 +01:00
mod config;
2021-04-20 00:40:07 +01:00
mod image_loader;
mod palette16;
2021-04-20 20:41:04 +01:00
mod rust_generator;
2021-04-20 00:40:07 +01:00
use image_loader::Image;
2021-04-19 23:15:03 +01:00
use colour::Colour;
#[derive(Debug, Clone, Copy)]
pub(crate) enum TileSize {
Tile8,
Tile16,
2021-10-30 18:22:09 +01:00
Tile32,
}
2021-04-20 00:40:07 +01:00
impl TileSize {
2021-04-20 22:56:47 +01:00
fn to_size(self) -> usize {
match self {
2021-04-20 00:40:07 +01:00
TileSize::Tile8 => 8,
TileSize::Tile16 => 16,
2021-10-30 18:22:09 +01:00
TileSize::Tile32 => 32,
2021-04-20 00:40:07 +01:00
}
}
}
#[proc_macro]
pub fn include_gfx(input: TokenStream) -> TokenStream {
2021-07-22 18:43:27 +01:00
let input = parse_macro_input!(input as syn::LitStr);
2021-06-05 17:47:13 +01:00
2021-07-22 18:43:27 +01:00
let filename = input.value();
let root = std::env::var("CARGO_MANIFEST_DIR").expect("Failed to get cargo manifest dir");
let path = Path::new(&root).join(&*filename);
2021-07-21 22:46:22 +01:00
let parent = path
.parent()
.expect("Expected a parent directory for the path");
let config = config::parse(&path.to_string_lossy());
let module_name = format_ident!(
"{}",
path.file_stem()
.expect("Expected a file stem")
.to_string_lossy()
);
let include_path = path.to_string_lossy();
let images = config.images();
let image_code = images.iter().map(|(image_name, &image)| {
convert_image(image, parent, &image_name, &config.crate_prefix())
});
let module = quote! {
2022-01-02 17:54:44 +00:00
mod #module_name {
const _: &[u8] = include_bytes!(#include_path);
#(#image_code)*
}
};
TokenStream::from(module)
2021-04-20 00:40:07 +01:00
}
2021-07-21 22:46:22 +01:00
fn convert_image(
settings: &dyn config::Image,
parent: &Path,
variable_name: &str,
crate_prefix: &str,
2021-07-22 19:04:30 +01:00
) -> proc_macro2::TokenStream {
let image_filename = &parent.join(&settings.filename());
let image = Image::load_from_file(image_filename);
2021-04-20 00:40:07 +01:00
let tile_size = settings.tilesize().to_size();
2021-04-20 00:40:07 +01:00
if image.width % tile_size != 0 || image.height % tile_size != 0 {
panic!("Image size not a multiple of tile size");
}
let optimiser = optimiser_for_image(&image, tile_size);
let optimisation_results = optimiser.optimise_palettes(settings.transparent_colour());
2021-04-20 00:40:07 +01:00
2021-04-20 20:41:04 +01:00
rust_generator::generate_code(
variable_name,
2021-04-20 20:41:04 +01:00
&optimisation_results,
&image,
&image_filename.to_string_lossy(),
settings.tilesize(),
crate_prefix.to_owned(),
)
}
2021-04-20 00:40:07 +01:00
fn optimiser_for_image(image: &Image, tile_size: usize) -> palette16::Palette16Optimiser {
let tiles_x = image.width / tile_size;
let tiles_y = image.height / tile_size;
let mut palette_optimiser = palette16::Palette16Optimiser::new();
for y in 0..tiles_y {
for x in 0..tiles_x {
let mut palette = palette16::Palette16::new();
for j in 0..tile_size {
for i in 0..tile_size {
let colour = image.colour(x * tile_size + i, y * tile_size + j);
2021-04-20 00:57:47 +01:00
palette.add_colour(colour);
2021-04-20 00:40:07 +01:00
}
}
palette_optimiser.add_palette(palette);
}
}
palette_optimiser
}