420 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			420 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /*******************************************************************************
 | |
| * Utility to generate font definition files                                    *
 | |
| *                                                                              *
 | |
| * Version: 1.14                                                                *
 | |
| * Date:    2008-08-03                                                          *
 | |
| * Author:  Olivier PLATHEY                                                     *
 | |
| *******************************************************************************/
 | |
| 
 | |
| function ReadMap($enc)
 | |
| {
 | |
| 	//Read a map file
 | |
| 	$file=dirname(__FILE__).'/'.strtolower($enc).'.map';
 | |
| 	$a=file($file);
 | |
| 	if(empty($a))
 | |
| 		die('<b>Error:</b> encoding not found: '.$enc);
 | |
| 	$cc2gn=array();
 | |
| 	foreach($a as $l)
 | |
| 	{
 | |
| 		if($l[0]=='!')
 | |
| 		{
 | |
| 			$e=preg_split('/[ \\t]+/',rtrim($l));
 | |
| 			$cc=hexdec(substr($e[0],1));
 | |
| 			$gn=$e[2];
 | |
| 			$cc2gn[$cc]=$gn;
 | |
| 		}
 | |
| 	}
 | |
| 	for($i=0;$i<=255;$i++)
 | |
| 	{
 | |
| 		if(!isset($cc2gn[$i]))
 | |
| 			$cc2gn[$i]='.notdef';
 | |
| 	}
 | |
| 	return $cc2gn;
 | |
| }
 | |
| 
 | |
| function ReadAFM($file, &$map)
 | |
