/* mkfont.c : Make font pngs for gzdoom, in an ugly cheap way. This code is a mess but I keep it here so people know how much I had to suffer to get this done. Copyright (c) 2020-2022 Marisa the Magician, UnSX Team Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /* Known bugs/limitations: - There is no handling of proportional fonts where glyphs are wider than their advance length. As this isn't supported by GZDoom itself, it's not a priority either. There's no way to encode the "real width" of a glyph into the PNG. Also, glyphs might end up rendered with cropped sides. - Non-bitmap fonts might sometimes spew lots of "out of bounds" drawing warnings. This is mainly due to the peculiarities of antialiasing. - Doesn't handle unicode blocks beyond the basic multilingual plane, might actually be an issue with freetype itself, or the fonts I use, though I CAN see that the glyphs are there if I use a character map program. Not that this doesn't even matter considering GZDoom only supports the 0000-FFFF range anyway. */ #include #include #include #include #include #include FT_FREETYPE_H uint8_t pal[768]; int palsize = 0; int xupshift = 0; int rightshift = 0; int bordshift = 0; uint32_t endianswap( uint32_t n ) { // if we're in a big endian system, we don't need this uint16_t testme = 0x1234; if ( *(uint8_t*)(&testme) == 0x12 ) return n; uint32_t on; for ( int i=0; i<4; i++ ) *(((uint8_t*)(&on))+i) = *(((uint8_t*)(&n))+(3-i)); return on; } int writepng( const char *filename, uint8_t *fdata, int w, int h, int p ) { if ( !filename ) return 0; png_structp pngp; png_infop infp; FILE *pf; if ( !(pf = fopen(filename,"wb")) ) return 0; pngp = png_create_write_struct(PNG_LIBPNG_VER_STRING,0,0,0); if ( !pngp ) { fclose(pf); return 0; } infp = png_create_info_struct(pngp); if ( !infp ) { fclose(pf); png_destroy_write_struct(&pngp,0); return 0; } if ( setjmp(png_jmpbuf(pngp)) ) { png_destroy_write_struct(&pngp,&infp); fclose(pf); return 0; } png_init_io(pngp,pf); png_set_IHDR(pngp,infp,w,h,8,palsize?PNG_COLOR_TYPE_RGBA:PNG_COLOR_TYPE_GA, PNG_INTERLACE_NONE,PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT); if ( xupshift || rightshift || bordshift ) { uint32_t grabs[2] = {endianswap(rightshift+bordshift),endianswap(xupshift+bordshift)}; png_unknown_chunk grab = { .name = "grAb", .data = (uint8_t*)grabs, .size = 8, .location = PNG_HAVE_IHDR }; png_set_unknown_chunks(pngp,infp,&grab,1); } png_write_info(pngp,infp); for ( int i=0; i 255 ) alph = 255; idata[tpos+1] = alph; // blend color int col = idata[tpos]*(a-255); col += v*a; col /= 255; idata[tpos] = col; } void putpixel_color( uint8_t v, uint8_t a, int x, int y ) { uint32_t tpos = (x+y*iw)*4; // add alpha int alph = idata[tpos+3]; alph += a; if ( alph > 255 ) alph = 255; idata[tpos+3] = alph; // blend color (RGB) for ( int i=0; i<3; i++ ) { int col = idata[tpos+i]*(a-255); int palent = (v*palsize)/256; col += pal[palent*3+i]*a; col /= 255; idata[tpos+i] = col; } } int oob = 0; void putpixel( uint8_t v, uint8_t a, int x, int y ) { if ( (x < 0) || (x >= iw) || (y < 0) || (y >= ih) ) { oob = 1; return; } if ( palsize == 0 ) putpixel_grayscale(v,a,x,y); else putpixel_color(v,a,x,y); } uint8_t lerpg( float a ) { if ( a >= 1. ) return 255; if ( a <= 0. ) return 64; return (uint8_t)(a*191+64); } int draw_glyph( FT_Bitmap *bmp, uint8_t v, int px, int py, int ox, int oy ) { if ( !bmp->buffer ) return 0; int drawn = 0; unsigned i, j; for ( j=0; jrows; j++ ) { uint8_t rv = v; // apply gradient, if any if ( v == 255 ) { float a; int ofs = j+oy+(upshift+1); if ( ofs < 0 ) a = 0.; else a = ofs/(float)(h+upshift+1); if ( (gradient&3) == 1 ) rv = lerpg(1.-a); else if ( (gradient&3) == 2 ) rv = lerpg(a); else if ( (gradient&3) == 3 ) rv = lerpg((a>.5)?((1.-a)*2.):(a*2.)); } for ( i=0; iwidth; i++ ) { if ( bmp->pixel_mode == FT_PIXEL_MODE_GRAY ) { uint8_t a = bmp->buffer[i+j*bmp->pitch]; drawn |= (a > 0); putpixel(rv,a,px+ox+i,py+oy+j); } else if ( bmp->pixel_mode == FT_PIXEL_MODE_MONO ) { // thanks to https://stackoverflow.com/a/14905971 unsigned p = bmp->pitch; uint8_t *row = &bmp->buffer[p*j]; uint8_t a = ((row[i>>3])&(128>>(i&7)))?255:0; drawn |= (a > 0); putpixel(rv,a,px+ox+i,py+oy+j); } } } return drawn; } int valid_row( FT_Bitmap *bmp, unsigned j ) { if ( !bmp->buffer ) return 0; int drawn = 0; for ( unsigned i=0; iwidth; i++ ) { if ( bmp->pixel_mode == FT_PIXEL_MODE_GRAY ) { uint8_t a = bmp->buffer[i+j*bmp->pitch]; drawn |= (a > 0); } else if ( bmp->pixel_mode == FT_PIXEL_MODE_MONO ) { // thanks to https://stackoverflow.com/a/14905971 unsigned p = bmp->pitch; uint8_t *row = &bmp->buffer[p*j]; uint8_t a = ((row[i>>3])&(128>>(i&7)))?255:0; drawn |= (a > 0); } } return drawn; } unsigned row_width( FT_Bitmap *bmp, unsigned j ) { if ( !bmp->buffer ) return 0; unsigned dw = 0; for ( unsigned i=0; iwidth; i++ ) { if ( bmp->pixel_mode == FT_PIXEL_MODE_GRAY ) { uint8_t a = bmp->buffer[i+j*bmp->pitch]; if ( (a > 0) && (i > dw) ) dw = i; } else if ( bmp->pixel_mode == FT_PIXEL_MODE_MONO ) { // thanks to https://stackoverflow.com/a/14905971 unsigned p = bmp->pitch; uint8_t *row = &bmp->buffer[p*j]; uint8_t a = ((row[i>>3])&(128>>(i&7)))?255:0; if ( (a > 0) && (i > dw) ) dw = i; } } return dw; } int palinv = 0; void loadpalette( const char *path ) { FILE *f = fopen(path,"rb"); if ( !f ) { fprintf(stderr,"warning: could not open palette file '%s', falling back to grayscale\n",path); return; } fseek(f,0,SEEK_END); long sz = ftell(f); fseek(f,0,SEEK_SET); if ( sz <= 0 ) { fprintf(stderr,"warning: palette is empty, falling back to grayscale\n"); goto palend; } if ( !(sz%3) ) { // RGB8 palette if ( sz > 768 ) { fprintf(stderr,"warning: palette has more than 256 entries, extra colors will be ignored\n"); palsize = 256; } else palsize = sz/3; for ( int i=0; i 1024 ) { fprintf(stderr,"warning: palette has more than 256 entries, extra colors will be ignored\n"); palsize = 256; } else palsize = sz/4; for ( int i=0; i " " [gradient type] [color palette] [-palinv]\n"); return 1; } if ( FT_Init_FreeType(&ftlib) ) { fprintf(stderr,"error: failed to init freetype library\n"); return 2; } uint32_t range[2] = {0x0000,0x00FF}; sscanf(argv[2],"%d",&pxsiz); if ( !strcmp(argv[3],"auto") ) autosize = 1; else sscanf(argv[3],"%dx%d",&w,&h); sscanf(argv[4],"%x-%x",&range[0],&range[1]); if ( argc > 5 ) sscanf(argv[5],"%d",&gradient); if ( argc > 7 ) palinv = !strcmp(argv[7],"-palinv"); if ( argc > 6 ) loadpalette(argv[6]); if ( FT_New_Face(ftlib,argv[1],0,&fnt) ) { fprintf(stderr,"error: failed to open font '%s'\n",argv[1]); return 4; } if ( FT_Set_Pixel_Sizes(fnt,0,pxsiz) ) { if ( fnt->num_fixed_sizes <= 0 ) { fprintf(stderr,"error: font pixel size of '%d' not available.\n",pxsiz); return 8; } fprintf(stderr,"warning: failed to set pixel size '%d', trying fixed sizes.\n",pxsiz); int match = -1; for ( int i=0; inum_fixed_sizes; i++ ) { if ( fnt->available_sizes[i].height == pxsiz ) continue; match = i; break; } if ( (match == -1) || FT_Select_Size(fnt,match) ) { fprintf(stderr,"error: font fixed size of '%d' not available\n",pxsiz); fprintf(stderr,"available size(s): "); for ( int i=0; inum_fixed_sizes; i++ ) fprintf(stderr,"%u%s",fnt->available_sizes[i].height,(i<(fnt->num_fixed_sizes-1))?", ":".\n"); return 8; } } FT_Select_Charmap(fnt,FT_ENCODING_UNICODE); // first pass to compute baseline upshift upshift = 65535; for ( uint32_t i=range[0]; i<=range[1]; i++ ) { FT_UInt glyph = FT_Get_Char_Index(fnt,i); if ( !glyph || FT_Load_Glyph(fnt,glyph,LOADFLAGS) ) continue; FT_Render_Glyph(fnt->glyph,RENDERMODE); int valid = 0; for ( unsigned j=0; jglyph->bitmap.rows; j++ ) valid |= valid_row(&fnt->glyph->bitmap,j); if ( !valid ) continue; int gshift = (fnt->glyph->metrics.horiBearingY-fnt->glyph->metrics.height)>>6; if ( gshift < upshift ) upshift = gshift; } fprintf(stderr,"info: estimated baseline upshift is %d.\n",upshift); // second pass to compute "real" upshift, which is used for the grAb Y offset // as well as the "rightshift" for the X offset xupshift = 65535; for ( uint32_t i=range[0]; i<=range[1]; i++ ) { FT_UInt glyph = FT_Get_Char_Index(fnt,i); if ( !glyph || FT_Load_Glyph(fnt,glyph,LOADFLAGS) ) continue; FT_Render_Glyph(fnt->glyph,RENDERMODE); int valid = 0; for ( unsigned j=0; jglyph->bitmap.rows; j++ ) valid |= valid_row(&fnt->glyph->bitmap,j); if ( !valid ) continue; int xx = fnt->glyph->bitmap_left; if ( -xx > rightshift ) rightshift = -xx; int yy = upshift+1+(pxsiz-fnt->glyph->bitmap_top); for ( unsigned j=0; jglyph->bitmap.rows; j++ ) { if ( !valid_row(&fnt->glyph->bitmap,j) ) yy++; else break; } if ( yy < 0 ) { int xup = -yy; if ( xup < xupshift ) xupshift = xup; } } if ( xupshift == 65535 ) xupshift = 0; if ( xupshift ) { fprintf(stderr,"info: real upshift is %d.\n",upshift+xupshift); fprintf(stderr,"info: grAb Y offset %d will be used.\n",xupshift); } if ( rightshift ) fprintf(stderr,"info: right shift detected, grAb X offset %d will be used.\n",rightshift); // third pass to compute the maximum cell size (for memory allocation) if ( autosize ) { h = fnt->size->metrics.height>>6; if ( FT_IS_FIXED_WIDTH(fnt) ) { w = fnt->size->metrics.max_advance>>6; autosize = 0; // don't readjust per-glyph } else for ( uint32_t i=range[0]; i<=range[1]; i++ ) { FT_UInt glyph = FT_Get_Char_Index(fnt,i); if ( !glyph || FT_Load_Glyph(fnt,glyph,LOADFLAGS) ) continue; FT_Render_Glyph(fnt->glyph,RENDERMODE); int adv = fnt->glyph->linearHoriAdvance>>16; int gw = fnt->glyph->bitmap.width+fnt->glyph->bitmap_left; if ( gw > adv ) adv = gw; if ( (adv+rightshift) > w ) w = adv+rightshift; } fprintf(stderr,"info: max cell size is %dx%d.\n",w,h); } iw = w+1; ih = h+1; if ( gradient&4 ) { bordshift = 1; iw++; ih++; } datsiz = palsize?(iw*ih*4):(iw*ih*2); fprintf(stderr,"info: gradient selected is '%s'.\n",grads[gradient&7]); idata = calloc(datsiz,1); uint32_t drawn = 0; for ( uint32_t i=range[0]; i<=range[1]; i++ ) { FT_UInt glyph = FT_Get_Char_Index(fnt,i); if ( !glyph || FT_Load_Glyph(fnt,glyph,LOADFLAGS) ) continue; FT_Render_Glyph(fnt->glyph,RENDERMODE); int valid = 0; for ( unsigned j=0; jglyph->bitmap.rows; j++ ) valid |= valid_row(&fnt->glyph->bitmap,j); if ( !valid ) continue; if ( autosize ) { // readjust cell width (but not height) for this character int adv = fnt->glyph->linearHoriAdvance>>16; w = adv+rightshift; iw = w+((gradient&4)?2:1); } int xx = rightshift; int yy = upshift+xupshift+1; valid = 0; oob = 0; if ( gradient&4 ) { // draw outline first draw_glyph(&fnt->glyph->bitmap,0,xx,yy,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx+1,yy,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx+2,yy,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx,yy+1,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx+2,yy+1,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx,yy+2,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx+1,yy+2,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); draw_glyph(&fnt->glyph->bitmap,0,xx+2,yy+2,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); valid = draw_glyph(&fnt->glyph->bitmap,255,xx+1,yy+1,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); } else { // draw drop shadow first draw_glyph(&fnt->glyph->bitmap,0,xx+1,yy+1,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); valid = draw_glyph(&fnt->glyph->bitmap,255,xx,yy,fnt->glyph->bitmap_left,pxsiz-fnt->glyph->bitmap_top); } if ( valid ) { if ( oob ) fprintf(stderr,"warning: glyph %04X drawn out of bounds, cell size may be incorrect.\n",i); char fname[256]; snprintf(fname,256,"%04X.png",i); writepng(fname,idata,iw,ih,palsize?(iw*4):(iw*2)); drawn++; } memset(idata,0,datsiz); } fprintf(stderr,"info: %u glyphs drawn.\n",drawn); free(idata); FT_Done_Face(fnt); FT_Done_FreeType(ftlib); return 0; }