#define _CRT_SECURE_NO_WARNINGS #define STBI_FAILURE_USERMSG #define STB_IMAGE_IMPLEMENTATION #include #define SBTT_STATIC #define STB_TRUETYPE_IMPLEMENTATION #include //#define WT_RENDER_FONT_FILE #ifdef WT_RENDER_FONT_FILE #define STB_IMAGE_WRITE_IMPLEMENTATION #include #endif #include #include "Dengine/AssetManager.h" #include "Dengine/Debug.h" #include "Dengine/MemoryArena.h" #include "Dengine/OpenGL.h" #include "Dengine/Platform.h" /* ********************************* * XML Operations ********************************* */ enum XmlTokenType { xmltokentype_unknown, xmltokentype_openArrow, xmltokentype_closeArrow, xmltokentype_name, xmltokentype_value, xmltokentype_equals, xmltokentype_quotes, xmltokentype_backslash, xmltokentype_count, }; typedef struct XmlToken { // TODO(doyle): Dynamic size string in tokens maybe. enum XmlTokenType type; char string[128]; i32 len; } XmlToken; INTERNAL XmlToken *tokeniseXmlBuffer(MemoryArena *arena, char *buffer, i32 bufferSize, int *numTokens) { XmlToken *xmlTokens = PLATFORM_MEM_ALLOC(arena, 8192, XmlToken); i32 tokenIndex = 0; for (i32 i = 0; i < bufferSize; i++) { char c = (CAST(char *) buffer)[i]; switch (c) { case '<': case '>': case '=': case '/': { enum XmlTokenType type = xmltokentype_unknown; if (c == '<') { type = xmltokentype_openArrow; } else if (c == '>') { type = xmltokentype_closeArrow; } else if (c == '=') { type = xmltokentype_equals; } else { type = xmltokentype_backslash; } xmlTokens[tokenIndex].type = type; xmlTokens[tokenIndex].len = 1; tokenIndex++; break; } case '"': { xmlTokens[tokenIndex].type = xmltokentype_value; for (i32 j = i + 1; j < bufferSize; j++) { char c = (CAST(char *) buffer)[j]; if (c == '"') { break; } else { xmlTokens[tokenIndex].string[xmlTokens[tokenIndex].len++] = c; #ifdef DENGINE_DEBUG ASSERT(xmlTokens[tokenIndex].len < ARRAY_COUNT(xmlTokens[tokenIndex].string)); #endif } } // NOTE(doyle): +1 to skip the closing quotes i += (xmlTokens[tokenIndex].len + 1); tokenIndex++; break; } default: { if ((c >= 'a' && c <= 'z') || c >= 'A' && c <= 'Z') { xmlTokens[tokenIndex].type = xmltokentype_name; for (i32 j = i; j < bufferSize; j++) { char c = (CAST(char *) buffer)[j]; if (c == ' ' || c == '=' || c == '>' || c == '<' || c == '\\') { break; } else { xmlTokens[tokenIndex] .string[xmlTokens[tokenIndex].len++] = c; #ifdef DENGINE_DEBUG ASSERT(xmlTokens[tokenIndex].len < ARRAY_COUNT(xmlTokens[tokenIndex].string)); #endif } } i += xmlTokens[tokenIndex].len; tokenIndex++; } break; } } } // TODO(doyle): Dynamic token allocation *numTokens = 8192; return xmlTokens; } INTERNAL XmlNode *buildXmlTree(MemoryArena *arena, XmlToken *xmlTokens, i32 numTokens) { XmlNode *root = PLATFORM_MEM_ALLOC(arena, 1, XmlNode); if (!root) return NULL; XmlNode *node = root; node->parent = node; // NOTE(doyle): Used for when closing a node with many children. We // automatically assign the next child after each child close within // a group. Hence on the last child, we open another node but the next // token indicates the group is closing- we need to set the last child's // next reference to NULL XmlNode *prevNode = NULL; for (i32 i = 0; i < numTokens; i++) { XmlToken *token = &xmlTokens[i]; switch (token->type) { case xmltokentype_openArrow: { /* Open arrows indicate closing parent node or new node name */ XmlToken *nextToken = &xmlTokens[++i]; if (nextToken->type == xmltokentype_backslash) { nextToken = &xmlTokens[++i]; if (common_strcmp(nextToken->string, node->parent->name) == 0) { node->parent->isClosed = TRUE; if (prevNode) { prevNode->next = NULL; } XmlNode *parent = node->parent; PLATFORM_MEM_FREE(arena, node, sizeof(XmlNode)); node = node->parent; } else { #ifdef DENGINE_DEBUG DEBUG_LOG( "Closing xml node name does not match parent name"); #endif } } else if (nextToken->type == xmltokentype_name) { node->name = nextToken->string; } else { #ifdef DENGINE_DEBUG DEBUG_LOG("Unexpected token type after open arrow"); #endif } token = nextToken; break; } case xmltokentype_name: { // TODO(doyle): Store latest attribute pointer so we aren't always // chasing the linked list each iteration. Do the same for children // node /* Xml Attributes are a linked list, get first free entry */ XmlAttribute *attribute = &node->attribute; if (attribute->init) { while (attribute->next) attribute = attribute->next; attribute->next = PLATFORM_MEM_ALLOC(arena, 1, XmlAttribute); attribute = attribute->next; } /* Just plain text is a node attribute name */ attribute->name = token->string; /* Followed by the value */ token = &xmlTokens[++i]; attribute->value = token->string; attribute->init = TRUE; break; } case xmltokentype_closeArrow: { XmlToken prevToken = xmlTokens[i - 1]; prevNode = node; /* Closed node means we can return to parent */ if (prevToken.type == xmltokentype_backslash) { node->isClosed = TRUE; node = node->parent; } if (!node->isClosed) { /* Unclosed node means next fields will be children of node */ /* If the first child is free allocate, otherwise we have to * iterate through the child's next node(s) */ if (!node->child) { // TODO(doyle): Mem alloc error checking node->child = PLATFORM_MEM_ALLOC(arena, 1, XmlNode); node->child->parent = node; node = node->child; } else { XmlNode *nodeToCheck = node->child; while (nodeToCheck->next) nodeToCheck = nodeToCheck->next; nodeToCheck->next = PLATFORM_MEM_ALLOC(arena, 1, XmlNode); nodeToCheck->next->parent = node; node = nodeToCheck->next; } } break; } default: { break; } } } return root; } INTERNAL void parseXmlTreeToGame(AssetManager *assetManager, MemoryArena *arena, XmlNode *root) { XmlNode *node = root; while (node) { if (common_strcmp(node->name, "TextureAtlas") == 0) { XmlNode *atlasXmlNode = node; TexAtlas *atlasEntry = NULL; if (common_strcmp(node->attribute.name, "imagePath") == 0) { char *imageName = atlasXmlNode->attribute.value; atlasEntry = asset_makeTexAtlas(assetManager, arena, imageName); char *dataDir = "data/textures/WorldTraveller/"; char imagePath[512] = {0}; common_strncat(imagePath, dataDir, common_strlen(dataDir)); common_strncat(imagePath, imageName, common_strlen(imageName)); asset_loadTextureImage(assetManager, arena, imagePath, imageName); atlasEntry->key = PLATFORM_MEM_ALLOC( arena, common_strlen(imageName) + 1, char); common_strncpy(atlasEntry->key, imageName, common_strlen(imageName)); atlasEntry->tex = asset_getTex(assetManager, imageName); XmlNode *atlasChildNode = atlasXmlNode->child; while (atlasChildNode) { if (common_strcmp(atlasChildNode->name, "SubTexture") == 0) { XmlAttribute *subTextureAttrib = &atlasChildNode->attribute; AtlasSubTexture newSubTexEntry = {0}; while (subTextureAttrib) { // TODO(doyle): Work around for now in xml reading, // reading the last node closing node not being // merged to the parent if (!subTextureAttrib->name) continue; if (common_strcmp(subTextureAttrib->name, "name") == 0) { char *value = subTextureAttrib->value; newSubTexEntry.key = value; } else if (common_strcmp(subTextureAttrib->name, "x") == 0) { char *value = subTextureAttrib->value; i32 valueLen = common_strlen(value); i32 intValue = common_atoi(value, valueLen); newSubTexEntry.rect.pos.x = CAST(f32) intValue; } else if (common_strcmp(subTextureAttrib->name, "y") == 0) { char *value = subTextureAttrib->value; i32 valueLen = common_strlen(value); i32 intValue = common_atoi(value, valueLen); newSubTexEntry.rect.pos.y = CAST(f32) intValue; } else if (common_strcmp(subTextureAttrib->name, "width") == 0) { char *value = subTextureAttrib->value; i32 valueLen = common_strlen(value); i32 intValue = common_atoi(value, valueLen); newSubTexEntry.rect.size.w = CAST(f32) intValue; } else if (common_strcmp(subTextureAttrib->name, "height") == 0) { char *value = subTextureAttrib->value; i32 valueLen = common_strlen(value); i32 intValue = common_atoi(value, valueLen); newSubTexEntry.rect.size.h = CAST(f32) intValue; } else { #ifdef DENGINE_DEBUG DEBUG_LOG( "Unsupported xml attribute in SubTexture"); #endif } subTextureAttrib = subTextureAttrib->next; } // TODO(doyle): XML specifies 0,0 top left, we // prefer 0,0 bottom right, so offset by size since 0,0 // is top left and size creates a bounding box below it newSubTexEntry.rect.pos.y = 1024 - newSubTexEntry.rect.pos.y; newSubTexEntry.rect.pos.y -= newSubTexEntry.rect.size.h; #ifdef DENGINE_DEBUG ASSERT(newSubTexEntry.key) #endif u32 subTexHashIndex = common_murmurHash2( newSubTexEntry.key, common_strlen(newSubTexEntry.key), 0xDEADBEEF); subTexHashIndex = subTexHashIndex % ARRAY_COUNT(atlasEntry->subTex); // NOTE(doyle): Hash collision AtlasSubTexture *subTexEntry = &atlasEntry->subTex[subTexHashIndex]; if (subTexEntry->key) { #ifdef DENGINE_DEBUG // NOTE(doyle): Two textures have the same access // name ASSERT(common_strcmp(subTexEntry->key, newSubTexEntry.key) != 0); #endif while (subTexEntry->next) subTexEntry = subTexEntry->next; subTexEntry->next = PLATFORM_MEM_ALLOC(arena, 1, AtlasSubTexture); subTexEntry = subTexEntry->next; } *subTexEntry = newSubTexEntry; i32 keyLen = common_strlen(newSubTexEntry.key); subTexEntry->key = PLATFORM_MEM_ALLOC(arena, keyLen + 1, char); common_strncpy(subTexEntry->key, newSubTexEntry.key, keyLen); } else { #ifdef DENGINE_DEBUG DEBUG_LOG("Unsupported xml node name not parsed"); #endif } atlasChildNode = atlasChildNode->next; } } else { #ifdef DENGINE_DEBUG DEBUG_LOG("Unsupported xml node"); #endif } } else { #ifdef DENGINE_DEBUG DEBUG_LOG("Unsupported xml node name not parsed"); #endif } node = node->next; } } INTERNAL void recursiveFreeXmlTree(MemoryArena *arena, XmlNode *node) { if (!node) { return; } else { // NOTE(doyle): First attribute is statically allocated, only if there's // more attributes do we dynamically allocate XmlAttribute *attrib = node->attribute.next; while (attrib) { XmlAttribute *next = attrib->next; attrib->name = NULL; attrib->value = NULL; PLATFORM_MEM_FREE(arena, attrib, sizeof(XmlAttribute)); attrib = next; } recursiveFreeXmlTree(arena, node->child); recursiveFreeXmlTree(arena, node->next); node->name = NULL; node->isClosed = FALSE; PLATFORM_MEM_FREE(arena, node, sizeof(XmlNode)); } } INTERNAL void freeXmlData(MemoryArena *arena, XmlToken *tokens, i32 numTokens, XmlNode *tree) { if (tree) recursiveFreeXmlTree(arena, tree); if (tokens) PLATFORM_MEM_FREE(arena, tokens, numTokens * sizeof(XmlToken)); } i32 asset_loadXmlFile(AssetManager *assetManager, MemoryArena *arena, PlatformFileRead *fileRead) { i32 result = 0; /* Tokenise buffer */ i32 numTokens = 0; XmlToken *xmlTokens = tokeniseXmlBuffer(arena, fileRead->buffer, fileRead->size, &numTokens); /* Build XML tree from tokens */ XmlNode *xmlTree = buildXmlTree(arena, xmlTokens, numTokens); if (xmlTree) { /* Parse XML tree to game structures */ parseXmlTreeToGame(assetManager, arena, xmlTree); } else { result = -1; } /* Free data */ freeXmlData(arena, xmlTokens, numTokens, xmlTree); return result; } /* ********************************* * Texture Operations ********************************* */ INTERNAL AtlasSubTexture *getAtlasSubTex(TexAtlas *atlas, char *key) { u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(atlas->subTex)); AtlasSubTexture *result = &atlas->subTex[hashIndex]; if (result->key) { while (result && common_strcmp(result->key, key) != 0) result = result->next; } return result; } INTERNAL AtlasSubTexture *getFreeAtlasSubTexSlot(TexAtlas *atlas, MemoryArena *arena, char *key) { u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(atlas->subTex)); AtlasSubTexture *result = &atlas->subTex[hashIndex]; if (result->key) { while (result->next) { if (common_strcmp(result->key, key) == 0) { // TODO(doyle): Error correction whereby if a tex atlas already // exists ASSERT(INVALID_CODE_PATH); } result = result->next; } result->next = PLATFORM_MEM_ALLOC(arena, 1, AtlasSubTexture); result = result->next; } return result; } INTERNAL Animation *getFreeAnimationSlot(Animation *table, u32 tableSize, MemoryArena *arena, char *key) { u32 hashIndex = common_getHashIndex(key, tableSize); Animation *result = &table[hashIndex]; if (result->key) { while (result->next) { if (common_strcmp(result->key, key) == 0) { // TODO(doyle): Error correction whereby if a tex atlas already // exists ASSERT(INVALID_CODE_PATH); } result = result->next; } result->next = PLATFORM_MEM_ALLOC(arena, 1, Animation); result = result->next; } return result; } Rect asset_getSubTexRect(TexAtlas *atlas, char *key) { AtlasSubTexture *subTex = getAtlasSubTex(atlas, key); Rect result = subTex->rect; return result; } // TODO(doyle): Switch to hash based lookup // TODO(doyle): Use pointers, so we can forward declare all assets? AudioVorbis *asset_getVorbis(AssetManager *assetManager, const enum AudioList type) { if (type < audiolist_count) return &assetManager->audio[type]; #ifdef DENGINE_DEBUG ASSERT(INVALID_CODE_PATH); #endif return NULL; } Texture *asset_getTex(AssetManager *const assetManager, const char *const key) { u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(assetManager->textures)); Texture *result = &assetManager->textures[hashIndex]; if (result->key) { while (result && common_strcmp(result->key, key) != 0) result = result->next; } return result; } TexAtlas *asset_makeTexAtlas(AssetManager *const assetManager, MemoryArena *arena, const char *const key) { u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(assetManager->texAtlas)); TexAtlas *result = &assetManager->texAtlas[hashIndex]; if (result->key) { while (result->next) { if (common_strcmp(result->key, key) == 0) { // TODO(doyle): Error correction whereby if a tex atlas already // exists ASSERT(INVALID_CODE_PATH); } result = result->next; } result->next = PLATFORM_MEM_ALLOC(arena, 1, TexAtlas); result = result->next; } return result; } TexAtlas *asset_getTexAtlas(AssetManager *const assetManager, const char *const key) { u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(assetManager->texAtlas)); TexAtlas *result = &assetManager->texAtlas[hashIndex]; if (result->key) { while (result && common_strcmp(result->key, key) != 0) result = result->next; } else { return NULL; } return result; } Animation *asset_getAnim(AssetManager *assetManager, char *key) { u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(assetManager->anims)); Animation *result = &assetManager->anims[hashIndex]; if (result->key) { while (result && common_strcmp(result->key, key) != 0) result = result->next; } else { return NULL; } return result; } const i32 asset_loadVorbis(AssetManager *assetManager, MemoryArena *arena, const char *const path, const enum AudioList type) { // TODO(doyle): Remember to free vorbis file if we remove from memory PlatformFileRead fileRead = {0}; platform_readFileToBuffer(arena, path, &fileRead); i32 error; AudioVorbis audio = {0}; audio.type = type; audio.file = stb_vorbis_open_memory(fileRead.buffer, fileRead.size, &error, NULL); if (!audio.file) { printf("stb_vorbis_open_memory() failed: Error code %d\n", error); platform_closeFileRead(arena, &fileRead); stb_vorbis_close(audio.file); return 0; } audio.info = stb_vorbis_get_info(audio.file); audio.lengthInSamples = stb_vorbis_stream_length_in_samples(audio.file); audio.lengthInSeconds = stb_vorbis_stream_length_in_seconds(audio.file); audio.data = CAST(u8 *) fileRead.buffer; audio.size = fileRead.size; assetManager->audio[type] = audio; return 0; } // TODO(doyle): Revise passing in the assetmanager Texture *asset_getAndAllocFreeTexSlot(AssetManager *assetManager, MemoryArena *arena, const char *const key) { i32 texListSize = ARRAY_COUNT(assetManager->textures); u32 hashIndex = common_getHashIndex(key, ARRAY_COUNT(assetManager->textures)); Texture *result = &assetManager->textures[hashIndex]; if (result->key) { while (result->next) { if (common_strcmp(result->key, key) == 0) { // TODO(doyle): Error correction whereby if a tex atlas already // exists ASSERT(INVALID_CODE_PATH); } result = result->next; } result->next = PLATFORM_MEM_ALLOC(arena, 1, Texture); result = result->next; } return result; } const i32 asset_loadTextureImage(AssetManager *assetManager, MemoryArena *arena, const char *const path, const char *const key) { /* Open the texture image */ i32 imgWidth, imgHeight, bytesPerPixel; stbi_set_flip_vertically_on_load(TRUE); u8 *image = stbi_load(path, &imgWidth, &imgHeight, &bytesPerPixel, 0); #ifdef DENGINE_DEBUG if (imgWidth != imgHeight) { printf( "asset_loadTextureImage() warning: Sprite sheet is not square: " "%dx%dpx\n", imgWidth, imgHeight); } #endif if (!image) { printf("stdbi_load() failed: %s\n", stbi_failure_reason()); return -1; } Texture *tex = asset_getAndAllocFreeTexSlot(assetManager, arena, key); *tex = texture_gen(CAST(GLuint)(imgWidth), CAST(GLuint)(imgHeight), CAST(GLint)(bytesPerPixel), image); GL_CHECK_ERROR(); stbi_image_free(image); return 0; } Shader *asset_getShader(AssetManager *assetManager, const enum ShaderList type) { if (type < shaderlist_count) return &assetManager->shaders[type]; #ifdef DENGINE_DEBUG ASSERT(INVALID_CODE_PATH); #endif return NULL; } INTERNAL GLuint createShaderFromPath(MemoryArena *arena, const char *const path, GLuint shadertype) { PlatformFileRead file = {0}; i32 status = platform_readFileToBuffer(arena, path, &file); if (status) return status; const GLchar *source = CAST(char *)file.buffer; GLuint result = glCreateShader(shadertype); glShaderSource(result, 1, &source, NULL); glCompileShader(result); GLint success; GLchar infoLog[512]; glGetShaderiv(result, GL_COMPILE_STATUS, &success); if (!success) { glGetShaderInfoLog(result, 512, NULL, infoLog); printf("glCompileShader() failed: %s\n", infoLog); } platform_closeFileRead(arena, &file); return result; } INTERNAL i32 shaderLoadProgram(Shader *const shader, const GLuint vertexShader, const GLuint fragmentShader) { shader->id = glCreateProgram(); glAttachShader(shader->id, vertexShader); glAttachShader(shader->id, fragmentShader); glLinkProgram(shader->id); glDeleteShader(fragmentShader); glDeleteShader(vertexShader); GLint success; GLchar infoLog[512]; glGetProgramiv(shader->id, GL_LINK_STATUS, &success); if (!success) { glGetProgramInfoLog(shader->id, 512, NULL, infoLog); printf("glLinkProgram failed: %s\n", infoLog); return -1; } return 0; } const i32 asset_loadShaderFiles(AssetManager *assetManager, MemoryArena *arena, const char *const vertexPath, const char *const fragmentPath, const enum ShaderList type) { GLuint vertexShader = createShaderFromPath(arena, vertexPath, GL_VERTEX_SHADER); GLuint fragmentShader = createShaderFromPath(arena, fragmentPath, GL_FRAGMENT_SHADER); Shader shader; i32 result = shaderLoadProgram(&shader, vertexShader, fragmentShader); if (result) return result; assetManager->shaders[type] = shader; return 0; } /* Individual glyph bitmap generated from STB used for creating a font sheet */ typedef struct GlyphBitmap { v2 dimensions; u32 *pixels; i32 codepoint; } GlyphBitmap; const i32 asset_loadTTFont(AssetManager *assetManager, MemoryArena *arena, const char *filePath) { PlatformFileRead fontFileRead = {0}; i32 result = platform_readFileToBuffer(arena, filePath, &fontFileRead); if (result) return result; stbtt_fontinfo fontInfo = {0}; stbtt_InitFont(&fontInfo, fontFileRead.buffer, stbtt_GetFontOffsetForIndex(fontFileRead.buffer, 0)); /* Initialise Assetmanager Font */ Font *font = &assetManager->font; font->codepointRange = V2i(32, 127); v2 codepointRange = font->codepointRange; const i32 numGlyphs = CAST(i32)(codepointRange.y - codepointRange.x); GlyphBitmap *glyphBitmaps = PLATFORM_MEM_ALLOC(arena, numGlyphs, GlyphBitmap); v2 largestGlyphDimension = V2(0, 0); const f32 targetFontHeight = 15.0f; f32 scaleY = stbtt_ScaleForPixelHeight(&fontInfo, targetFontHeight); i32 ascent, descent, lineGap; stbtt_GetFontVMetrics(&fontInfo, &ascent, &descent, &lineGap); ascent = CAST(i32)(ascent * scaleY); descent = CAST(i32)(descent * scaleY); lineGap = CAST(i32)(lineGap * scaleY); font->metrics = CAST(FontMetrics){ascent, descent, lineGap}; font->charMetrics = PLATFORM_MEM_ALLOC(arena, numGlyphs, CharMetrics); /* Use STB_TrueType to generate a series of bitmap characters */ i32 glyphIndex = 0; for (i32 codepoint = CAST(i32) codepointRange.x; codepoint < CAST(i32) codepointRange.y; codepoint++) { // NOTE(doyle): ScaleX if not specified, is then calculated based on the // ScaleY component i32 width, height, xOffset, yOffset; u8 *const monoBitmap = stbtt_GetCodepointBitmap(&fontInfo, 0, scaleY, codepoint, &width, &height, &xOffset, &yOffset); u8 *source = monoBitmap; u32 *colorBitmap = PLATFORM_MEM_ALLOC(arena, width * height, u32); u32 *dest = colorBitmap; // NOTE(doyle): STB generates 1 byte per pixel bitmaps, we use 4bpp, so // duplicate the alpha byte (essentially) for all RGBA components for (i32 y = 0; y < height; y++) { for (i32 x = 0; x < width; x++) { u8 monoByte = *source++; *dest++ = (monoByte << 24) | (monoByte << 16) | (monoByte << 8) | (monoByte << 0); } } /* Get individual character metrics */ i32 advance, leftSideBearing; stbtt_GetCodepointHMetrics(&fontInfo, codepoint, &advance, &leftSideBearing); advance = CAST(i32)(advance * scaleY); leftSideBearing = CAST(i32)(leftSideBearing * scaleY); font->charMetrics[glyphIndex] = CAST(CharMetrics){advance, leftSideBearing, NULL, V2i(xOffset, yOffset), V2i(width, height)}; /* Store bitmap into intermediate storage */ stbtt_FreeBitmap(monoBitmap, NULL); // TODO(doyle): Dimensions is used twice in font->trueSize and this glyphBitmaps[glyphIndex].dimensions = V2i(width, height); glyphBitmaps[glyphIndex].codepoint = codepoint; glyphBitmaps[glyphIndex++].pixels = colorBitmap; if (height > CAST(f32)largestGlyphDimension.h) largestGlyphDimension.h = CAST(f32)height; if (width > CAST(f32)largestGlyphDimension.w) largestGlyphDimension.w = CAST(f32)width; #ifdef DENGINE_DEBUG if ((largestGlyphDimension.h - CAST(i32)targetFontHeight) >= 50) { printf( "asset_loadTTFont() warning: The loaded font file has a glyph " "considerably larger than our target .. font packing is " "unoptimal\n"); } #endif } /* * NOTE(doyle): Use rasterised TTF bitmap-characters combine them all into * one bitmap as an atlas. We determine how many glyphs we can fit per row * by determining the largest glyph size we have. * * For the amount of glyphs we fit per row, we iterate through them and * write each row of the glyph adjacent to the next until we finish writing * all pixels for the glyphs, then move onto the next set of glyphs. */ font->maxSize = largestGlyphDimension; i32 glyphsPerRow = (MAX_TEXTURE_SIZE / CAST(i32)font->maxSize.w); #ifdef DENGINE_DEBUG i32 glyphsPerCol = MAX_TEXTURE_SIZE / CAST(i32)font->maxSize.h; if ((glyphsPerRow * glyphsPerCol) <= numGlyphs) { printf( "asset_loadTTFont() warning: The target font height creates a " "glyph sheet that exceeds the available space!"); ASSERT(INVALID_CODE_PATH); } #endif i32 bitmapSize = SQUARED(TARGET_TEXTURE_SIZE) * TARGET_BYTES_PER_PIXEL; u32 *fontBitmap = PLATFORM_MEM_ALLOC(arena, bitmapSize, u32); const i32 pitch = MAX_TEXTURE_SIZE * TARGET_BYTES_PER_PIXEL; // Check value to determine when a row of glyphs is completely printed i32 verticalPixelsBlitted = 0; i32 startingGlyphIndex = 0; i32 glyphsRemaining = numGlyphs; i32 glyphsOnCurrRow = (glyphsPerRow < glyphsRemaining) ? glyphsPerRow : glyphsRemaining; // TODO(doyle): We copy over the bitmap direct to the font sheet, should we // align the baselines up so we don't need to do baseline adjusting at // render? char charToEncode = CAST(char)codepointRange.x; TexAtlas *fontAtlas = asset_makeTexAtlas(assetManager, arena, "font"); for (i32 row = 0; row < MAX_TEXTURE_SIZE; row++) { u32 *destRow = fontBitmap + (row * MAX_TEXTURE_SIZE); for (i32 glyphIndex = 0; glyphIndex < glyphsOnCurrRow; glyphIndex++) { i32 activeGlyphIndex = startingGlyphIndex + glyphIndex; GlyphBitmap activeGlyph = glyphBitmaps[activeGlyphIndex]; /* Store the location of glyph into atlas */ if (verticalPixelsBlitted == 0) { #ifdef DENGINE_DEBUG ASSERT(activeGlyph.codepoint < ARRAY_COUNT(fontAtlas->subTex)); #endif v2 origin = V2(CAST(f32)(glyphIndex * font->maxSize.w), CAST(f32) row); // NOTE(doyle): Since charToEncode starts from 0 and we record // all ascii characters, charToEncode represents the character // 1:1 char charTmp[2] = {0}; charTmp[0] = charToEncode; AtlasSubTexture *subTex = getFreeAtlasSubTexSlot(fontAtlas, arena, charTmp); subTex->key = PLATFORM_MEM_ALLOC(arena, 1, char); subTex->key[0] = charToEncode; subTex->rect = CAST(Rect){origin, font->maxSize}; charToEncode++; } /* Copy over exactly one row of pixels */ i32 numPixelsToPad = CAST(i32)font->maxSize.w; if (verticalPixelsBlitted < activeGlyph.dimensions.h) { const i32 srcPitch = CAST(i32) activeGlyph.dimensions.w * verticalPixelsBlitted; const u32 *src = activeGlyph.pixels + srcPitch; const i32 numPixelsToCopy = CAST(i32)activeGlyph.dimensions.w; for (i32 count = 0; count < numPixelsToCopy; count++) *destRow++ = *src++; /* * NOTE(doyle): If the glyph is smaller than largest glyph * available size, don't advance src pointer any further * (NULL/mixes up rows), instead just advance the final bitmap * pointer by the remaining distance */ numPixelsToPad = CAST(i32)(font->maxSize.w - activeGlyph.dimensions.w); } destRow += numPixelsToPad; } /* A row of glyphs has been fully formed on the atlas */ if (verticalPixelsBlitted++ >= font->maxSize.h) { verticalPixelsBlitted = 0; startingGlyphIndex += glyphsPerRow; glyphsRemaining -= glyphsPerRow; if (glyphsRemaining <= 0) break; else if (glyphsRemaining <= glyphsPerRow) { // NOTE(doyle): This allows us to modify the glyph iterator to // prevent over-running of the available glyphs glyphsOnCurrRow = glyphsRemaining; } } } Texture tex = texture_gen(MAX_TEXTURE_SIZE, MAX_TEXTURE_SIZE, 4, CAST(u8 *) fontBitmap); assetManager->textures[texlist_font] = tex; #ifdef WT_RENDER_FONT_FILE /* save out a 4 channel image */ stbi_write_png("out.png", MAX_TEXTURE_SIZE, MAX_TEXTURE_SIZE, 4, fontBitmap, MAX_TEXTURE_SIZE * 4); #endif PLATFORM_MEM_FREE(arena, fontBitmap, bitmapSize); fontAtlas->tex = &assetManager->textures[texlist_font]; font->atlas = fontAtlas; // NOTE(doyle): Formula derived from STB Font font->verticalSpacing = font->metrics.ascent - font->metrics.descent + font->metrics.lineGap; for (i32 i = 0; i < numGlyphs; i++) { i32 glyphBitmapSizeInBytes = CAST(i32) glyphBitmaps[i].dimensions.w * CAST(i32) glyphBitmaps[i].dimensions.h * sizeof(u32); PLATFORM_MEM_FREE(arena, glyphBitmaps[i].pixels, glyphBitmapSizeInBytes); } PLATFORM_MEM_FREE(arena, glyphBitmaps, numGlyphs * sizeof(GlyphBitmap)); platform_closeFileRead(arena, &fontFileRead); return 0; } void asset_addAnimation(AssetManager *assetManager, MemoryArena *arena, char *animName, TexAtlas *atlas, char **subTextureNames, i32 numSubTextures, f32 frameDuration) { Animation *anim = getFreeAnimationSlot( assetManager->anims, ARRAY_COUNT(assetManager->anims), arena, animName); anim->atlas = atlas; anim->frameDuration = frameDuration; anim->numFrames = numSubTextures; // NOTE(doyle): +1 for the null terminator anim->name = PLATFORM_MEM_ALLOC(arena, common_strlen(animName) + 1, char); common_strncpy(anim->name, animName, common_strlen(animName)); anim->frameList = PLATFORM_MEM_ALLOC(arena, numSubTextures, char*); for (i32 i = 0; i < numSubTextures; i++) { AtlasSubTexture *subTex = getAtlasSubTex(atlas, subTextureNames[i]); anim->frameList[i] = subTex->key; } } v2 asset_stringDimInPixels(const Font *const font, const char *const string) { v2 stringDim = V2(0, 0); for (i32 i = 0; i < common_strlen(string); i++) { i32 codepoint = string[i]; #ifdef DENGINE_DEBUG ASSERT(codepoint >= font->codepointRange.x && codepoint <= font->codepointRange.y) #endif i32 relativeIndex = CAST(i32)(codepoint - font->codepointRange.x); v2 charDim = font->charMetrics[relativeIndex].trueSize; stringDim.x += charDim.x; stringDim.y = (charDim.y > stringDim.y) ? charDim.y : stringDim.y; } return stringDim; } void asset_unitTest(MemoryArena *arena) { PlatformFileRead xmlFileRead = {0}; i32 result = platform_readFileToBuffer( arena, "data/textures/WorldTraveller/ClaudeSprite.xml", &xmlFileRead); if (result) { DEBUG_LOG( "unitTest() error: Could not load XML file for memory free test"); } else { /* Tokenise buffer */ i32 memBefore = arena->bytesAllocated; i32 numTokens = 0; XmlToken *xmlTokens = tokeniseXmlBuffer(arena, xmlFileRead.buffer, xmlFileRead.size, &numTokens); /* Build XML tree from tokens */ XmlNode *xmlTree = buildXmlTree(arena, xmlTokens, numTokens); freeXmlData(arena, xmlTokens, numTokens, xmlTree); i32 memAfter = arena->bytesAllocated; ASSERT(memBefore == memAfter); } }