| {
 | |
| 	//Read a font metric file
 | |
| 	$a=file($file);
 | |
| 	if(empty($a))
 | |
| 		die('File not found');
 | |
| 	$widths=array();
 | |
| 	$fm=array();
 | |
| 	$fix=array('Edot'=>'Edotaccent','edot'=>'edotaccent','Idot'=>'Idotaccent','Zdot'=>'Zdotaccent','zdot'=>'zdotaccent',
 | |
| 		'Odblacute'=>'Ohungarumlaut','odblacute'=>'ohungarumlaut','Udblacute'=>'Uhungarumlaut','udblacute'=>'uhungarumlaut',
 | |
| 		'Gcedilla'=>'Gcommaaccent','gcedilla'=>'gcommaaccent','Kcedilla'=>'Kcommaaccent','kcedilla'=>'kcommaaccent',
 | |
| 		'Lcedilla'=>'Lcommaaccent','lcedilla'=>'lcommaaccent','Ncedilla'=>'Ncommaaccent','ncedilla'=>'ncommaaccent',
 | |
| 		'Rcedilla'=>'Rcommaaccent','rcedilla'=>'rcommaaccent','Scedilla'=>'Scommaaccent','scedilla'=>'scommaaccent',
 | |
| 		'Tcedilla'=>'Tcommaaccent','tcedilla'=>'tcommaaccent','Dslash'=>'Dcroat','dslash'=>'dcroat','Dmacron'=>'Dcroat','dmacron'=>'dcroat',
 | |
| 		'combininggraveaccent'=>'gravecomb','combininghookabove'=>'hookabovecomb','combiningtildeaccent'=>'tildecomb',
 | |
| 		'combiningacuteaccent'=>'acutecomb','combiningdotbelow'=>'dotbelowcomb','dongsign'=>'dong');
 | |
| 	foreach($a as $l)
 | |
| 	{
 | |
| 		$e=explode(' ',rtrim($l));
 | |
| 		if(count($e)<2)
 | |
| 			continue;
 | |
| 		$code=$e[0];
 | |
| 		$param=$e[1];
 | |
| 		if($code=='C')
 | |
| 		{
 | |
| 			//Character metrics
 | |
| 			$cc=(int)$e[1];
 | |
| 			$w=$e[4];
 | |
| 			$gn=$e[7];
 | |
| 			if(substr($gn,-4)=='20AC')
 | |
| 				$gn='Euro';
 | |
| 			if(isset($fix[$gn]))
 | |
| 			{
 | |
| 				//Fix incorrect glyph name
 | |
| 				foreach($map as $c=>$n)
 | |
| 				{
 | |
| 					if($n==$fix[$gn])
 | |
| 						$map[$c]=$gn;
 | |
| 				}
 | |
| 			}
 | |
| 			if(empty($map))
 | |
| 			{
 | |
| 				//Symbolic font: use built-in encoding
 | |
| 				$widths[$cc]=$w;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$widths[$gn]=$w;
 | |
| 				if($gn=='X')
 | |
| 					$fm['CapXHeight']=$e[13];
 | |
| 			}
 | |
| 			if($gn=='.notdef')
 | |
| 				$fm['MissingWidth']=$w;
 | |
| 		}
 | |
| 		elseif($code=='FontName')
 | |
| 			$fm['FontName']=$param;
 | |
| 		elseif($code=='Weight')
 | |
| 			$fm['Weight']=$param;
 | |
| 		elseif($code=='ItalicAngle')
 | |
| 			$fm['ItalicAngle']=(double)$param;
 | |
| 		elseif($code=='Ascender')
 | |
| 			$fm['Ascender']=(int)$param;
 | |
| 		elseif($code=='Descender')
 | |
| 			$fm['Descender']=(int)$param;
 | |
| 		elseif($code=='UnderlineThickness')
 | |
| 			$fm['UnderlineThickness']=(int)$param;
 | |
| 		elseif($code=='UnderlinePosition')
 | |
| 			$fm['UnderlinePosition']=(int)$param;
 | |
| 		elseif($code=='IsFixedPitch')
 | |
| 			$fm['IsFixedPitch']=($param=='true');
 | |
| 		elseif($code=='FontBBox')
 | |
| 			$fm['FontBBox']=array($e[1],$e[2],$e[3],$e[4]);
 | |
| 		elseif($code=='CapHeight')
 | |
| 			$fm['CapHeight']=(int)$param;
 | |
| 		elseif($code=='StdVW')
 | |
| 			$fm['StdVW']=(int)$param;
 | |
| 	}
 | |
| 	if(!isset($fm['FontName']))
 | |
| 		die('FontName not found');
 | |
| 	if(!empty($map))
 | |
| 	{
 | |
| 		if(!isset($widths['.notdef']))
 | |
| 			$widths['.notdef']=600;
 | |
| 		if(!isset($widths['Delta']) && isset($widths['increment']))
 | |
| 			$widths['Delta']=$widths['increment'];
 | |
| 		//Order widths according to map
 | |
| 		for($i=0;$i<=255;$i++)
 | |
| 		{
 | |
| 			if(!isset($widths[$map[$i]]))
 | |
| 			{
 | |
| 				echo '<b>Warning:</b> character '.$map[$i].' is missing<br>';
 | |
| 				$widths[$i]=$widths['.notdef'];
 | |
| 			}
 | |
| 			else
 | |
| 				$widths[$i]=$widths[$map[$i]];
 | |
| 		}
 | |
| 	}
 | |
| 	$fm['Widths']=$widths;
 | |
| 	return $fm;
 | |
| }
 | |
| 
 | |
| function MakeFontDescriptor($fm, $symbolic)
 | |
