shithub: pokecrystal

ref: d4a6a1b2bcf94915020f2a6c96b8004479185325
dir: /tools/pokemon_animation_graphics.c/

View raw version
#define PROGRAM_NAME "pokemon_animation_graphics"
#define USAGE_OPTS "[-h|--help] [-o|--output front.animated.2bpp] [-t|--tilemap front.animated.tilemap] [--girafarig] front.2bpp front.dimensions"

#include "common.h"

struct Options {
	const char *out_filename;
	const char *map_filename;
	bool girafarig;
};

void parse_args(int argc, char *argv[], struct Options *options) {
	struct option long_options[] = {
		{"output", required_argument, 0, 'o'},
		{"tilemap", required_argument, 0, 't'},
		{"girafarig", no_argument, 0, 'g'},
		{"help", no_argument, 0, 'h'},
		{0}
	};
	for (int opt; (opt = getopt_long(argc, argv, "o:t:h", long_options)) != -1;) {
		switch (opt) {
		case 'o':
			options->out_filename = optarg;
			break;
		case 't':
			options->map_filename = optarg;
			break;
		case 'g':
			options->girafarig = true;
			break;
		case 'h':
			usage_exit(0);
			break;
		default:
			usage_exit(1);
		}
	}
}

#define TILE_SIZE 16

void transpose_tiles(uint8_t *tiles, int width, int size) {
	uint8_t *new_tiles = malloc_verbose(size);
	for (int i = 0; i < size; i++) {
		int j = i / TILE_SIZE * width * TILE_SIZE;
		j = (j / size) * TILE_SIZE + j % size + i % TILE_SIZE;
		new_tiles[j] = tiles[i];
	}
	memcpy(tiles, new_tiles, size);
	free(new_tiles);
}

int get_tile_index(const uint8_t *tile, const uint8_t *tiles, int num_tiles, int preferred_tile_id) {
	if (preferred_tile_id >= 0 && preferred_tile_id < num_tiles) {
		if (!memcmp(tile, &tiles[preferred_tile_id * TILE_SIZE], TILE_SIZE)) {
			return preferred_tile_id;
		}
	}
	for (int i = 0; i < num_tiles; i++) {
		if (!memcmp(tile, &tiles[i * TILE_SIZE], TILE_SIZE)) {
			return i;
		}
	}
	return -1;
}

uint8_t *read_tiles(const char *filename, int width, long *tiles_size) {
	int frame_size = width * width * TILE_SIZE;

	uint8_t *tiles = read_u8(filename, tiles_size);
	if (!*tiles_size) {
		error_exit("%s: empty file\n", filename);
	} else if (*tiles_size % TILE_SIZE) {
		error_exit("%s: not divisible into 8x8-px 2bpp tiles\n", filename);
	} else if (*tiles_size % frame_size) {
		error_exit("%s: not divisible into %dx%d-tile frames\n", filename, width, width);
	}

	int num_frames = *tiles_size / frame_size;
	for (int i = 0; i < num_frames; i++) {
		transpose_tiles(&tiles[i * frame_size], width, frame_size);
	}

	return tiles;
}

void write_graphics(const char *filename, const uint8_t *tiles, long tiles_size, int num_tiles_per_frame, bool girafarig) {
	int max_size = tiles_size;
	int max_num_tiles = max_size / TILE_SIZE;
	if (girafarig) {
		// Ensure space for a duplicate of tile 0 at the end
		max_size += TILE_SIZE;
	}
	uint8_t *data = malloc_verbose(max_size);

	int num_tiles = 0;
#define DATA_APPEND_TILES(tile, length) do { \
	memcpy(&data[num_tiles * TILE_SIZE], &tiles[(tile) * TILE_SIZE], (length) * TILE_SIZE); \
	num_tiles += (length); \
} while (0)
	// Copy the first frame directly
	DATA_APPEND_TILES(0, num_tiles_per_frame);
	// Skip redundant tiles in the animated frames
	for (int i = num_tiles_per_frame; i < max_num_tiles; i++) {
		int index = get_tile_index(&tiles[i * TILE_SIZE], data, num_tiles, i % num_tiles_per_frame);
		if (index == -1) {
			DATA_APPEND_TILES(i, 1);
		}
	}
	if (girafarig) {
		// Add a duplicate of tile 0 to the end
		DATA_APPEND_TILES(0, 1);
	}
#undef DATA_APPEND_TILES

	write_u8(filename, data, num_tiles * TILE_SIZE);
	free(data);
}

void write_tilemap(const char *filename, const uint8_t *tiles, long tiles_size, int num_tiles_per_frame, bool girafarig) {
	int size = tiles_size / TILE_SIZE;
	uint8_t *data = malloc_verbose(size);

	int num_tiles = num_tiles_per_frame;
	// Copy the first frame directly
	for (int i = 0; i < num_tiles_per_frame; i++) {
		data[i] = i;
	}
	// Skip redundant tiles in the animated frames
	for (int i = num_tiles_per_frame; i < size; i++) {
		int index = get_tile_index(&tiles[i * TILE_SIZE], tiles, i, i % num_tiles_per_frame);
		int tile;
		if (girafarig && index == 0) {
			tile = num_tiles;
		} else if (index == -1) {
			tile = num_tiles++;
		} else {
			tile = data[index];
		}
		data[i] = tile;
	}

	write_u8(filename, data, size);
	free(data);
}

int main(int argc, char *argv[]) {
	struct Options options = {0};
	parse_args(argc, argv, &options);

	argc -= optind;
	argv += optind;
	if (argc < 2) {
		usage_exit(1);
	}

	int width;
	read_dimensions(argv[1], &width);
	long tiles_size;
	uint8_t *tiles = read_tiles(argv[0], width, &tiles_size);

	if (options.out_filename) {
		write_graphics(options.out_filename, tiles, tiles_size, width * width, options.girafarig);
	}
	if (options.map_filename) {
		write_tilemap(options.map_filename, tiles, tiles_size, width * width, options.girafarig);
	}

	free(tiles);
	return 0;
}