Files
luos/thirdparty/html2ps_pdf/font.class.php
Paula Quispe 9eb7d6cac2 HOR-2689
2017-08-03 17:00:30 -04:00

219 lines
6.1 KiB
PHP

<?php
$GLOBALS['g_last_assigned_font_id'] = 0;
class Font {
var $underline_position;
var $underline_thickness;
var $ascender;
var $descender;
var $char_widths;
var $bbox;
function ascender() {
return $this->ascender;
}
function descender() {
return $this->descender;
}
function error_message() {
return $this->error_message;
}
function Font() {}
function linethrough_position() {
return $this->bbox[3]*0.25;
}
function name() {
return $this->name;
}
function overline_position() {
return $this->bbox[3]*0.8;
}
function points($fontsize, $dimension) {
return $dimension * $fontsize / 1000;
}
function stringwidth($string) {
$width = 0;
$length = strlen($string);
for ($i=0; $i<$length; $i++) {
$width += $this->char_widths[$string{$i}];
};
return $width;
}
function underline_position() {
return $this->underline_position;
}
function underline_thickness() {
return $this->underline_thickness;
}
}
class FontTrueType extends Font {
function create($fontfile, $encoding) {
$font = new FontTrueType();
$font->_read(TTF_FONTS_REPOSITORY.$fontfile, $encoding);
return $font;
}
/**
* TODO: cache results; replace makefont with this utility
*/
function _read($file, $encoding) {
error_log(sprintf("Parsing font file file %s for encoding %s", $file, $encoding));
$font = new OpenTypeFile();
$font->open($file);
$hhea = $font->getTable('hhea');
$head = $font->getTable('head');
$hmtx = $font->getTable('hmtx');
$post = $font->getTable('post');
$cmap = $font->getTable('cmap');
$subtable = $cmap->findSubtable(OT_CMAP_PLATFORM_WINDOWS,
OT_CMAP_PLATFORM_WINDOWS_UNICODE);
/**
* Read character widths for selected encoding
*/
$widths = array();
$manager = ManagerEncoding::get();
$map = $manager->getEncodingVector($encoding);
foreach ($map as $code => $ucs2) {
$glyphIndex = $subtable->lookup($ucs2);
if (!is_null($glyphIndex)) {
$widths[$code] = floor($hmtx->_hMetrics[$glyphIndex]['advanceWidth']*1000/$head->_unitsPerEm);
} else {
$widths[$code] = DEFAULT_CHAR_WIDTH;
};
};
// Fill unknown characters with the default char width
for ($i=0; $i<256; $i++) {
if (!isset($widths[chr($i)])) {
$widths[chr($i)] = DEFAULT_CHAR_WIDTH;
};
};
$this->ascender = floor($hhea->_ascender*1000/$head->_unitsPerEm);
$this->descender = floor($hhea->_descender*1000/$head->_unitsPerEm);
$this->bbox = array($head->_xMin*1000/$head->_unitsPerEm,
$head->_yMin*1000/$head->_unitsPerEm,
$head->_xMax*1000/$head->_unitsPerEm,
$head->_yMax*1000/$head->_unitsPerEm);
$this->underline_position = floor($post->_underlinePosition*1000/$head->_unitsPerEm);
$this->underline_thickness = floor($post->_underlineThickness*1000/$head->_unitsPerEm);
$this->char_widths = $widths;
$font->close();
}
}
// Note that ALL font dimensions are measured in 1/1000 of font size units;
//
class FontType1 extends Font {
function &create($typeface, $encoding, $font_resolver, &$error_message) {
$font = new FontType1();
$font->underline_position = 0;
$font->underline_thickness = 0;
$font->ascender;
$font->descender;
$font->char_widths = array();
$font->bbox = array();
global $g_last_assigned_font_id;
$g_last_assigned_font_id++;
$font->name = "font".$g_last_assigned_font_id;
// Get and load the metrics file
$afm = $font_resolver->get_afm_mapping($typeface);
if (!$font->_parse_afm($afm, $typeface, $encoding)) {
$error_message = $font->error_message();
$dummy = null;
return $dummy;
};
return $font;
}
// Parse the AFM metric file; keep only sized of glyphs present in the chosen encoding
function _parse_afm($afm, $typeface, $encoding) {
global $g_manager_encodings;
$encoding_data = $g_manager_encodings->get_glyph_to_code_mapping($encoding);
$filename = TYPE1_FONTS_REPOSITORY.$afm.".afm";
$file = @fopen($filename, 'r');
if (!$file) {
$_filename = $filename;
$_typeface = $typeface;
ob_start();
include(HTML2PS_DIR.'/templates/error._missing_afm.tpl');
$this->error_message = ob_get_contents();
ob_end_clean();
error_log(sprintf("Missing font metrics file: %s",$filename));
return false;
};
while ($line = fgets($file)) {
if (preg_match("/C\s-?\d+\s;\sWX\s(\d+)\s;\sN\s(\S+)\s;/",$line,$matches)) {
$glyph_width = $matches[1];
$glyph_name = $matches[2];
// This line is a character width definition
if (isset($encoding_data[$glyph_name])) {
foreach ($encoding_data[$glyph_name] as $c) {
$this->char_widths[$c] = $glyph_width;
};
};
} elseif (preg_match("/UnderlinePosition ([\d-]+)/",$line,$matches)) {
// This line is an underline position line
$this->underline_position = $matches[1];
} elseif (preg_match("/UnderlineThickness ([\d-]+)/",$line,$matches)) {
// This line is an underline thickness line
$this->underline_thickness = $matches[1];
} elseif (preg_match("/Ascender ([\d-]+)/",$line,$matches)) {
// This line is an ascender line
$this->ascender = $matches[1];
} elseif (preg_match("/Descender ([\d-]+)/",$line,$matches)) {
// This line is an descender line
$this->descender = $matches[1];
} elseif (preg_match("/FontBBox ([\d-]+) ([\d-]+) ([\d-]+) ([\d-]+)/",$line,$matches)) {
// This line is an font BBox line
$this->bbox = array($matches[1], $matches[2], $matches[3], $matches[4]);
};
};
fclose($file);
// Fill unknown characters with the default char width
for ($i=0; $i<256; $i++) {
if (!isset($this->char_widths[chr($i)])) {
$this->char_widths[chr($i)] = DEFAULT_CHAR_WIDTH;
};
};
return true;
}
}
?>