| {
 | |
| 	//Ascent
 | |
| 	$asc=(isset($fm['Ascender']) ? $fm['Ascender'] : 1000);
 | |
| 	$fd="array('Ascent'=>".$asc;
 | |
| 	//Descent
 | |
| 	$desc=(isset($fm['Descender']) ? $fm['Descender'] : -200);
 | |
| 	$fd.=",'Descent'=>".$desc;
 | |
| 	//CapHeight
 | |
| 	if(isset($fm['CapHeight']))
 | |
| 		$ch=$fm['CapHeight'];
 | |
| 	elseif(isset($fm['CapXHeight']))
 | |
| 		$ch=$fm['CapXHeight'];
 | |
| 	else
 | |
| 		$ch=$asc;
 | |
| 	$fd.=",'CapHeight'=>".$ch;
 | |
| 	//Flags
 | |
| 	$flags=0;
 | |
| 	if(isset($fm['IsFixedPitch']) && $fm['IsFixedPitch'])
 | |
| 		$flags+=1<<0;
 | |
| 	if($symbolic)
 | |
| 		$flags+=1<<2;
 | |
| 	if(!$symbolic)
 | |
| 		$flags+=1<<5;
 | |
| 	if(isset($fm['ItalicAngle']) && $fm['ItalicAngle']!=0)
 | |
| 		$flags+=1<<6;
 | |
| 	$fd.=",'Flags'=>".$flags;
 | |
| 	//FontBBox
 | |
| 	if(isset($fm['FontBBox']))
 | |
| 		$fbb=$fm['FontBBox'];
 | |
| 	else
 | |
| 		$fbb=array(0,$desc-100,1000,$asc+100);
 | |
| 	$fd.=",'FontBBox'=>'[".$fbb[0].' '.$fbb[1].' '.$fbb[2].' '.$fbb[3]."]'";
 | |
| 	//ItalicAngle
 | |
| 	$ia=(isset($fm['ItalicAngle']) ? $fm['ItalicAngle'] : 0);
 | |
| 	$fd.=",'ItalicAngle'=>".$ia;
 | |
| 	//StemV
 | |
| 	if(isset($fm['StdVW']))
 | |
| 		$stemv=$fm['StdVW'];
 | |
| 	elseif(isset($fm['Weight']) && preg_match('/bold|black/i',$fm['Weight']))
 | |
| 		$stemv=120;
 | |
| 	else
 | |
| 		$stemv=70;
 | |
| 	$fd.=",'StemV'=>".$stemv;
 | |
| 	//MissingWidth
 | |
| 	if(isset($fm['MissingWidth']))
 | |
| 		$fd.=",'MissingWidth'=>".$fm['MissingWidth'];
 | |
| 	$fd.=')';
 | |
| 	return $fd;
 | |
| }
 | |
| 
 | |
| function MakeWidthArray($fm)
 | |
| {
 | |
| 	//Make character width array
 | |
| 	$s="array(\n\t";
 | |
| 	$cw=$fm['Widths'];
 | |
| 	for($i=0;$i<=255;$i++)
 | |
| 	{
 | |
| 		if(chr($i)=="'")
 | |
| 			$s.="'\\''";
 | |
| 		elseif(chr($i)=="\\")
 | |
| 			$s.="'\\\\'";
 | |
| 		elseif($i>=32 && $i<=126)
 | |
| 			$s.="'".chr($i)."'";
 | |
| 		else
 | |
| 			$s.="chr($i)";
 | |
| 		$s.='=>'.$fm['Widths'][$i];
 | |
| 		if($i<255)
 | |
| 			$s.=',';
 | |
| 		if(($i+1)%22==0)
 | |
| 			$s.="\n\t";
 | |
| 	}
 | |
| 	$s.=')';
 | |
| 	return $s;
 | |
| }
 | |
| 
 | |
| function MakeFontEncoding($map)
 | |
| {
 | |
| 	//Build differences from reference encoding
 | |
| 	$ref=ReadMap('cp1252');
 | |
| 	$s='';
 | |
| 	$last=0;
 | |
| 	for($i=32;$i<=255;$i++)
 | |
| 	{
 | |
| 		if($map[$i]!=$ref[$i])
 | |
| 		{
 | |
| 			if($i!=$last+1)
 | |
| 				$s.=$i.' ';
 | |
| 			$last=$i;
 | |
| 			$s.='/'.$map[$i].' ';
 | |
| 		}
 | |
| 	}
 | |
| 	return rtrim($s);
 | |
| }
 | |
| 
 | |
| function SaveToFile($file, $s, $mode)
 | |
| {
 | |
| 	$f=fopen($file,'w'.$mode);
 | |
| 	if(!$f)
 | |
| 		die('Can\'t write to file '.$file);
 | |
| 	fwrite($f,$s,strlen($s));
 | |
| 	fclose($f);
 | |
| }
 | |
| 
 | |
| function ReadShort($f)
 | |
