// Copyright (c) 2015 YamaArashi #include <stdio.h> #include <stdlib.h> #include <stdint.h> #include <stdbool.h> #include "global.h" #include "font.h" #include "gfx.h" #include "util.h" unsigned char gFontPalette[][3] = { {0x90, 0xC8, 0xFF}, // bg (saturated blue that contrasts well with the shadow color) {0x38, 0x38, 0x38}, // fg (dark grey) {0xD8, 0xD8, 0xD8}, // shadow (light grey) {0xFF, 0xFF, 0xFF} // box (white) }; static void ConvertFromLatinFont(unsigned char *src, unsigned char *dest, unsigned int numRows) { unsigned int srcPixelsOffset = 0; for (unsigned int row = 0; row < numRows; row++) { for (unsigned int column = 0; column < 16; column++) { for (unsigned int glyphTile = 0; glyphTile < 4; glyphTile++) { unsigned int pixelsX = (column * 16) + ((glyphTile & 1) * 8); for (unsigned int i = 0; i < 8; i++) { unsigned int pixelsY = (row * 16) + ((glyphTile >> 1) * 8) + i; unsigned int destPixelsOffset = (pixelsY * 64) + (pixelsX / 4); dest[destPixelsOffset] = src[srcPixelsOffset + 1]; dest[destPixelsOffset + 1] = src[srcPixelsOffset]; srcPixelsOffset += 2; } } } } } static void ConvertToLatinFont(unsigned char *src, unsigned char *dest, unsigned int numRows) { unsigned int destPixelsOffset = 0; for (unsigned int row = 0; row < numRows; row++) { for (unsigned int column = 0; column < 16; column++) { for (unsigned int glyphTile = 0; glyphTile < 4; glyphTile++) { unsigned int pixelsX = (column * 16) + ((glyphTile & 1) * 8); for (unsigned int i = 0; i < 8; i++) { unsigned int pixelsY = (row * 16) + ((glyphTile >> 1) * 8) + i; unsigned int srcPixelsOffset = (pixelsY * 64) + (pixelsX / 4); dest[destPixelsOffset] = src[srcPixelsOffset + 1]; dest[destPixelsOffset + 1] = src[srcPixelsOffset]; destPixelsOffset += 2; } } } } } static void ConvertFromHalfwidthJapaneseFont(unsigned char *src, unsigned char *dest, unsigned int numRows) { for (unsigned int row = 0; row < numRows; row++) { for (unsigned int column = 0; column < 16; column++) { unsigned int glyphIndex = (row * 16) + column; for (unsigned int glyphTile = 0; glyphTile < 2; glyphTile++) { unsigned int pixelsX = column * 8; unsigned int srcPixelsOffset = 512 * (glyphIndex >> 4) + 16 * (glyphIndex & 0xF) + 256 * glyphTile; for (unsigned int i = 0; i < 8; i++) { unsigned int pixelsY = (row * 16) + (glyphTile * 8) + i; unsigned int destPixelsOffset = (pixelsY * 32) + (pixelsX / 4); dest[destPixelsOffset] = src[srcPixelsOffset + 1]; dest[destPixelsOffset + 1] = src[srcPixelsOffset]; srcPixelsOffset += 2; } } } } } static void ConvertToHalfwidthJapaneseFont(unsigned char *src, unsigned char *dest, unsigned int numRows) { for (unsigned int row = 0; row < numRows; row++) { for (unsigned int column = 0; column < 16; column++) { unsigned int glyphIndex = (row * 16) + column; for (unsigned int glyphTile = 0; glyphTile < 2; glyphTile++) { unsigned int pixelsX = column * 8; unsigned int destPixelsOffset = 512 * (glyphIndex >> 4) + 16 * (glyphIndex & 0xF) + 256 * glyphTile; for (unsigned int i = 0; i < 8; i++) { unsigned int pixelsY = (row * 16) + (glyphTile * 8) + i; unsigned int srcPixelsOffset = (pixelsY * 32) + (pixelsX / 4); dest[destPixelsOffset] = src[srcPixelsOffset + 1]; dest[destPixelsOffset + 1] = src[srcPixelsOffset]; destPixelsOffset += 2; } } } } } static void ConvertFromFullwidthJapaneseFont(unsigned char *src, unsigned char *dest, unsigned int numRows) { for (unsigned int row = 0; row < numRows; row++) { for (unsigned int column = 0; column < 16; column++) { unsigned int glyphIndex = (row * 16) + column; for (unsigned int glyphTile = 0; glyphTile < 4; glyphTile++) { unsigned int pixelsX = (column * 16) + ((glyphTile & 1) * 8); unsigned int srcPixelsOffset = 512 * (glyphIndex >> 3) + 32 * (glyphIndex & 7) + 256 * (glyphTile >> 1) + 16 * (glyphTile & 1); for (unsigned int i = 0; i < 8; i++) { unsigned int pixelsY = (row * 16) + ((glyphTile >> 1) * 8) + i; unsigned int destPixelsOffset = (pixelsY * 64) + (pixelsX / 4); dest[destPixelsOffset] = src[srcPixelsOffset + 1]; dest[destPixelsOffset + 1] = src[srcPixelsOffset]; srcPixelsOffset += 2; } } } } } static void ConvertToFullwidthJapaneseFont(unsigned char *src, unsigned char *dest, unsigned int numRows) { for (unsigned int row = 0; row < numRows; row++) { for (unsigned int column = 0; column < 16; column++) { unsigned int glyphIndex = (row * 16) + column; for (unsigned int glyphTile = 0; glyphTile < 4; glyphTile++) { unsigned int pixelsX = (column * 16) + ((glyphTile & 1) * 8); unsigned int destPixelsOffset = 512 * (glyphIndex >> 3) + 32 * (glyphIndex & 7) + 256 * (glyphTile >> 1) + 16 * (glyphTile & 1); for (unsigned int i = 0; i < 8; i++) { unsigned int pixelsY = (row * 16) + ((glyphTile >> 1) * 8) + i; unsigned int srcPixelsOffset = (pixelsY * 64) + (pixelsX / 4); dest[destPixelsOffset] = src[srcPixelsOffset + 1]; dest[destPixelsOffset + 1] = src[srcPixelsOffset]; destPixelsOffset += 2; } } } } } static void SetFontPalette(struct Image *image) { image->hasPalette = true; image->palette.numColors = 4; for (int i = 0; i < image->palette.numColors; i++) { image->palette.colors[i].red = gFontPalette[i][0]; image->palette.colors[i].green = gFontPalette[i][1]; image->palette.colors[i].blue = gFontPalette[i][2]; } image->hasTransparency = false; } void ReadLatinFont(char *path, struct Image *image) { int fileSize; unsigned char *buffer = ReadWholeFile(path, &fileSize); int numGlyphs = fileSize / 64; if (numGlyphs % 16 != 0) FATAL_ERROR("The number of glyphs (%d) is not a multiple of 16.\n", numGlyphs); int numRows = numGlyphs / 16; image->width = 256; image->height = numRows * 16; image->bitDepth = 2; image->pixels = malloc(fileSize); if (image->pixels == NULL) FATAL_ERROR("Failed to allocate memory for font.\n"); ConvertFromLatinFont(buffer, image->pixels, numRows); free(buffer); SetFontPalette(image); } void WriteLatinFont(char *path, struct Image *image) { if (image->width != 256) FATAL_ERROR("The width of the font image (%d) is not 256.\n", image->width); if (image->height % 16 != 0) FATAL_ERROR("The height of the font image (%d) is not a multiple of 16.\n", image->height); int numRows = image->height / 16; int bufferSize = numRows * 16 * 64; unsigned char *buffer = malloc(bufferSize); if (buffer == NULL) FATAL_ERROR("Failed to allocate memory for font.\n"); ConvertToLatinFont(image->pixels, buffer, numRows); WriteWholeFile(path, buffer, bufferSize); free(buffer); } void ReadHalfwidthJapaneseFont(char *path, struct Image *image) { int fileSize; unsigned char *buffer = ReadWholeFile(path, &fileSize); int glyphSize = 32; if (fileSize % glyphSize != 0) FATAL_ERROR("The file size (%d) is not a multiple of %d.\n", fileSize, glyphSize); int numGlyphs = fileSize / glyphSize; if (numGlyphs % 16 != 0) FATAL_ERROR("The number of glyphs (%d) is not a multiple of 16.\n", numGlyphs); int numRows = numGlyphs / 16; image->width = 128; image->height = numRows * 16; image->bitDepth = 2; image->pixels = malloc(fileSize); if (image->pixels == NULL) FATAL_ERROR("Failed to allocate memory for font.\n"); ConvertFromHalfwidthJapaneseFont(buffer, image->pixels, numRows); free(buffer); SetFontPalette(image); } void WriteHalfwidthJapaneseFont(char *path, struct Image *image) { if (image->width != 128) FATAL_ERROR("The width of the font image (%d) is not 128.\n", image->width); if (image->height % 16 != 0) FATAL_ERROR("The height of the font image (%d) is not a multiple of 16.\n", image->height); int numRows = image->height / 16; int bufferSize = numRows * 16 * 32; unsigned char *buffer = malloc(bufferSize); if (buffer == NULL) FATAL_ERROR("Failed to allocate memory for font.\n"); ConvertToHalfwidthJapaneseFont(image->pixels, buffer, numRows); WriteWholeFile(path, buffer, bufferSize); free(buffer); } void ReadFullwidthJapaneseFont(char *path, struct Image *image) { int fileSize; unsigned char *buffer = ReadWholeFile(path, &fileSize); int numGlyphs = fileSize / 64; if (numGlyphs % 16 != 0) FATAL_ERROR("The number of glyphs (%d) is not a multiple of 16.\n", numGlyphs); int numRows = numGlyphs / 16; image->width = 256; image->height = numRows * 16; image->bitDepth = 2; image->pixels = malloc(fileSize); if (image->pixels == NULL) FATAL_ERROR("Failed to allocate memory for font.\n"); ConvertFromFullwidthJapaneseFont(buffer, image->pixels, numRows); free(buffer); SetFontPalette(image); } void WriteFullwidthJapaneseFont(char *path, struct Image *image) { if (image->width != 256) FATAL_ERROR("The width of the font image (%d) is not 256.\n", image->width); if (image->height % 16 != 0) FATAL_ERROR("The height of the font image (%d) is not a multiple of 16.\n", image->height); int numRows = image->height / 16; int bufferSize = numRows * 16 * 64; unsigned char *buffer = malloc(bufferSize); if (buffer == NULL) FATAL_ERROR("Failed to allocate memory for font.\n"); ConvertToFullwidthJapaneseFont(image->pixels, buffer, numRows); WriteWholeFile(path, buffer, bufferSize); free(buffer); }