You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
230 lines
5.8 KiB
230 lines
5.8 KiB
|
|
//#define DEBUG 0 |
|
|
|
#include <klocale.h> |
|
#include <malloc.h> |
|
#include <kdebug.h> |
|
|
|
#include "font.h" |
|
|
|
|
|
extern "C" { |
|
#include <kpathsea/config.h> |
|
#include <kpathsea/c-fopen.h> |
|
#include <kpathsea/c-stat.h> |
|
#include <kpathsea/magstep.h> |
|
#include <kpathsea/tex-glyph.h> |
|
#include "dvi.h" |
|
} |
|
|
|
#include <stdio.h> |
|
#include "oconfig.h" |
|
|
|
extern FILE *xfopen(const char *filename, const char *type); |
|
extern void oops(QString message); |
|
|
|
/** We try for a VF first because that's what dvips does. Also, it's |
|
* easier to avoid running MakeTeXPK if we have a VF this way. */ |
|
|
|
FILE *font::font_open (char *font, char **font_ret, double dpi, int *dpi_ret, char **filename_ret) |
|
{ |
|
FILE *ret; |
|
char *name = kpse_find_vf (font); |
|
|
|
if (name) { |
|
// VF fonts don't have a resolution, but loadfont will complain if |
|
// we don't return what it asked for. |
|
*dpi_ret = (int)dpi; |
|
*font_ret = NULL; |
|
} else { |
|
kpse_glyph_file_type file_ret; |
|
name = kpse_find_glyph (font, (unsigned) (dpi + .5), kpse_any_glyph_format, &file_ret); |
|
if (name) { |
|
// If we got it normally, from an alias, or from MakeTeXPK, |
|
// don't fill in FONT_RET. That tells load_font to complain. |
|
|
|
// tell load_font we found something good |
|
*font_ret = file_ret.source == kpse_glyph_source_fallback ? file_ret.name : NULL; |
|
*dpi_ret = file_ret.dpi; |
|
} |
|
// If no VF and no PK, FONT_RET is irrelevant? |
|
} |
|
|
|
// If we found a name, return the stream. |
|
ret = name ? xfopen(name, FOPEN_R_MODE) : NULL; |
|
*filename_ret = name; |
|
|
|
return ret; |
|
} |
|
|
|
|
|
|
|
extern int n_files_left; |
|
|
|
font::font(char *nfontname, float nfsize, long chk, int mag, double dconv) |
|
{ |
|
#ifdef DEBUG_FONT |
|
kdDebug() << "constructing font " << nfontname << " at " << (int) (nfsize + 0.5) << " dpi" << endl; |
|
#endif |
|
|
|
fontname = nfontname; |
|
fsize = nfsize; |
|
checksum = chk; |
|
magstepval = mag; |
|
flags = font::FONT_IN_USE; |
|
file = NULL; |
|
filename = NULL; |
|
dimconv = dconv; |
|
} |
|
|
|
font::~font() |
|
{ |
|
#ifdef DEBUG_FONT |
|
kdDebug() << "discarding font " << fontname << " at " << (int)(fsize + 0.5) << " dpi" << endl; |
|
#endif |
|
|
|
free(fontname); |
|
|
|
if (flags & FONT_LOADED) { |
|
if (file != NULL) { |
|
fclose(file); |
|
++n_files_left; |
|
} |
|
free(filename); |
|
|
|
if (flags & FONT_VIRTUAL) { |
|
for (macro *m = macrotable; m < macrotable + max_num_of_chars_in_font; ++m) |
|
if (m->free_me) |
|
free((char *) m->pos); |
|
free((char *) macrotable); |
|
vf_table.clear(); |
|
} else { |
|
for (glyph *g = glyphtable; g < glyphtable + max_num_of_chars_in_font; ++g) |
|
delete g; |
|
free((char *) glyphtable); |
|
} |
|
} |
|
} |
|
|
|
|
|
#define PK_PRE 247 |
|
#define PK_ID 89 |
|
#define PK_MAGIC (PK_PRE << 8) + PK_ID |
|
#define GF_PRE 247 |
|
#define GF_ID 131 |
|
#define GF_MAGIC (GF_PRE << 8) + GF_ID |
|
#define VF_PRE 247 |
|
#define VF_ID_BYTE 202 |
|
#define VF_MAGIC (VF_PRE << 8) + VF_ID_BYTE |
|
|
|
/** load_font locates the raster file and reads the index of |
|
characters, plus whatever other preprocessing is done (depending |
|
on the format). */ |
|
|
|
unsigned char font::load_font(void) |
|
{ |
|
#ifdef DEBUG_FONT |
|
kdDebug() << "loading font " << fontname << " at " << (int) (fsize + 0.5) << " dpi" << endl; |
|
#endif |
|
|
|
int dpi = (int)(fsize + 0.5); |
|
char *font_found; |
|
int size_found; |
|
int magic; |
|
|
|
flags |= font::FONT_LOADED; |
|
file = font_open(fontname, &font_found, (double)fsize, &size_found, &filename); |
|
if (file == NULL) { |
|
kdError() << i18n("Can't find font ") << fontname << "." << endl; |
|
return True; |
|
} |
|
--n_files_left; |
|
if (font_found != NULL) { |
|
kdError() << QString(i18n("Can't find font %1; using %2 instead at %3 dpi.")).arg(fontname).arg(font_found).arg(dpi) << endl; |
|
free(fontname); |
|
fontname = font_found; |
|
} |
|
else |
|
if (!kpse_bitmap_tolerance ((double) size_found, fsize)) |
|
kdError() << i18n("Can't find font ") << fontname << i18n(" at ") << dpi << i18n(" dpi; using ") |
|
<< size_found << i18n(" dpi instead.") << endl; |
|
fsize = size_found; |
|
timestamp = ++current_timestamp; |
|
set_char_p = &dviWindow::set_char; |
|
magic = two(file); |
|
|
|
if (magic == PK_MAGIC) |
|
read_PK_index(); |
|
else |
|
if (magic == GF_MAGIC) |
|
oops(QString(i18n("The GF format for font file %1 is no longer supported")).arg(filename) ); |
|
else |
|
if (magic == VF_MAGIC) |
|
read_VF_index(); |
|
else |
|
oops(QString(i18n("Cannot recognize format for font file %1")).arg(filename) ); |
|
|
|
return False; |
|
} |
|
|
|
|
|
|
|
/** mark_as_used marks the font, and all the fonts it referrs to, as |
|
used, i.e. their FONT_IN_USE-flag is set. */ |
|
|
|
void font::mark_as_used(void) |
|
{ |
|
#ifdef DEBUG_FONT |
|
kdDebug() << "marking font " << fontname << " at " << (int) (fsize + 0.5) << " dpi" << endl; |
|
#endif |
|
|
|
if (flags & font::FONT_IN_USE) |
|
return; |
|
|
|
flags |= font::FONT_IN_USE; |
|
|
|
// For virtual fonts, also go through the list of referred fonts |
|
if (flags & font::FONT_VIRTUAL) { |
|
QIntDictIterator<font> it(vf_table); |
|
while( it.current() ) { |
|
it.current()->flags |= font::FONT_IN_USE; |
|
++it; |
|
} |
|
} |
|
} |
|
|
|
/** Returns a pointer to glyph number ch in the font, or NULL, if this |
|
number does not exist. This function first reads the bitmap of the |
|
character from the PK-file, if necessary */ |
|
|
|
struct glyph *font::glyphptr(unsigned int ch) { |
|
|
|
struct glyph *g = glyphtable+ch; |
|
if (g->bitmap.bits == NULL) { |
|
if (g->addr == 0) { |
|
kdError() << i18n("Character ") << ch << i18n(" not defined in font ") << fontname << endl; |
|
g->addr = -1; |
|
return NULL; |
|
} |
|
if (g->addr == -1) |
|
return NULL; /* previously flagged missing char */ |
|
|
|
if (file == NULL) { |
|
file = xfopen(filename, OPEN_MODE); |
|
if (file == NULL) { |
|
oops(QString(i18n("Font file disappeared: %1")).arg(filename) ); |
|
return NULL; |
|
} |
|
} |
|
Fseek(file, g->addr, 0); |
|
read_PK_char(ch); |
|
timestamp = ++current_timestamp; |
|
|
|
if (g->bitmap.bits == NULL) { |
|
g->addr = -1; |
|
return NULL; |
|
} |
|
} |
|
|
|
return g; |
|
}
|
|
|