| {
 | |
| 	$a=unpack('n1n',fread($f,2));
 | |
| 	return $a['n'];
 | |
| }
 | |
| 
 | |
| function ReadLong($f)
 | |
| {
 | |
| 	$a=unpack('N1N',fread($f,4));
 | |
| 	return $a['N'];
 | |
| }
 | |
| 
 | |
| function CheckTTF($file)
 | |
| {
 | |
| 	//Check if font license allows embedding
 | |
| 	$f=fopen($file,'rb');
 | |
| 	if(!$f)
 | |
| 		die('<b>Error:</b> Can\'t open '.$file);
 | |
| 	//Extract number of tables
 | |
| 	fseek($f,4,SEEK_CUR);
 | |
| 	$nb=ReadShort($f);
 | |
| 	fseek($f,6,SEEK_CUR);
 | |
| 	//Seek OS/2 table
 | |
| 	$found=false;
 | |
| 	for($i=0;$i<$nb;$i++)
 | |
| 	{
 | |
| 		if(fread($f,4)=='OS/2')
 | |
| 		{
 | |
| 			$found=true;
 | |
| 			break;
 | |
| 		}
 | |
| 		fseek($f,12,SEEK_CUR);
 | |
| 	}
 | |
| 	if(!$found)
 | |
| 	{
 | |
| 		fclose($f);
 | |
| 		return;
 | |
| 	}
 | |
| 	fseek($f,4,SEEK_CUR);
 | |
| 	$offset=ReadLong($f);
 | |
| 	fseek($f,$offset,SEEK_SET);
 | |
| 	//Extract fsType flags
 | |
| 	fseek($f,8,SEEK_CUR);
 | |
| 	$fsType=ReadShort($f);
 | |
| 	$rl=($fsType & 0x02)!=0;
 | |
| 	$pp=($fsType & 0x04)!=0;
 | |
| 	$e=($fsType & 0x08)!=0;
 | |
| 	fclose($f);
 | |
| 	if($rl && !$pp && !$e)
 | |
| 		echo '<b>Warning:</b> font license does not allow embedding';
 | |
| }
 | |
| 
 | |
| /*******************************************************************************
 | |
| * fontfile: path to TTF file (or empty string if not to be embedded)           *
 | |
| * afmfile:  path to AFM file                                                   *
 | |
| * enc:      font encoding (or empty string for symbolic fonts)                 *
 | |
| * patch:    optional patch for encoding                                        *
 | |
| * type:     font type if fontfile is empty                                     *
 | |
| *******************************************************************************/
 | |
| function MakeFont($fontfile, $afmfile, $enc='cp1252', $patch=array(), $type='TrueType')
 | |
