raylib_loadbmfont_extender
raylib loadbmfont extender
문서의 이전 판입니다!
Purpose
Raylib's internal Bitmap Font loader support only one font atlas.
So if you want import large amount of font glyphs - like hangeul(korean language) - you must make font atlas resolustion too big .For instance, 3096 X 3096 resolution will be needed. But it's not convenient.
That is because raylib's internal LoadBMFont() method is not support pages parameter which is in FNT file.
So I modified that.
This is my modified font library
LoadBMFontEx library
- dkfont.c
#include <stdlib.h> #include "raylib.h" #include "string.h" static Font LoadBMFontEX(const char *fileName); static int GetLine(const char *origin, char *buffer, int maxLength); // Load a BMFont file (AngelCode font file) // REQUIRES: strstr(), sscanf(), strrchr(), memcpy() Font LoadBMFontEX(const char *fileName) { #define MAX_BUFFER_SIZE 256 Font font = { 0 }; char buffer[MAX_BUFFER_SIZE] = { 0 }; char *searchPoint = NULL; int fontSize = 0; int glyphCount = 0; int imWidth = 0; int imHeight = 0; int totalPage = 1; int curPage = 0; char imFileName[10][129] = { 0 }; int base = 0; // Useless data char *fileText = LoadFileText(fileName); if (fileText == NULL) return font; char *fileTextPtr = fileText; // NOTE: We skip first line, it contains no useful information int lineBytes = GetLine(fileTextPtr, buffer, MAX_BUFFER_SIZE); fileTextPtr += (lineBytes + 1); // Read line data lineBytes = GetLine(fileTextPtr, buffer, MAX_BUFFER_SIZE); searchPoint = strstr(buffer, "lineHeight"); sscanf(searchPoint, "lineHeight=%i base=%i scaleW=%i scaleH=%i pages=%i", &fontSize, &base, &imWidth, &imHeight, &totalPage); fileTextPtr += (lineBytes + 1); // This is just for debugging printf("FONT: [%s] Loaded font info:\n", fileName); printf(" > Base size: %i\n", fontSize); printf(" > Texture scale: %ix%i\n", imWidth, imHeight); for (int i = 0; i < totalPage; i++) { lineBytes = GetLine(fileTextPtr, buffer, MAX_BUFFER_SIZE); searchPoint = strstr(buffer, "id"); sscanf(searchPoint, "id=%i file=\"%128[^\"]\"", &curPage, imFileName[i]); // This is just for debugging printf(" > Texture filename: %s\n", imFileName[i]); fileTextPtr += (lineBytes + 1); } lineBytes = GetLine(fileTextPtr, buffer, MAX_BUFFER_SIZE); searchPoint = strstr(buffer, "count"); sscanf(searchPoint, "count=%i", &glyphCount); fileTextPtr += (lineBytes + 1); // This is just for debugging printf(" > Chars count: %i\n", glyphCount); // Compose correct path using route of .fnt file (fileName) and imFileName char *imPath[10] = {0}; char *lastSlash = NULL; for (int i = 0; i< totalPage; i++) { lastSlash = strrchr(fileName, '/'); if (lastSlash == NULL) lastSlash = strrchr(fileName, '\\'); if (lastSlash != NULL) { // NOTE: We need some extra space to avoid memory corruption on next allocations! imPath[i] = (char *)RL_CALLOC(TextLength(fileName) - TextLength(lastSlash) + TextLength(imFileName[i]) + 4, 1); memcpy(imPath[i], fileName, TextLength(fileName) - TextLength(lastSlash) + 1); memcpy(imPath[i] + TextLength(fileName) - TextLength(lastSlash) + 1, imFileName[i], TextLength(imFileName[i])); } else imPath[i] = imFileName[i]; // For debug printf(" > Image loading path: %s\n", imPath[i]); // TRACELOGD(" > Image loading path: %s", imPath); } Image imFont[10]; for (int i = 0; i < totalPage; i++) { imFont[i] = LoadImage(imPath[i]); // For debug printf("Image [%i] width : %i\n", i, imFont[i].width); if (imFont[i].format == PIXELFORMAT_UNCOMPRESSED_GRAYSCALE) { // Convert image to GRAYSCALE + ALPHA, using the mask as the alpha channel Image imFontAlpha = { .data = RL_CALLOC(imFont[i].width*imFont[i].height, 2), .width = imFont[i].width, .height = imFont[i].height, .mipmaps = 1, .format = PIXELFORMAT_UNCOMPRESSED_GRAY_ALPHA }; for (int p = 0, pi = 0; p < (imFont[i].width*imFont[i].height*2); p += 2, pi++) { ((unsigned char *)(imFontAlpha.data))[p] = 0xff; ((unsigned char *)(imFontAlpha.data))[p + 1] = ((unsigned char *)imFont[i].data)[pi]; } UnloadImage(imFont[i]); imFont[i] = imFontAlpha; } if (lastSlash != NULL) RL_FREE(imPath[i]); } // Resize and ReDraw Font Image Image fullFont; fullFont = imFont[0]; ImageResizeCanvas(&fullFont, imWidth, imHeight * totalPage, 0, 0, BLACK); for (int index = 1; index <= totalPage; index++) { ImageDraw(&fullFont, imFont[index], Rectangle{ 0, 0, float(imWidth), float(imHeight)}, Rectangle{ 0, float(imHeight) * index,float(imWidth), float(imHeight)}, WHITE); } // Image to Texure font.texture = LoadTextureFromImage(fullFont); // Fill font characters info data font.baseSize = fontSize; font.glyphCount = glyphCount; font.glyphPadding = 0; font.glyphs = (GlyphInfo *)RL_MALLOC(glyphCount*sizeof(GlyphInfo)); font.recs = (Rectangle *)RL_MALLOC(glyphCount*sizeof(Rectangle)); int charId, charX, charY, charWidth, charHeight, charOffsetX, charOffsetY, charAdvanceX, pageID; for (int i = 0; i < glyphCount; i++) { lineBytes = GetLine(fileTextPtr, buffer, MAX_BUFFER_SIZE); sscanf(buffer, "char id=%i x=%i y=%i width=%i height=%i xoffset=%i yoffset=%i xadvance=%i page=%i", &charId, &charX, &charY, &charWidth, &charHeight, &charOffsetX, &charOffsetY, &charAdvanceX, &pageID); fileTextPtr += (lineBytes + 1); // Get character rectangle in the font atlas texture font.recs[i] = (Rectangle){ (float)charX, (float)charY + 1024 * pageID, (float)charWidth, (float)charHeight }; // Save data properly in sprite font font.glyphs[i].value = charId; font.glyphs[i].offsetX = charOffsetX; font.glyphs[i].offsetY = charOffsetY; font.glyphs[i].advanceX = charAdvanceX; // Fill character image data from imFont data font.glyphs[i].image = ImageFromImage(fullFont, font.recs[i]); } // Unload UnloadImage(fullFont); UnloadFileText(fileText); if (font.texture.id == 0) { UnloadFont(font); font = GetFontDefault(); // TRACELOG(LOG_WARNING, "FONT: [%s] Failed to load texture, reverted to default font", fileName); printf("FONT: [%s] Failed to load texture, reverted to default font\n", fileName); } else // TRACELOG(LOG_INFO, "FONT: [%s] Font loaded successfully (%i glyphs)", fileName, font.glyphCount); printf("FONT: [%s] Font loaded successfully (%i glyphs)", fileName, font.glyphCount); return font; } static int GetLine(const char *origin, char *buffer, int maxLength) { int count = 0; for (; count < maxLength; count++) if (origin[count] == '\n') break; memcpy(buffer, origin, count); return count; }
Usage
Just define font by using “LoadBMFontEx(filename)”.
It support multiple font atlas which is associated with fnt file.
raylib_loadbmfont_extender.1698895135.txt.gz · 마지막으로 수정됨: 2023/11/02 12:18 (바깥 편집)
로그인