/**[txh]********************************************************************
Copyright 1996-2003 by Salvador Eduardo Tropea (SET)
This file is covered by the GPL license.
Module: TVFontCollection
Comments:
This module handles collections of 8xN bitmaped fonts useful for text
consoles. The collection is created from a fonts files. Each item is a font
of certain height. The collection can return interpolated and extrapolated
fonts in a range of +/- one from the available fonts. The fonts file should
contain fonts with all the supported symbols.@p
@
SFT Files: This is currently platform dependent! It means SFT files should
be generated in the same platform they are used. It will be fixed in the
future.
17 bytes: SET's editor font
1 byte: 26 (DOS EOF)
int: version (1 or 2)
int: number of fonts contained.
2 bytes: size of name
the name, the length includes the EOS, but this isn't stored on
disk.
Here are the fonts, each one have this format:
version 1:
int: first symbol defined.
int: last symbol defined.
unsigned: height of the font.
The width is assumed to be 8 pixels (1 byte)
version 2, the same but adds:
unsigned: width of the font.
The fonts itself follows the internal code.
@
***************************************************************************/
#define Uses_stdlib
#define Uses_string
#define Uses_dirent
#define Uses_limits
#define Uses_TVCodePage
#define Uses_TVFontCollection
#include
//#define TEST
const int SizeInDisk1=3*sizeof(int);
const int SizeInDisk2=SizeInDisk1+sizeof(int);
const char TVFontCollection::Signature[]="SET's editor font\x1A";
const char TVFontCollection::SFTExtension[]=".sft";
int TVFontCollection::invertEndian=0;
struct SizeFont
{
unsigned width,height;
};
static
int isWordChar(int i)
{
return i=='_' || TVCodePage::isAlpha(i);
}
/**[txh]********************************************************************
Description:
A specialized free to free the memory used by the font structure.
***************************************************************************/
void TVFontCollection::freeItem(void *item)
{
TVBitmapFont *p=(TVBitmapFont *)item;
if (p)
{
DeleteArray(p->font);
DeleteArray(p->fontFull);
delete p;
}
}
/**[txh]********************************************************************
Description:
Internally used during searchs to look for a font that matchs a specified
size.
***************************************************************************/
Boolean TVFontCollection::CheckForLines(void *item, void *arg)
{
TVBitmapFont *p=(TVBitmapFont *)item;
SizeFont *s=(SizeFont *)arg;
if (p && p->lines==s->height && p->width==s->width)
return True;
return False;
}
/**[txh]********************************************************************
Description:
Creates a font of the specified height from another of the same height
plus one. The destination buffer should be large enough (num*height bytes).
Special care is taked for letters.
***************************************************************************/
void TVFontCollection::ReduceOne(uchar *dest, uchar *ori, int height,
int wBytes, int num)
{
int i;
unsigned sizeDest=wBytes*height;
unsigned sizeOri=sizeDest+wBytes;
// height is the dest size, ori is one more
for (i=0; ifont)
return NULL;
unsigned size=256*height*p->wBytes;
uchar *fontShape=new uchar[size];
if (oneMore)
ReduceOne(fontShape,p->font,height,p->wBytes);
else if (oneLess)
EnlargeOne(fontShape,p->font,height,p->wBytes);
else
memcpy(fontShape,p->font,size);
return fontShape;
}
/**[txh]********************************************************************
Description:
Returns a newly allocated block of memory containing a font of the
specified height. This is a full font and not just one for the current code
page.
Return: NULL if the font isn't available.
***************************************************************************/
uchar *TVFontCollection::GetFontFull(int width, int height, int &first,
int &last)
{
int oneMore=0,oneLess=0;
SizeFont sz={width,height};
TVBitmapFont *p=(TVBitmapFont *)firstThat(CheckForLines,&sz);
// If we can't find a font of the right size look for 1 more and one less
if (!p)
{
sz.height++;
p=(TVBitmapFont *)firstThat(CheckForLines,&sz);
if (p)
oneMore=1;
else
{
sz.height-=2;
p=(TVBitmapFont *)firstThat(CheckForLines,&sz);
if (p)
oneLess=1;
}
}
if (!p || !p->fontFull)
return NULL;
first=p->first; last=p->last;
unsigned symbols=last-first+1;
unsigned size=symbols*height*p->wBytes;
uchar *fontShape=new uchar[size];
if (oneMore)
ReduceOne(fontShape,p->fontFull,height,p->wBytes,symbols);
else if (oneLess)
EnlargeOne(fontShape,p->fontFull,height,p->wBytes,symbols);
else
memcpy(fontShape,p->fontFull,size);
return fontShape;
}
/**[txh]********************************************************************
Description:
Internally used to check if we opened a fonts file.
***************************************************************************/
int TVFontCollection::CheckSignature(FILE *f)
{
char buf[sizeof(Signature)];
fread(buf,sizeof(Signature)-1,1,f);
buf[sizeof(Signature)-1]=0;
return strcmp(Signature,buf)==0;
}
/**[txh]********************************************************************
Description:
Internally used to get the font's name.
***************************************************************************/
char *TVFontCollection::ReadName(FILE *f)
{
uint16 strLen;
fread(&strLen,2,1,f);
Swap(&strLen);
char *aux=new char[strLen];
strLen--;
fread(aux,strLen,1,f);
aux[strLen]=0;
return aux;
}
#define SwapMacro(a,b) t=v[a]; v[a]=v[b]; v[b]=t
void TVFontCollection::Swap(int *value)
{
if (!invertEndian) return;
char *v=(char *)value, t;
SwapMacro(0,3);
SwapMacro(1,2);
}
void TVFontCollection::Swap(uint16 *value)
{
if (!invertEndian) return;
char *v=(char *)value, t;
SwapMacro(0,1);
}
/**[txh]********************************************************************
Description:
Internally used to read the version and number of fonts in the file.
***************************************************************************/
void TVFontCollection::ReadVersionNum(FILE *f, int *version, int *numfonts)
{
fread(version,4,1,f);
fread(numfonts,4,1,f);
invertEndian=*version>0x1000;
Swap(version);
Swap(numfonts);
}
/**[txh]********************************************************************
Description:
Internally used to read the information about a font contained in an SFT
file.
Return: Size of the data font.
***************************************************************************/
unsigned TVFontCollection::ReadFontInfo(FILE *f, int version, TVBitmapFont *p)
{
if (version==1)
{
fread(p,SizeInDisk1,1,f);
p->width=8;
p->wBytes=1;
}
else
{
fread(p,SizeInDisk2,1,f);
Swap(&p->width);
p->wBytes=(p->width+7)/8;
}
Swap(&p->first);
Swap(&p->last);
Swap(&p->lines);
return (p->last-p->first+1)*p->lines*p->wBytes;
}
/**[txh]********************************************************************
Description:
Internally used to create a font for a desired code page.
***************************************************************************/
void TVFontCollection::CreateFont(void *item, void *arg)
{
TVBitmapFont *f=(TVBitmapFont *)item;
ushort *map=(ushort *)arg;
DeleteArray(f->font);
unsigned size1=f->lines*f->wBytes;
unsigned size=256*size1;
f->font=new uchar[size];
int i;
uchar *dest=f->font;
memset(f->font,0,size);
for (i=0; i<256; i++,dest+=size1)
{
int index=map[i];
if (index>f->last)
{
index=TVCodePage::LookSimilarInRange(index,f->last);
if (index==-1)
index=f->first;
}
index-=f->first;
memcpy(dest,&f->fontFull[index*size1],size1);
}
#if 0
// This code stores the generated font to disk, is used for debug
// purposes
FILE *F;
char b[PATH_MAX],*t;
t=getenv("TMP");
if (!t) t="/tmp";
sprintf(b,"%s/font.%03d",t,f->lines);
F=fopen(b,"wb");
fwrite(f->font,size,1,F);
fclose(F);
#endif
}
/**[txh]********************************************************************
Description:
Sets the encoding of the fonts returned by GetFont.
***************************************************************************/
void TVFontCollection::SetCodePage(int id)
{
ushort *map=TVCodePage::GetTranslate(id);
if (map)
forEach(CreateFont,map);
}
TVFontCollection::~TVFontCollection()
{
DeleteArray(fontName);
DeleteArray(fileName);
}
/**[txh]********************************************************************
Description:
Creates a font collection from the specified file and using the specified
code page. You must check the error status with GetError before using the
collection.
***************************************************************************/
TVFontCollection::TVFontCollection(const char *file, int cp) :
TNSCollection(2,2)
{
error=0;
fontName=fileName=NULL;
if (!file)
{
error=1;
return;
}
FILE *f=fopen(file,"rb");
if (!f)
{
error=2;
return;
}
if (!CheckSignature(f))
{
fclose(f);
error=3;
return;
}
int version;
int numfonts;
ReadVersionNum(f,&version,&numfonts);
int i;
TVBitmapFont *p;
uchar *fData;
unsigned size;
fontName=ReadName(f);
fileName=newStr(file);
for (i=0; ifontFull=fData;
p->font=0;
insert(p);
}
fclose(f);
SetCodePage(cp);
}
/**[txh]********************************************************************
Description:
Constructs a string collection of fonts files that provides shapes inside
the specified range. The fonts are searched in the specified directory. If
you specify NULL as directory then the search is done in the current
directory. Each element in the collection is a string that contains two
ASCIIZ strings. The first is the fantasy name of the font and the second is
the name of the file containing it. It is useful to display the name in
a TSortedListBox and easilly locate which file contains it using strlen.@*
This is an static member.
Return: The collection of available fonts or NULL if none.
***************************************************************************/
TVBitmapFontDescCol *TVFontCollection::CreateListOfFonts(const char *dir,
unsigned wmin, unsigned wmax, unsigned hmin, unsigned hmax)
{
char *FullName=new char[PATH_MAX];
TVBitmapFontDescCol *col=new TVBitmapFontDescCol();
if (!dir)
dir=".";
DIR *d=opendir(dir);
struct dirent *de;
int version;
int numfonts;
unsigned size;
TVBitmapFont stF;
TVBitmapFontSize sizeSt;
if (d)
{
while ((de=readdir(d))!=0)
{
if (strstr(de->d_name,SFTExtension))
{
strcpy(FullName,dir);
strcat(FullName,"/");
strcat(FullName,de->d_name);
FILE *f=fopen(FullName,"rb");
if (f)
{
if (CheckSignature(f))
{
ReadVersionNum(f,&version,&numfonts);
char *name=ReadName(f);
TVBitmapFontDesc *d=NULL;
int i,j,added;
ccIndex pos;
for (i=0; i=wmin && stF.width<=wmax &&
(stF.lines+j)>=hmin && (stF.lines+j)<=hmax)
{
if (!d)
{
d=new TVBitmapFontDesc;
d->name=name;
d->file=newStr(FullName);
d->sizes=new TVBitmapFontSizeCol();
}
sizeSt.w=stF.width; sizeSt.h=stF.lines+j;
if (!d->sizes->search(&sizeSt,pos))
{
TVBitmapFontSize *p=new TVBitmapFontSize;
p->w=stF.width;
p->h=stF.lines+j;
d->sizes->insert(p);
}
}
}
fseek(f,size,SEEK_CUR);
}
if (d)
col->insert(d);
else
DeleteArray(name);
}
fclose(f);
}
}
}
closedir(d);
}
delete[] FullName;
if (col->getCount()==0)
{
CLY_destroy(col);
return NULL;
}
return col;
}
void TVBitmapFontDescCol::freeItem(void *item)
{
TVBitmapFontDesc *p=(TVBitmapFontDesc *)item;
DeleteArray(p->name);
DeleteArray(p->file);
CLY_destroy(p->sizes);
delete p;
}
void *TVBitmapFontDescCol::keyOf(void *item)
{
TVBitmapFontDesc *p=(TVBitmapFontDesc *)item;
return (void *)p->name;
}
void TVBitmapFontDescLBox::getText(char *dest, ccIndex item, short maxChars)
{
TVBitmapFontDesc *p=(TVBitmapFontDesc *)items->at(item);
strncpy(dest,p->name,maxChars);
dest[maxChars]=0;
}
int TVBitmapFontSizeCol::compare(void *key1, void *key2)
{
TVBitmapFontSize *v1=(TVBitmapFontSize *)key1;
TVBitmapFontSize *v2=(TVBitmapFontSize *)key2;
if (v1->w>v2->w) return 1;
if (v1->ww) return -1;
return (v1->h>v2->h)-(v1->hh);
}
void TVBitmapFontSizeLBox::getText(char *dest, ccIndex item, short maxChars)
{
TVBitmapFontSize *p=(TVBitmapFontSize *)items->at(item);
// Limit the value to know we won't exceed the buffer
unsigned w=p->w; if (w>999) w=999;
unsigned h=p->h; if (h>999) h=999;
char b[12];
sprintf(b,"%3d x %-3d",w,h);
strncpy(dest,b,maxChars);
dest[maxChars]=0;
}