| {
 | |
| 	//Generate a font definition file
 | |
| 	if(get_magic_quotes_runtime())
 | |
| 		@set_magic_quotes_runtime(0);
 | |
| 	ini_set('auto_detect_line_endings','1');
 | |
| 	if($enc)
 | |
| 	{
 | |
| 		$map=ReadMap($enc);
 | |
| 		foreach($patch as $cc=>$gn)
 | |
| 			$map[$cc]=$gn;
 | |
| 	}
 | |
| 	else
 | |
| 		$map=array();
 | |
| 	if(!file_exists($afmfile))
 | |
| 		die('<b>Error:</b> AFM file not found: '.$afmfile);
 | |
| 	$fm=ReadAFM($afmfile,$map);
 | |
| 	if($enc)
 | |
| 		$diff=MakeFontEncoding($map);
 | |
| 	else
 | |
| 		$diff='';
 | |
| 	$fd=MakeFontDescriptor($fm,empty($map));
 | |
| 	//Find font type
 | |
| 	if($fontfile)
 | |
| 	{
 | |
| 		$ext=strtolower(substr($fontfile,-3));
 | |
| 		if($ext=='ttf')
 | |
| 			$type='TrueType';
 | |
| 		elseif($ext=='pfb')
 | |
| 			$type='Type1';
 | |
| 		else
 | |
| 			die('<b>Error:</b> unrecognized font file extension: '.$ext);
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		if($type!='TrueType' && $type!='Type1')
 | |
| 			die('<b>Error:</b> incorrect font type: '.$type);
 | |
| 	}
 | |
| 	//Start generation
 | |
| 	$s='<?php'."\n";
 | |
| 	$s.='$type=\''.$type."';\n";
 | |
| 	$s.='$name=\''.$fm['FontName']."';\n";
 | |
| 	$s.='$desc='.$fd.";\n";
 | |
| 	if(!isset($fm['UnderlinePosition']))
 | |
| 		$fm['UnderlinePosition']=-100;
 | |
| 	if(!isset($fm['UnderlineThickness']))
 | |
| 		$fm['UnderlineThickness']=50;
 | |
| 	$s.='$up='.$fm['UnderlinePosition'].";\n";
 | |
| 	$s.='$ut='.$fm['UnderlineThickness'].";\n";
 | |
| 	$w=MakeWidthArray($fm);
 | |
| 	$s.='$cw='.$w.";\n";
 | |
| 	$s.='$enc=\''.$enc."';\n";
 | |
| 	$s.='$diff=\''.$diff."';\n";
 | |
| 	$basename=substr(basename($afmfile),0,-4);
 | |
| 	if($fontfile)
 | |
| 	{
 | |
| 		//Embedded font
 | |
| 		if(!file_exists($fontfile))
 | |
| 			die('<b>Error:</b> font file not found: '.$fontfile);
 | |
| 		if($type=='TrueType')
 | |
| 			CheckTTF($fontfile);
 | |
| 		$f=fopen($fontfile,'rb');
 | |
| 		if(!$f)
 | |
| 			die('<b>Error:</b> Can\'t open '.$fontfile);
 | |
| 		$file=fread($f,filesize($fontfile));
 | |
| 		fclose($f);
 | |
| 		if($type=='Type1')
 | |
| 		{
 | |
| 			//Find first two sections and discard third one
 | |
| 			$header=(ord($file[0])==128);
 | |
| 			if($header)
 | |
| 			{
 | |
| 				//Strip first binary header
 | |
| 				$file=substr($file,6);
 | |
| 			}
 | |
| 			$pos=strpos($file,'eexec');
 | |
| 			if(!$pos)
 | |
| 				die('<b>Error:</b> font file does not seem to be valid Type1');
 | |
| 			$size1=$pos+6;
 | |
| 			if($header && ord($file[$size1])==128)
 | |
| 			{
 | |
| 				//Strip second binary header
 | |
| 				$file=substr($file,0,$size1).substr($file,$size1+6);
 | |
| 			}
 | |
| 			$pos=strpos($file,'00000000');
 | |
| 			if(!$pos)
 | |
| 				die('<b>Error:</b> font file does not seem to be valid Type1');
 | |
| 			$size2=$pos-$size1;
 | |
| 			$file=substr($file,0,$size1+$size2);
 | |
| 		}
 | |
| 		if(function_exists('gzcompress'))
 | |
| 		{
 | |
| 			$cmp=$basename.'.z';
 | |
| 			SaveToFile($cmp,gzcompress($file),'b');
 | |
| 			$s.='$file=\''.$cmp."';\n";
 | |
| 			echo 'Font file compressed ('.$cmp.')<br>';
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			$s.='$file=\''.basename($fontfile)."';\n";
 | |
| 			echo '<b>Notice:</b> font file could not be compressed (zlib extension not available)<br>';
 | |
| 		}
 | |
| 		if($type=='Type1')
 | |
| 		{
 | |
| 			$s.='$size1='.$size1.";\n";
 | |
| 			$s.='$size2='.$size2.";\n";
 | |
| 		}
 | |
| 		else
 | |
| 			$s.='$originalsize='.filesize($fontfile).";\n";
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		//Not embedded font
 | |
| 		$s.='$file='."'';\n";
 | |
| 	}
 | |
| 	$s.="?>\n";
 | |
| 	SaveToFile($basename.'.php',$s,'t');
 | |
| 	echo 'Font definition file generated ('.$basename.'.php'.')<br>';
 | |
| }
 | |
| ?>
 |