/* * Descent 3 * Copyright (C) 2024 Parallax Software * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . --- HISTORICAL COMMENTS FOLLOW --- * $Logfile: /DescentIII/Main/2dlib/font.cpp $ * $Revision: 15 $ * $Date: 4/17/99 6:15p $ * $Author: Samir $ * * will load and manage a font's capabilities. * * $Log: /DescentIII/Main/2dlib/font.cpp $ * * 15 4/17/99 6:15p Samir * replaced gr.h with grdefs.h and fixed resulting compile bugs. * * 14 10/13/98 10:31a Sean * Fixed a mem_free (free()) conflict * * 13 4/23/98 6:38p Jason * made bitmaps use 1555 format * * 12 3/31/98 3:48p Jason * added memory lib * * 11 12/19/97 12:31p Samir * load font registration * * 10 12/12/97 6:40p Samir * Some viewport font functions. * * 9 11/16/97 6:53p Samir * Reinstated instance count for fonts. * * 8 11/16/97 4:41p Matt * Commented-out an assert. * * 7 11/14/97 12:06p Samir * When you use a font for the first time, load it. Never free font until * game closes. * * 6 10/15/97 5:20p Jason * did a HUGE overhaul of the bitmap system * * 5 9/26/97 2:53p Samir * Delete old m_CharSurf if creating a new one. * * 4 9/19/97 5:37p Samir * Software font coloring works. * * 3 8/07/97 3:15p Samir * May have fixed horizontal left clipping problem. * * 22 6/25/97 10:10a Samir * Removed unused code causing warning. * * 21 6/24/97 7:42p Matt * Included bitmap.h instead of font.h * * 20 6/16/97 4:45p Samir * Fixed transparency probs * * 19 6/16/97 3:01p Samir * Changed font system to work better with renderer. Fonts are rendered * on bitmaps of 128x128 at load time. * * 18 6/12/97 6:27p Samir * DDGR v2.0 (GR v1.1) Changes to ddgr system reflected in 2d system * * 17 5/14/97 8:07p Samir * some pathname problem which screwed up Mac. * * 16 3/07/97 4:01p Samir * Took out malloc.h for ANSI compliance. * * 15 3/03/97 6:20p Matt * Changed cfile functions to use D3 naming convention * * 14 2/28/97 3:17 PM Jeremy * changed a call of strcmpi to stricmp * * 13 2/28/97 2:57p Samir * Took out __cdecl from atexit * * 12 2/10/97 12:14p Samir * Allow multiple font loading per viewport. * * 11 2/06/97 6:33p Samir * Added clipping character functions for font. * * 10 2/06/97 12:43p Samir * Mono fonts work. * * 9 2/05/97 12:19p Samir * Stopped redundant increases in font reference count for same font * object. * * 8 2/05/97 10:35a Samir * Fixed text drawing at wrong location problem * * $NoKeywords: $ */ #include "pserror.h" #include "renderer.h" #include "gr.h" #include "mono.h" #include "cfile.h" #include "bitmap.h" #include "mem.h" #include #include #define FT_COLOR 1 #define FT_PROPORTIONAL 2 #define FT_KERNED 4 #define FT_GRADIENT 8 #define GRFONT_SURFACE_WIDTH 128 #define GRFONT_SURFACE_HEIGHT 128 // ---------------------------------------------------------------------------- // Macros for file io. // ---------------------------------------------------------------------------- typedef CFILE *FONTFILE; static inline int READ_FONT_INT(FONTFILE ffile); static inline short READ_FONT_SHORT(FONTFILE ffile); static inline ubyte READ_FONT_BYTE(FONTFILE ffile); static inline int READ_FONT_DATA(FONTFILE ffile, void *buf, int size, int nelem); static inline FONTFILE OPEN_FONT(char *filename, bool &success); static inline void CLOSE_FONT(FONTFILE ffile); #define BITS_TO_BYTES(_c) (((_c) + 7) >> 3) #define BITS_TO_SHORTS(_c) (((_c) + 15) >> 4) inline int READ_FONT_INT(FONTFILE ffile) { return cf_ReadInt(ffile); } inline short READ_FONT_SHORT(FONTFILE ffile) { return cf_ReadShort(ffile); } inline ubyte READ_FONT_BYTE(FONTFILE ffile) { return (ubyte)cf_ReadByte(ffile); } inline int READ_FONT_DATA(FONTFILE ffile, void *buf, int size, int nelem) { int i; i = cf_ReadBytes((ubyte *)buf, size * nelem, ffile); ASSERT(i == (size * nelem)); return i; } inline FONTFILE OPEN_FONT(char *filename, bool &success) { FONTFILE fp; int file_id; success = false; fp = (FONTFILE)cfopen(filename, "rb"); if (!fp) return NULL; file_id = READ_FONT_INT(fp); if (file_id != 0xfeedbaba) return (FONTFILE)(-1); success = true; return fp; } inline void CLOSE_FONT(FONTFILE ffile) { cfclose(ffile); } // ---------------------------------------------------------------------------- // static variables and functions // ---------------------------------------------------------------------------- gr_font_record grFont::m_FontList[MAX_FONTS]; grMemorySurface *grFont::m_CharSurf = NULL; void grFont::init_system() { for (int i = 0; i < MAX_FONTS; i++) { grFont::m_FontList[i].references = 0; grFont::m_FontList[i].filename[0] = 0; } if (grFont::m_CharSurf) { delete grFont::m_CharSurf; grFont::m_CharSurf = NULL; } if (!grFont::m_CharSurf) { grFont::m_CharSurf = new grMemorySurface; grFont::m_CharSurf->init(0, 0, BPP_16, NULL, 0, 0, "char_surf"); } atexit(grFont::close_system); } void grFont::close_system() { if (grFont::m_CharSurf) { delete grFont::m_CharSurf; grFont::m_CharSurf = NULL; } // we should free any fonts currently allocated here. mprintf((0, "Freeing cached fonts...")); for (int i = 0; i < MAX_FONTS; i++) { gr_font_record *ft; if (grFont::m_FontList[i].references) { ft = &grFont::m_FontList[i]; grFont::free_font(ft); grFont::m_FontList[i].references = 0; } } } int grFont::register_font(char *fontname, char *filename) { /* look for a slot with no fontname and place fontname in it */ ASSERT(fontname != NULL); int i; for (i = 0; i < MAX_FONTS; i++) { if (grFont::m_FontList[i].filename[0] == 0) { strcpy(grFont::m_FontList[i].filename, filename); strcpy(grFont::m_FontList[i].name, fontname); grFont::load(m_FontList[i].filename, i); grFont::m_FontList[i].references = 1; break; } } if (i == MAX_FONTS) i = -1; return i; } // ---------------------------------------------------------------------------- // constructor and destructor of font object // ---------------------------------------------------------------------------- grFont::grFont() { m_FontHandle = -1; } grFont::~grFont() { if (m_FontHandle > -1) grFont::free(); } // ---------------------------------------------------------------------------- // high level font initialization and cleanup routines // ---------------------------------------------------------------------------- void grFont::init(const char *fontname) { int i, slot = -1; // Look for this font in the list, check if it has any references, and load it if // there are none. if (fontname == NULL) { slot = 0; } else { for (i = 0; i < MAX_FONTS; i++) { if (stricmp(grFont::m_FontList[i].name, fontname) == 0) { slot = i; if (grFont::m_FontList[i].references == 0) { break; } else { m_FontHandle = slot; return; } } } ASSERT(slot != -1); // clear up surface and bitmap lists grFont::m_FontList[slot].ch_w = NULL; grFont::m_FontList[slot].ch_h = NULL; grFont::m_FontList[slot].ch_u = NULL; grFont::m_FontList[slot].ch_v = NULL; for (i = 0; i < MAX_FONT_BITMAPS; i++) { grFont::m_FontList[slot].bmps[i] = -1; grFont::m_FontList[slot].surfs[i] = NULL; } // We must load this font from the 'hogfile' grFont::load(m_FontList[slot].filename, slot); grFont::m_FontList[slot].references++; } m_FontHandle = slot; } void grFont::free() { if (m_FontHandle == -1) return; return; //!! Removed the following assert because there are bugs in the reference count system. Matt, 11-15-97. //!! ASSERT(grFont::m_FontList[m_FontHandle].references > 0); //@@ // free font if there was only one reference left. //@@ if (grFont::m_FontList[m_FontHandle].references == 1) { //@@ gr_font_record *ft; //@@ //@@ // free up all memory allocated to this font. //@@ ft = &grFont::m_FontList[m_FontHandle]; //@@ //@@ grFont::free_font(ft); //@@ } //@@ //@@ grFont::m_FontList[m_FontHandle].references--; m_FontHandle = -1; } void grFont::free_font(gr_font_record *ft) { int i; // delete font surface info. if (ft->ch_wf) delete[] ft->ch_wf; if (ft->ch_hf) delete[] ft->ch_hf; if (ft->ch_uf) delete[] ft->ch_uf; if (ft->ch_vf) delete[] ft->ch_vf; if (ft->ch_w) delete[] ft->ch_w; if (ft->ch_h) delete[] ft->ch_h; if (ft->ch_u) delete[] ft->ch_u; if (ft->ch_v) delete[] ft->ch_v; if (ft->ch_surf) delete[] ft->ch_surf; for (i = 0; i < MAX_FONT_BITMAPS; i++) { if (ft->bmps[i] != -1) bm_FreeBitmap(ft->bmps[i]); if (ft->surfs[i]) delete ft->surfs[i]; } // delete font file info. if (ft->font.flags & FT_PROPORTIONAL) { delete[] ft->font.char_widths; } ::mem_free(ft->font.raw_data); ::mem_free(ft->font.char_data); } void grFont::load(char *filename, int slot) { FONTFILE ff; gr_font_file_record fnt, *ft; char fontname[32]; int num_char; int i; ft = &fnt; bool success; ff = OPEN_FONT(filename, success); if (!ff) { Error("Unable to open font in file %s.\n", filename); } else if (!success) { Error("Illegal font file: %s.\n", filename); } mprintf((0, "%s font.\n", grFont::m_FontList[slot].name)); ft->width = READ_FONT_SHORT(ff); ft->height = READ_FONT_SHORT(ff); ft->flags = READ_FONT_SHORT(ff); ft->baseline = READ_FONT_SHORT(ff); ft->min_ascii = READ_FONT_BYTE(ff); ft->max_ascii = READ_FONT_BYTE(ff); READ_FONT_DATA(ff, fontname, 32, 1); mprintf((0, " ::::::", ft->height, ft->min_ascii, ft->max_ascii, ft->baseline)); num_char = ft->max_ascii - ft->min_ascii + 1; // Read in all widths if (ft->flags & FT_PROPORTIONAL) { ft->char_widths = new short[num_char]; for (i = 0; i < num_char; i++) ft->char_widths[i] = READ_FONT_SHORT(ff); mprintf((0, "::proportional")); } else { ft->char_widths = NULL; } // Read in pixel data. // for color fonts, read in byte count and then the data, // generate character data pointer table // for mono fonts, read in byte count, then the data, convert to bits and store // generate character data pointer table int bytesize = READ_FONT_INT(ff); ft->raw_data = (ubyte *)mem_malloc(bytesize); ft->char_data = (ubyte **)mem_malloc(num_char * sizeof(ubyte *)); READ_FONT_DATA(ff, ft->raw_data, bytesize, 1); if (ft->flags & FT_COLOR) { int off = 0; mprintf((0, "::color")); for (i = 0; i < num_char; i++) { ft->char_data[i] = ft->raw_data + off; if (ft->flags & FT_PROPORTIONAL) off += (ft->char_widths[i] * ft->height * BITS_TO_BYTES(BPP_16)); else off += (ft->width * ft->height * BITS_TO_BYTES(BPP_16)); } } else { // Monochrome ubyte *ptr = ft->raw_data; mprintf((0, "::mono")); for (i = 0; i < num_char; i++) { ft->char_data[i] = ptr; if (ft->flags & FT_PROPORTIONAL) ptr += BITS_TO_BYTES(ft->char_widths[i]) * ft->height; else ptr += BITS_TO_BYTES(ft->width) * ft->height; } } // Read in kerning data ft->kern_data = NULL; // Then read in CLOSE_FONT(ff); mprintf((0, "\n")); grFont::m_FontList[slot].font = fnt; // draw font to bitmaps, load into surfaces too. grFont::translate_to_surfaces(slot); } // translates a font to surfaces void grFont::translate_to_surfaces(int slot) { gr_font_file_record *fntfile; gr_font_record *fnt; fnt = &grFont::m_FontList[slot]; fntfile = &grFont::m_FontList[slot].font; // start creating font surfaces, map these surfaces onto bitmaps created via bitmap library // this is needed for the renderer library. // create a 128x128 surface first. // draw each character into surface until we need to create another // surface. ubyte u = 0, v = 0, w; int ch, num_ch; ubyte surf_index = 0; num_ch = fntfile->max_ascii - fntfile->min_ascii + 1; // initialize memory fnt->ch_w = new ubyte[num_ch]; fnt->ch_h = new ubyte[num_ch]; fnt->ch_u = new ubyte[num_ch]; fnt->ch_v = new ubyte[num_ch]; fnt->ch_surf = new ubyte[num_ch]; fnt->ch_uf = new float[num_ch]; fnt->ch_vf = new float[num_ch]; fnt->ch_wf = new float[num_ch]; fnt->ch_hf = new float[num_ch]; fnt->bmps[surf_index] = bm_AllocBitmap(GRFONT_SURFACE_WIDTH, GRFONT_SURFACE_HEIGHT, 0); if (fnt->bmps[surf_index] == -1) Error("grFont::translate_to_surfaces "); fnt->surfs[surf_index] = new grMemorySurface; fnt->surfs[surf_index]->init(GRFONT_SURFACE_WIDTH, GRFONT_SURFACE_HEIGHT, BPP_16, (char *)bm_data(fnt->bmps[surf_index], 0), GRFONT_SURFACE_WIDTH * BITS_TO_BYTES(BPP_16), SURFFLAG_COLORKEY); fnt->surfs[surf_index]->clear(TRANSPARENT_COLOR32); surf_index++; for (ch = 0; ch < num_ch; ch++) { if (fntfile->flags & FT_PROPORTIONAL) w = (int)fntfile->char_widths[ch]; else w = (int)fntfile->width; // blt each character if (fntfile->flags & FT_COLOR) { m_CharSurf->init(w, fntfile->height, BPP_16, (char *)fntfile->char_data[ch], w * BITS_TO_BYTES(BPP_16), SURFFLAG_COLORKEY); fnt->surfs[surf_index - 1]->blt(u, v, m_CharSurf); } else { // font monochrome, convert bits to pixels grFont::translate_mono_char(fnt->surfs[surf_index - 1], u, v, ch, fntfile, w); } fnt->ch_h[ch] = fntfile->height; fnt->ch_w[ch] = w; fnt->ch_u[ch] = u; fnt->ch_v[ch] = v; fnt->ch_surf[ch] = surf_index - 1; fnt->ch_hf[ch] = ((float)fntfile->height) / ((float)GRFONT_SURFACE_HEIGHT); fnt->ch_wf[ch] = ((float)w) / ((float)GRFONT_SURFACE_WIDTH); fnt->ch_uf[ch] = ((float)u) / ((float)GRFONT_SURFACE_WIDTH); fnt->ch_vf[ch] = ((float)v) / ((float)GRFONT_SURFACE_HEIGHT); // check to adjust uv's if we are outside surface. u += w; if ((u + w) > GRFONT_SURFACE_WIDTH) { u = 0; v += fntfile->height; if ((v + fntfile->height) > GRFONT_SURFACE_HEIGHT) { if (surf_index == MAX_FONT_BITMAPS) Int3(); fnt->bmps[surf_index] = bm_AllocBitmap(GRFONT_SURFACE_WIDTH, GRFONT_SURFACE_HEIGHT, 0); if (fnt->bmps[surf_index] == -1) Error("grFont::translate_to_surfaces "); fnt->surfs[surf_index] = new grMemorySurface; fnt->surfs[surf_index]->init(GRFONT_SURFACE_WIDTH, GRFONT_SURFACE_HEIGHT, BPP_16, (char *)bm_data(fnt->bmps[surf_index], 0), GRFONT_SURFACE_WIDTH * BITS_TO_BYTES(BPP_16), SURFFLAG_COLORKEY); fnt->surfs[surf_index]->clear(TRANSPARENT_COLOR32); surf_index++; v = 0; } } } } // translate mono font data to the surface void grFont::translate_mono_char(grSurface *sf, int x, int y, int index, gr_font_file_record *ft, int width) { int row, col; // byte width of char int rowsize; ubyte bit_mask = 0, byte; ubyte *fp; fp = ft->char_data[index]; switch (sf->bpp()) { case BPP_16: { /* draw one-bit one color. */ ushort *dest_ptr; ushort col_w = GR_COLOR_TO_16(GR_RGB(255, 255, 255)); int rowsize_w; dest_ptr = (ushort *)sf->lock(&rowsize); rowsize_w = sf->rowsize() / 2; dest_ptr += (y * rowsize_w) + x; for (row = 0; row < ft->height; row++) { bit_mask = 0; for (col = 0; col < width; col++) { if (bit_mask == 0) { byte = *fp++; bit_mask = 0x80; } if (byte & bit_mask) dest_ptr[col] = col_w; bit_mask >>= 1; } dest_ptr += rowsize_w; } sf->unlock(); break; } case BPP_32: default: Int3(); // NOT SUPPORTED YET } } // ---------------------------------------------------------------------------- // draw_char function // give a viewport x,y and char // ---------------------------------------------------------------------------- int grFont::draw_char(grSurface *sf, int x, int y, int ch, tCharProperties *chprop) { gr_font_record *ft; int next_x = 0, width, index; ASSERT(m_FontHandle > -1); if ((ch < min_ascii()) || (ch > max_ascii())) return (next_x + 1); // save current x and get this font ft = &grFont::m_FontList[m_FontHandle]; next_x = x; index = ch - ft->font.min_ascii; // draw either a color bitmap or mono font. if (ft->font.flags & FT_PROPORTIONAL) { width = (int)ft->font.char_widths[index]; } else { width = (int)ft->font.width; } // perform blt. if (sf->get_flags() & SURFFLAG_RENDERER) { rend_SetCharacterParameters(chprop->col[0], chprop->col[1], chprop->col[2], chprop->col[3]); rend_DrawFontCharacter(ft->bmps[ft->ch_surf[index]], x, y, x + ft->ch_w[index] - 1, y + ft->ch_h[index] - 1, ft->ch_uf[index], ft->ch_vf[index], ft->ch_wf[index], ft->ch_hf[index]); } else if (ft->font.flags & FT_COLOR) { sf->blt(x, y, ft->surfs[ft->ch_surf[index]], ft->ch_u[index], ft->ch_v[index], ft->ch_w[index], ft->ch_h[index]); } else { charblt16(sf, chprop->col[0], x, y, ft->surfs[ft->ch_surf[index]], ft->ch_u[index], ft->ch_v[index], ft->ch_w[index], ft->ch_h[index]); } // adjust next x value with kerning and return it. next_x += width; return next_x; } int grFont::draw_char(grSurface *sf, int x, int y, int ch, int sx, int sy, int sw, int sh, tCharProperties *chprop) { gr_font_record *ft; int next_x = 0, width, index; ASSERT(m_FontHandle > -1); if ((ch < min_ascii()) || (ch > max_ascii())) return (next_x + 1); // save current x and get this font ft = &grFont::m_FontList[m_FontHandle]; next_x = x; index = ch - ft->font.min_ascii; // draw either a color bitmap or mono font. if (ft->font.flags & FT_PROPORTIONAL) { width = (int)ft->font.char_widths[index]; } else { width = (int)ft->font.width; } ASSERT(sx + sw <= width); ASSERT(sy + sh <= ft->font.height); // peform blt instead. // perform blt. if (sf->get_flags() & SURFFLAG_RENDERER) { rend_SetCharacterParameters(chprop->col[0], chprop->col[1], chprop->col[2], chprop->col[3]); rend_DrawFontCharacter(ft->bmps[ft->ch_surf[index]], x, y, x + sw - 1, y + sh - 1, ft->ch_uf[index] + (((float)sx) / ((float)GRFONT_SURFACE_WIDTH)), ft->ch_vf[index] + (((float)sy) / ((float)GRFONT_SURFACE_HEIGHT)), ((float)sw) / ((float)GRFONT_SURFACE_WIDTH), ((float)sh) / ((float)GRFONT_SURFACE_HEIGHT)); } else if (ft->font.flags & FT_COLOR) { sf->blt(x, y, ft->surfs[ft->ch_surf[index]], ft->ch_u[index] + sx, ft->ch_v[index] + sy, sw, sh); } else { charblt16(sf, chprop->col[0], x, y, ft->surfs[ft->ch_surf[index]], ft->ch_u[index] + sx, ft->ch_v[index] + sy, sw, sh); } // adjust next x value with kerning and return it. next_x += sw; return next_x; } // ---------------------------------------------------------------------------- // accessor functions // ---------------------------------------------------------------------------- int grFont::get_char_info(int ch, int *width) { gr_font_file_record *ft; ASSERT(m_FontHandle > -1); ft = &grFont::m_FontList[m_FontHandle].font; if (ch < ft->min_ascii || ch > ft->max_ascii) *width = 0; else if (ft->flags & FT_PROPORTIONAL) *width = ft->char_widths[ch - ft->min_ascii]; else *width = ft->width; return 0; } ubyte *grFont::get_kern_info(ubyte c1, ubyte c2) { // gr_font_file_record *ft; // ubyte *p; // p = ft->kern_data; // ft = &grFont::m_FontList[m_FontHandle].font; // c2 = c2 - ft->min_ascii; // c1 = c1 - ft->min_ascii; // while (*p!=255) // { // if (p[0]==c1 && p[1]==c2) return p; // else p+=3; // } return NULL; } void grFont::charblt16(grSurface *dsf, ddgr_color col, int dx, int dy, grSurface *ssf, int sx, int sy, int sw, int sh) { ushort *dbits; ushort *sbits; int srowsize_w, drowsize_w, row, coln; ushort scol = GR_COLOR_TO_16(col); dbits = (ushort *)dsf->lock(&drowsize_w); if (dbits) { sbits = (ushort *)ssf->lock(&srowsize_w); if (sbits) { srowsize_w >>= 1; // rowsize in shorts drowsize_w >>= 1; dbits = dbits + (dy * drowsize_w) + dx; sbits = sbits + (sy * srowsize_w) + sx; for (row = 0; row < sh; row++) { for (coln = 0; coln < sw; coln++) { if (sbits[coln] & OPAQUE_FLAG16) dbits[coln] = scol; } sbits += srowsize_w; dbits += drowsize_w; } ssf->unlock(); } dsf->unlock(); } }