[Top][All Lists]
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Phpcompta-dev] r4218 - in phpcompta/trunk/include/tfpdf: . font font/un
From: |
phpcompta-dev |
Subject: |
[Phpcompta-dev] r4218 - in phpcompta/trunk/include/tfpdf: . font font/unifont |
Date: |
Sat, 15 Oct 2011 03:02:42 +0200 (CEST) |
Author: danydb
Date: 2011-10-15 03:02:41 +0200 (Sat, 15 Oct 2011)
New Revision: 4218
Modified:
phpcompta/trunk/include/tfpdf/font/courier.php
phpcompta/trunk/include/tfpdf/font/helvetica.php
phpcompta/trunk/include/tfpdf/font/helveticab.php
phpcompta/trunk/include/tfpdf/font/helveticabi.php
phpcompta/trunk/include/tfpdf/font/helveticai.php
phpcompta/trunk/include/tfpdf/font/symbol.php
phpcompta/trunk/include/tfpdf/font/times.php
phpcompta/trunk/include/tfpdf/font/timesb.php
phpcompta/trunk/include/tfpdf/font/timesbi.php
phpcompta/trunk/include/tfpdf/font/timesi.php
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-Bold.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-BoldOblique.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-ExtraLight.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-Oblique.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed-Bold.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed-BoldOblique.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed-Oblique.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono-Bold.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono-BoldOblique.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono-Oblique.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif-Bold.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif-BoldItalic.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif-Italic.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed-Bold.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed-BoldItalic.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed-Italic.ttf
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed.ttf
phpcompta/trunk/include/tfpdf/font/unifont/ttfonts.php
phpcompta/trunk/include/tfpdf/font/zapfdingbats.php
phpcompta/trunk/include/tfpdf/tfpdf.php
Log:
update tfpdf to version 1.24
Must be more tested
Modified: phpcompta/trunk/include/tfpdf/font/courier.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/courier.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/courier.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,7 +1,8 @@
<?php
+$type = 'Core';
+$name = 'Courier';
+$up = -100;
+$ut = 50;
for($i=0;$i<=255;$i++)
- $fpdf_charwidths['courier'][chr($i)]=600;
-$fpdf_charwidths['courierB']=$fpdf_charwidths['courier'];
-$fpdf_charwidths['courierI']=$fpdf_charwidths['courier'];
-$fpdf_charwidths['courierBI']=$fpdf_charwidths['courier'];
+ $cw[chr($i)] = 600;
?>
Modified: phpcompta/trunk/include/tfpdf/font/helvetica.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/helvetica.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/helvetica.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['helvetica']=array(
+$type = 'Core';
+$name = 'Helvetica';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>278,chr(1)=>278,chr(2)=>278,chr(3)=>278,chr(4)=>278,chr(5)=>278,chr(6)=>278,chr(7)=>278,chr(8)=>278,chr(9)=>278,chr(10)=>278,chr(11)=>278,chr(12)=>278,chr(13)=>278,chr(14)=>278,chr(15)=>278,chr(16)=>278,chr(17)=>278,chr(18)=>278,chr(19)=>278,chr(20)=>278,chr(21)=>278,
chr(22)=>278,chr(23)=>278,chr(24)=>278,chr(25)=>278,chr(26)=>278,chr(27)=>278,chr(28)=>278,chr(29)=>278,chr(30)=>278,chr(31)=>278,'
'=>278,'!'=>278,'"'=>355,'#'=>556,'$'=>556,'%'=>889,'&'=>667,'\''=>191,'('=>333,')'=>333,'*'=>389,'+'=>584,
','=>278,'-'=>333,'.'=>278,'/'=>278,'0'=>556,'1'=>556,'2'=>556,'3'=>556,'4'=>556,'5'=>556,'6'=>556,'7'=>556,'8'=>556,'9'=>556,':'=>278,';'=>278,'<'=>584,'='=>584,'>'=>584,'?'=>556,'@'=>1015,'A'=>667,
Modified: phpcompta/trunk/include/tfpdf/font/helveticab.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/helveticab.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/helveticab.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['helveticaB']=array(
+$type = 'Core';
+$name = 'Helvetica-Bold';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>278,chr(1)=>278,chr(2)=>278,chr(3)=>278,chr(4)=>278,chr(5)=>278,chr(6)=>278,chr(7)=>278,chr(8)=>278,chr(9)=>278,chr(10)=>278,chr(11)=>278,chr(12)=>278,chr(13)=>278,chr(14)=>278,chr(15)=>278,chr(16)=>278,chr(17)=>278,chr(18)=>278,chr(19)=>278,chr(20)=>278,chr(21)=>278,
chr(22)=>278,chr(23)=>278,chr(24)=>278,chr(25)=>278,chr(26)=>278,chr(27)=>278,chr(28)=>278,chr(29)=>278,chr(30)=>278,chr(31)=>278,'
'=>278,'!'=>333,'"'=>474,'#'=>556,'$'=>556,'%'=>889,'&'=>722,'\''=>238,'('=>333,')'=>333,'*'=>389,'+'=>584,
','=>278,'-'=>333,'.'=>278,'/'=>278,'0'=>556,'1'=>556,'2'=>556,'3'=>556,'4'=>556,'5'=>556,'6'=>556,'7'=>556,'8'=>556,'9'=>556,':'=>333,';'=>333,'<'=>584,'='=>584,'>'=>584,'?'=>611,'@'=>975,'A'=>722,
Modified: phpcompta/trunk/include/tfpdf/font/helveticabi.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/helveticabi.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/helveticabi.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['helveticaBI']=array(
+$type = 'Core';
+$name = 'Helvetica-BoldOblique';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>278,chr(1)=>278,chr(2)=>278,chr(3)=>278,chr(4)=>278,chr(5)=>278,chr(6)=>278,chr(7)=>278,chr(8)=>278,chr(9)=>278,chr(10)=>278,chr(11)=>278,chr(12)=>278,chr(13)=>278,chr(14)=>278,chr(15)=>278,chr(16)=>278,chr(17)=>278,chr(18)=>278,chr(19)=>278,chr(20)=>278,chr(21)=>278,
chr(22)=>278,chr(23)=>278,chr(24)=>278,chr(25)=>278,chr(26)=>278,chr(27)=>278,chr(28)=>278,chr(29)=>278,chr(30)=>278,chr(31)=>278,'
'=>278,'!'=>333,'"'=>474,'#'=>556,'$'=>556,'%'=>889,'&'=>722,'\''=>238,'('=>333,')'=>333,'*'=>389,'+'=>584,
','=>278,'-'=>333,'.'=>278,'/'=>278,'0'=>556,'1'=>556,'2'=>556,'3'=>556,'4'=>556,'5'=>556,'6'=>556,'7'=>556,'8'=>556,'9'=>556,':'=>333,';'=>333,'<'=>584,'='=>584,'>'=>584,'?'=>611,'@'=>975,'A'=>722,
Modified: phpcompta/trunk/include/tfpdf/font/helveticai.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/helveticai.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/helveticai.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['helveticaI']=array(
+$type = 'Core';
+$name = 'Helvetica-Oblique';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>278,chr(1)=>278,chr(2)=>278,chr(3)=>278,chr(4)=>278,chr(5)=>278,chr(6)=>278,chr(7)=>278,chr(8)=>278,chr(9)=>278,chr(10)=>278,chr(11)=>278,chr(12)=>278,chr(13)=>278,chr(14)=>278,chr(15)=>278,chr(16)=>278,chr(17)=>278,chr(18)=>278,chr(19)=>278,chr(20)=>278,chr(21)=>278,
chr(22)=>278,chr(23)=>278,chr(24)=>278,chr(25)=>278,chr(26)=>278,chr(27)=>278,chr(28)=>278,chr(29)=>278,chr(30)=>278,chr(31)=>278,'
'=>278,'!'=>278,'"'=>355,'#'=>556,'$'=>556,'%'=>889,'&'=>667,'\''=>191,'('=>333,')'=>333,'*'=>389,'+'=>584,
','=>278,'-'=>333,'.'=>278,'/'=>278,'0'=>556,'1'=>556,'2'=>556,'3'=>556,'4'=>556,'5'=>556,'6'=>556,'7'=>556,'8'=>556,'9'=>556,':'=>278,';'=>278,'<'=>584,'='=>584,'>'=>584,'?'=>556,'@'=>1015,'A'=>667,
Modified: phpcompta/trunk/include/tfpdf/font/symbol.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/symbol.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/symbol.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['symbol']=array(
+$type = 'Core';
+$name = 'Symbol';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>250,chr(1)=>250,chr(2)=>250,chr(3)=>250,chr(4)=>250,chr(5)=>250,chr(6)=>250,chr(7)=>250,chr(8)=>250,chr(9)=>250,chr(10)=>250,chr(11)=>250,chr(12)=>250,chr(13)=>250,chr(14)=>250,chr(15)=>250,chr(16)=>250,chr(17)=>250,chr(18)=>250,chr(19)=>250,chr(20)=>250,chr(21)=>250,
chr(22)=>250,chr(23)=>250,chr(24)=>250,chr(25)=>250,chr(26)=>250,chr(27)=>250,chr(28)=>250,chr(29)=>250,chr(30)=>250,chr(31)=>250,'
'=>250,'!'=>333,'"'=>713,'#'=>500,'$'=>549,'%'=>833,'&'=>778,'\''=>439,'('=>333,')'=>333,'*'=>500,'+'=>549,
','=>250,'-'=>549,'.'=>250,'/'=>278,'0'=>500,'1'=>500,'2'=>500,'3'=>500,'4'=>500,'5'=>500,'6'=>500,'7'=>500,'8'=>500,'9'=>500,':'=>278,';'=>278,'<'=>549,'='=>549,'>'=>549,'?'=>444,'@'=>549,'A'=>722,
Modified: phpcompta/trunk/include/tfpdf/font/times.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/times.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/times.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['times']=array(
+$type = 'Core';
+$name = 'Times-Roman';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>250,chr(1)=>250,chr(2)=>250,chr(3)=>250,chr(4)=>250,chr(5)=>250,chr(6)=>250,chr(7)=>250,chr(8)=>250,chr(9)=>250,chr(10)=>250,chr(11)=>250,chr(12)=>250,chr(13)=>250,chr(14)=>250,chr(15)=>250,chr(16)=>250,chr(17)=>250,chr(18)=>250,chr(19)=>250,chr(20)=>250,chr(21)=>250,
chr(22)=>250,chr(23)=>250,chr(24)=>250,chr(25)=>250,chr(26)=>250,chr(27)=>250,chr(28)=>250,chr(29)=>250,chr(30)=>250,chr(31)=>250,'
'=>250,'!'=>333,'"'=>408,'#'=>500,'$'=>500,'%'=>833,'&'=>778,'\''=>180,'('=>333,')'=>333,'*'=>500,'+'=>564,
','=>250,'-'=>333,'.'=>250,'/'=>278,'0'=>500,'1'=>500,'2'=>500,'3'=>500,'4'=>500,'5'=>500,'6'=>500,'7'=>500,'8'=>500,'9'=>500,':'=>278,';'=>278,'<'=>564,'='=>564,'>'=>564,'?'=>444,'@'=>921,'A'=>722,
Modified: phpcompta/trunk/include/tfpdf/font/timesb.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/timesb.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/timesb.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['timesB']=array(
+$type = 'Core';
+$name = 'Times-Bold';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>250,chr(1)=>250,chr(2)=>250,chr(3)=>250,chr(4)=>250,chr(5)=>250,chr(6)=>250,chr(7)=>250,chr(8)=>250,chr(9)=>250,chr(10)=>250,chr(11)=>250,chr(12)=>250,chr(13)=>250,chr(14)=>250,chr(15)=>250,chr(16)=>250,chr(17)=>250,chr(18)=>250,chr(19)=>250,chr(20)=>250,chr(21)=>250,
chr(22)=>250,chr(23)=>250,chr(24)=>250,chr(25)=>250,chr(26)=>250,chr(27)=>250,chr(28)=>250,chr(29)=>250,chr(30)=>250,chr(31)=>250,'
'=>250,'!'=>333,'"'=>555,'#'=>500,'$'=>500,'%'=>1000,'&'=>833,'\''=>278,'('=>333,')'=>333,'*'=>500,'+'=>570,
','=>250,'-'=>333,'.'=>250,'/'=>278,'0'=>500,'1'=>500,'2'=>500,'3'=>500,'4'=>500,'5'=>500,'6'=>500,'7'=>500,'8'=>500,'9'=>500,':'=>333,';'=>333,'<'=>570,'='=>570,'>'=>570,'?'=>500,'@'=>930,'A'=>722,
Modified: phpcompta/trunk/include/tfpdf/font/timesbi.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/timesbi.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/timesbi.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['timesBI']=array(
+$type = 'Core';
+$name = 'Times-BoldItalic';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>250,chr(1)=>250,chr(2)=>250,chr(3)=>250,chr(4)=>250,chr(5)=>250,chr(6)=>250,chr(7)=>250,chr(8)=>250,chr(9)=>250,chr(10)=>250,chr(11)=>250,chr(12)=>250,chr(13)=>250,chr(14)=>250,chr(15)=>250,chr(16)=>250,chr(17)=>250,chr(18)=>250,chr(19)=>250,chr(20)=>250,chr(21)=>250,
chr(22)=>250,chr(23)=>250,chr(24)=>250,chr(25)=>250,chr(26)=>250,chr(27)=>250,chr(28)=>250,chr(29)=>250,chr(30)=>250,chr(31)=>250,'
'=>250,'!'=>389,'"'=>555,'#'=>500,'$'=>500,'%'=>833,'&'=>778,'\''=>278,'('=>333,')'=>333,'*'=>500,'+'=>570,
','=>250,'-'=>333,'.'=>250,'/'=>278,'0'=>500,'1'=>500,'2'=>500,'3'=>500,'4'=>500,'5'=>500,'6'=>500,'7'=>500,'8'=>500,'9'=>500,':'=>333,';'=>333,'<'=>570,'='=>570,'>'=>570,'?'=>500,'@'=>832,'A'=>667,
Modified: phpcompta/trunk/include/tfpdf/font/timesi.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/timesi.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/timesi.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['timesI']=array(
+$type = 'Core';
+$name = 'Times-Italic';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>250,chr(1)=>250,chr(2)=>250,chr(3)=>250,chr(4)=>250,chr(5)=>250,chr(6)=>250,chr(7)=>250,chr(8)=>250,chr(9)=>250,chr(10)=>250,chr(11)=>250,chr(12)=>250,chr(13)=>250,chr(14)=>250,chr(15)=>250,chr(16)=>250,chr(17)=>250,chr(18)=>250,chr(19)=>250,chr(20)=>250,chr(21)=>250,
chr(22)=>250,chr(23)=>250,chr(24)=>250,chr(25)=>250,chr(26)=>250,chr(27)=>250,chr(28)=>250,chr(29)=>250,chr(30)=>250,chr(31)=>250,'
'=>250,'!'=>333,'"'=>420,'#'=>500,'$'=>500,'%'=>833,'&'=>778,'\''=>214,'('=>333,')'=>333,'*'=>500,'+'=>675,
','=>250,'-'=>333,'.'=>250,'/'=>278,'0'=>500,'1'=>500,'2'=>500,'3'=>500,'4'=>500,'5'=>500,'6'=>500,'7'=>500,'8'=>500,'9'=>500,':'=>333,';'=>333,'<'=>675,'='=>675,'>'=>675,'?'=>500,'@'=>920,'A'=>611,
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-Bold.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-BoldOblique.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-ExtraLight.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans-Oblique.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSans.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed-Bold.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed-BoldOblique.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed-Oblique.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansCondensed.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono-Bold.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono-BoldOblique.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono-Oblique.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSansMono.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif-Bold.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif-BoldItalic.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif-Italic.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerif.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed-Bold.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed-BoldItalic.ttf
===================================================================
(Binary files differ)
Modified:
phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed-Italic.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/DejaVuSerifCondensed.ttf
===================================================================
(Binary files differ)
Modified: phpcompta/trunk/include/tfpdf/font/unifont/ttfonts.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/unifont/ttfonts.php 2011-10-14
23:00:50 UTC (rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/unifont/ttfonts.php 2011-10-15
01:02:41 UTC (rev 4218)
@@ -8,8 +8,8 @@
* written in Python - http://www.reportlab.com/software/opensource/
*
* together with ideas from the OpenOffice source code and others.
*
*
*
-* Version: 1.02
*
-* Date: 2010-06-07
*
+* Version: 1.04
*
+* Date: 2011-09-18
*
* Author: Ian Back <address@hidden>
*
* License: LGPL
*
* Copyright (c) Ian Back, 2010
*
@@ -18,7 +18,14 @@
*
*
*******************************************************************************/
+// Define the value used in the "head" table of a created TTF file
+// 0x74727565 "true" for Mac
+// 0x00010000 for Windows
+// Either seems to work for a font embedded in a PDF file
+// when read by Adobe Reader on a Windows PC(!)
+define("_TTF_MAC_HEADER", false);
+
// TrueType Font Glyph operators
define("GF_WORDS",(1 << 0));
define("GF_SCALE",(1 << 3));
@@ -30,6 +37,7 @@
class TTFontFile {
+var $maxUni;
var $_pos;
var $numTables;
var $searchRange;
@@ -66,44 +74,38 @@
}
- function getMetrics($file, $presubset=0) {
+ function getMetrics($file) {
$this->filename = $file;
$this->fh = fopen($file,'rb') or die('Can\'t open file ' .
$file);
$this->_pos = 0;
- $this->charWidths = array();
- $this->hmetrics = array();
+ $this->charWidths = '';
$this->glyphPos = array();
$this->charToGlyph = array();
$this->tables = array();
$this->otables = array();
$this->ascent = 0;
$this->descent = 0;
- if ($presubset) { $this->skip(4); }
- else { $this->readHeader(); }
- $this->readTableDirectory($presubset);
- $this->extractInfo($presubset);
- fclose($this->fh);
- }
-
- function readHeader() {
- // read the sfnt header at the current position
+ $this->TTCFonts = array();
$this->version = $version = $this->read_ulong();
if ($version==0x4F54544F)
die("Postscript outlines are not supported");
if ($version==0x74746366)
- die("TTC Font collections are not supported");
+ die("ERROR - TrueType Fonts Collections not supported");
if (!in_array($version, array(0x00010000,0x74727565)))
die("Not a TrueType font: version=".$version);
- return true;
+ $this->readTableDirectory();
+ $this->extractInfo();
+ fclose($this->fh);
}
- function readTableDirectory($presubset=1) {
- $this->numTables = $this->read_ushort();
+
+ function readTableDirectory() {
+ $this->numTables = $this->read_ushort();
$this->searchRange = $this->read_ushort();
$this->entrySelector = $this->read_ushort();
$this->rangeShift = $this->read_ushort();
$this->tables = array();
- for ($i=0;$i<$this->numTables;$i++) { // 1.02
+ for ($i=0;$i<$this->numTables;$i++) {
$record = array();
$record['tag'] = $this->read_tag();
$record['checksum'] =
array($this->read_ushort(),$this->read_ushort());
@@ -111,27 +113,8 @@
$record['length'] = $this->read_ulong();
$this->tables[$record['tag']] = $record;
}
- if (!$presubset) $this->checksumTables();
}
- function checksumTables() {
- // Check the checksums for all tables
- foreach($this->tables AS $t) {
- if ($t['length'] > 0 && $t['length'] < $this->maxStrLenRead)
{ // 1.02
- $table = $this->get_chunk($t['offset'], $t['length']);
- $checksum = $this->calcChecksum($table);
- if ($t['tag'] == 'head') {
- $up = unpack('n*', substr($table,8,4));
- $adjustment[0] = $up[1];
- $adjustment[1] = $up[2];
- $checksum = $this->sub32($checksum, $adjustment);
- }
- $xchecksum = $t['checksum'];
- if ($xchecksum != $checksum)
- die(sprintf('TTF file "%s": invalid checksum %s table: %s
(expected %s)',
$this->filename,dechex($checksum[0]).dechex($checksum[1]),$t['tag'],dechex($xchecksum[0]).dechex($xchecksum[1])));
- }
- }
- }
function sub32($x, $y) {
$xlo = $x[1];
@@ -196,6 +179,14 @@
return $a;
}
+ function unpack_short($s) {
+ $a = (ord($s[0])<<8) + ord($s[1]);
+ if ($a & (1 << 15) ) {
+ $a = ($a - (1 << 16));
+ }
+ return $a;
+ }
+
function read_ushort() {
$this->_pos += 2;
$s = fread($this->fh,2);
@@ -240,15 +231,25 @@
return $this->splice($stream, $offset, $up);
}
+ function _set_short($stream, $offset, $val) {
+ if ($val<0) {
+ $val = abs($val);
+ $val = ~$val;
+ $val += 1;
+ }
+ $up = pack("n",$val);
+ return $this->splice($stream, $offset, $up);
+ }
+
function get_chunk($pos, $length) {
fseek($this->fh,$pos);
- if ($length <1) { return ''; } // 1.02
+ if ($length <1) { return ''; }
return (fread($this->fh,$length));
}
function get_table($tag) {
list($pos, $length) = $this->get_table_pos($tag);
- if ($length == 0) { die('Truetype font ('.$this->filename.'):
error reading table: '.$tag); } // 1.02
+ if ($length == 0) { die('Truetype font ('.$this->filename.'):
error reading table: '.$tag); }
fseek($this->fh,$pos);
return (fread($this->fh,$length));
}
@@ -263,22 +264,23 @@
/////////////////////////////////////////////////////////////////////////////////////////
+/////////////////////////////////////////////////////////////////////////////////////////
+/////////////////////////////////////////////////////////////////////////////////////////
- function extractInfo($presubset=0) {
- // = 0 ; validate; does not need glyphPos or charToGlyph
- // presubset = 1; no validation; does not need name and other
metrics
+ function extractInfo() {
///////////////////////////////////
// name - Naming table
///////////////////////////////////
- if (!$presubset) {
+ $this->sFamilyClass = 0;
+ $this->sFamilySubClass = 0;
+
$name_offset = $this->seek_table("name");
$format = $this->read_ushort();
if ($format != 0)
die("Unknown name table format ".$format);
$numRecords = $this->read_ushort();
$string_data_offset = $name_offset +
$this->read_ushort();
-
$names = array(1=>'',2=>'',3=>'',4=>'',6=>'');
$K = array_keys($names);
$nameCount = count($names);
@@ -300,7 +302,7 @@
$N = '';
while ($length > 0) {
$char = $this->read_ushort();
- $N .= (chr($char)); // 1.02
+ $N .= (chr($char));
$length -= 1;
}
$this->_pos = $opos;
@@ -319,7 +321,7 @@
}
}
if ($names[6])
- $psName = preg_replace('/ /','-',$names[6]);
+ $psName = $names[6];
else if ($names[4])
$psName = preg_replace('/ /','-',$names[4]);
else if ($names[1])
@@ -328,49 +330,27 @@
$psName = '';
if (!$psName)
die("Could not find PostScript font name");
- for ($i=0;$i<strlen($psName);$i++) {
- $c = $psName{$i}; // 1.02
- $oc = ord($c);
- if ($oc>126 || strpos(' [](){}<>/%',$c)!==false)
- die("psName=".$psName." contains
invalid character ".$c." ie U+".ord(c));
- }
$this->name = $psName;
if ($names[1]) { $this->familyName = $names[1]; } else
{ $this->familyName = $psName; }
if ($names[2]) { $this->styleName = $names[2]; } else {
$this->styleName = 'Regular'; }
if ($names[4]) { $this->fullName = $names[4]; } else {
$this->fullName = $psName; }
if ($names[3]) { $this->uniqueFontID = $names[3]; }
else { $this->uniqueFontID = $psName; }
- }
+ if ($names[6]) { $this->fullName = $names[6]; }
///////////////////////////////////
// head - Font header table
///////////////////////////////////
$this->seek_table("head");
- if ($presubset) { $this->skip(18); }
- else {
- $ver_maj = $this->read_ushort();
- $ver_min = $this->read_ushort();
- if ($ver_maj != 1)
- die('Unknown head table version '. $ver_maj
.'.'. $ver_min);
- $this->fontRevision = $this->read_ushort() .
$this->read_ushort();
-
- $this->skip(4);
- $magic = $this->read_ulong();
- if ($magic != 0x5F0F3CF5)
- die('Invalid head table magic ' .$magic);
- $this->skip(2);
- }
- $this->unitsPerEm = $unitsPerEm = $this->read_ushort(); // 1.02
- $scale = 1000 / $unitsPerEm; // 1.02
- if ($presubset) { $this->skip(30); } // 1.02
- else {
- $this->skip(16);
- $xMin = $this->read_short();
- $yMin = $this->read_short();
- $xMax = $this->read_short();
- $yMax = $this->read_short();
- $this->bbox = array(($xMin*$scale), ($yMin*$scale),
($xMax*$scale), ($yMax*$scale));
- $this->skip(3*2);
- }
+ $this->skip(18);
+ $this->unitsPerEm = $unitsPerEm = $this->read_ushort();
+ $scale = 1000 / $unitsPerEm;
+ $this->skip(16);
+ $xMin = $this->read_short();
+ $yMin = $this->read_short();
+ $xMax = $this->read_short();
+ $yMax = $this->read_short();
+ $this->bbox = array(($xMin*$scale), ($yMin*$scale),
($xMax*$scale), ($yMax*$scale));
+ $this->skip(3*2);
$indexToLocFormat = $this->read_ushort();
$glyphDataFormat = $this->read_ushort();
if ($glyphDataFormat != 0)
@@ -399,9 +379,17 @@
$usWeightClass = $this->read_ushort();
$this->skip(2);
$fsType = $this->read_ushort();
- // if ($fsType == 0x0002 || ($fsType & 0x0300) != 0)
- // die('Font does not allow subsetting/embedding');
- $this->skip(58); //11*2 + 10 + 4*4 + 4 + 3*2
+ if ($fsType == 0x0002 || ($fsType & 0x0300) != 0) {
+ die('ERROR - Font file '.$this->filename.'
cannot be embedded due to copyright restrictions.');
+ $this->restrictedUse = true;
+ }
+ $this->skip(20);
+ $sF = $this->read_short();
+ $this->sFamilyClass = ($sF >> 8);
+ $this->sFamilySubClass = ($sF & 0xFF);
+ $this->_pos += 10; //PANOSE = 10 byte length
+ $panose = fread($this->fh,10);
+ $this->skip(26);
$sTypoAscender = $this->read_short();
$sTypoDescender = $this->read_short();
if (!$this->ascent) $this->ascent =
($sTypoAscender*$scale);
@@ -427,13 +415,7 @@
// post - PostScript table
///////////////////////////////////
$this->seek_table("post");
- if ($presubset) { $this->skip(4); }
- else {
- $ver_maj = $this->read_ushort();
- $ver_min = $this->read_ushort();
- if ($ver_maj <1 || $ver_maj >4)
- die('Unknown post table version '.$ver_maj);
- }
+ $this->skip(4);
$this->italicAngle = $this->read_short() + $this->read_ushort()
/ 65536.0;
$this->underlinePosition = $this->read_short() * $scale;
$this->underlineThickness = $this->read_short() * $scale;
@@ -452,14 +434,7 @@
// hhea - Horizontal header table
///////////////////////////////////
$this->seek_table("hhea");
- if ($presubset) { $this->skip(32); }
- else {
- $ver_maj = $this->read_ushort();
- $ver_min = $this->read_ushort();
- if ($ver_maj != 1)
- die('Unknown hhea table version '.$ver_maj);
- $this->skip(28);
- }
+ $this->skip(32);
$metricDataFormat = $this->read_ushort();
if ($metricDataFormat != 0)
die('Unknown horizontal metric data format
'.$metricDataFormat);
@@ -471,15 +446,10 @@
// maxp - Maximum profile table
///////////////////////////////////
$this->seek_table("maxp");
- if ($presubset) { $this->skip(4); }
- else {
- $ver_maj = $this->read_ushort();
- $ver_min = $this->read_ushort();
- if ($ver_maj != 1)
- die('Unknown maxp table version '.$ver_maj);
- }
+ $this->skip(4);
$numGlyphs = $this->read_ushort();
+
///////////////////////////////////
// cmap - Character to glyph index mapping table
///////////////////////////////////
@@ -492,14 +462,88 @@
$encodingID = $this->read_ushort();
$offset = $this->read_ulong();
$save_pos = $this->_pos;
- if ($platformID == 3 && $encodingID == 1) { //
Microsoft, Unicode
+ if (($platformID == 3 && $encodingID == 1) ||
$platformID == 0) { // Microsoft, Unicode
$format = $this->get_ushort($cmap_offset +
$offset);
if ($format == 4) {
- $unicode_cmap_offset = $cmap_offset +
$offset;
+ if (!$unicode_cmap_offset)
$unicode_cmap_offset = $cmap_offset + $offset;
break;
}
}
- else if ($platformID == 0) { // Unicode -- assume all
encodings are compatible
+ $this->seek($save_pos );
+ }
+ if (!$unicode_cmap_offset)
+ die('Font ('.$this->filename .') does not have cmap for
Unicode (platform 3, encoding 1, format 4, or platform 0, any encoding, format
4)');
+
+
+ $glyphToChar = array();
+ $charToGlyph = array();
+ $this->getCMAP4($unicode_cmap_offset, $glyphToChar,
$charToGlyph );
+
+ ///////////////////////////////////
+ // hmtx - Horizontal metrics table
+ ///////////////////////////////////
+ $this->getHMTX($numberOfHMetrics, $numGlyphs, $glyphToChar,
$scale);
+
+ }
+
+
+/////////////////////////////////////////////////////////////////////////////////////////
+/////////////////////////////////////////////////////////////////////////////////////////
+
+
+ function makeSubset($file, &$subset) {
+ $this->filename = $file;
+ $this->fh = fopen($file ,'rb') or die('Can\'t open file ' .
$file);
+ $this->_pos = 0;
+ $this->charWidths = '';
+ $this->glyphPos = array();
+ $this->charToGlyph = array();
+ $this->tables = array();
+ $this->otables = array();
+ $this->ascent = 0;
+ $this->descent = 0;
+ $this->skip(4);
+ $this->maxUni = 0;
+ $this->readTableDirectory();
+
+
+ ///////////////////////////////////
+ // head - Font header table
+ ///////////////////////////////////
+ $this->seek_table("head");
+ $this->skip(50);
+ $indexToLocFormat = $this->read_ushort();
+ $glyphDataFormat = $this->read_ushort();
+
+ ///////////////////////////////////
+ // hhea - Horizontal header table
+ ///////////////////////////////////
+ $this->seek_table("hhea");
+ $this->skip(32);
+ $metricDataFormat = $this->read_ushort();
+ $orignHmetrics = $numberOfHMetrics = $this->read_ushort();
+
+ ///////////////////////////////////
+ // maxp - Maximum profile table
+ ///////////////////////////////////
+ $this->seek_table("maxp");
+ $this->skip(4);
+ $numGlyphs = $this->read_ushort();
+
+
+ ///////////////////////////////////
+ // cmap - Character to glyph index mapping table
+ ///////////////////////////////////
+ $cmap_offset = $this->seek_table("cmap");
+ $this->skip(2);
+ $cmapTableCount = $this->read_ushort();
+ $unicode_cmap_offset = 0;
+ for ($i=0;$i<$cmapTableCount;$i++) {
+ $platformID = $this->read_ushort();
+ $encodingID = $this->read_ushort();
+ $offset = $this->read_ulong();
+ $save_pos = $this->_pos;
+ if (($platformID == 3 && $encodingID == 1) ||
$platformID == 0) { // Microsoft, Unicode
$format = $this->get_ushort($cmap_offset +
$offset);
if ($format == 4) {
$unicode_cmap_offset = $cmap_offset +
$offset;
@@ -510,166 +554,68 @@
}
if (!$unicode_cmap_offset)
- die('Font does not have cmap for Unicode (platform 3,
encoding 1, format 4, or platform 0, any encoding, format 4)');
- $this->seek($unicode_cmap_offset + 2);
- $length = $this->read_ushort();
- $limit = $unicode_cmap_offset + $length;
- $this->skip(2);
+ die('Font ('.$this->filename .') does not have cmap for
Unicode (platform 3, encoding 1, format 4, or platform 0, any encoding, format
4)');
- $segCount = $this->read_ushort() / 2;
- $this->skip(6);
- $endCount = array();
- for($i=0; $i<$segCount; $i++) { $endCount[] =
$this->read_ushort(); }
- $this->skip(2);
- $startCount = array();
- for($i=0; $i<$segCount; $i++) { $startCount[] =
$this->read_ushort(); }
- $idDelta = array();
- for($i=0; $i<$segCount; $i++) { $idDelta[] =
$this->read_short(); } // ???? was unsigned short
- $idRangeOffset_start = $this->_pos;
- $idRangeOffset = array();
- for($i=0; $i<$segCount; $i++) { $idRangeOffset[] =
$this->read_ushort(); }
$glyphToChar = array();
$charToGlyph = array();
- for ($n=0;$n<$segCount;$n++) {
- for ($unichar=$startCount[$n];$unichar<($endCount[$n] +
1);$unichar++) {
- if ($idRangeOffset[$n] == 0)
- $glyph = ($unichar + $idDelta[$n]) &
0xFFFF;
- else {
- $offset = ($unichar - $startCount[$n])
* 2 + $idRangeOffset[$n];
- $offset = $idRangeOffset_start + 2 * $n
+ $offset;
- if ($offset >= $limit)
- $glyph = 0;
- else {
- $glyph =
$this->get_ushort($offset);
- if ($glyph != 0)
- $glyph = ($glyph +
$idDelta[$n]) & 0xFFFF;
- }
- }
- if ($presubset) $charToGlyph[$unichar] = $glyph;
- $glyphToChar[$glyph][] = $unichar;
- }
- }
- if ($presubset) $this->charToGlyph = $charToGlyph;
+ $this->getCMAP4($unicode_cmap_offset, $glyphToChar,
$charToGlyph );
+ $this->charToGlyph = $charToGlyph;
+
///////////////////////////////////
// hmtx - Horizontal metrics table
///////////////////////////////////
- $this->seek_table("hmtx");
- $aw = 0;
- $this->charWidths = array();
- $this->hmetrics = array();
- for( $glyph=0; $glyph<$numberOfHMetrics; $glyph++) {
- $aw = $this->read_ushort();
- $lsb = $this->read_short();
- $this->hmetrics[] = array($aw, $lsb);
- $aw = $scale*$aw;
- if ($glyph == 0)
- $this->defaultWidth = $aw;
- if (isset($glyphToChar[$glyph])) {
- foreach($glyphToChar[$glyph] AS $char) {
- $this->charWidths[$char] = round($aw);
- }
- }
- }
- for( $glyph=$numberOfHMetrics; $glyph<$numGlyphs; $glyph++) {
- $lsb = $this->read_ushort();
- $this->hmetrics[] = array($aw, $lsb);
- if (isset($glyphToChar[$glyph])) {
- foreach($glyphToChar[$glyph] AS $char) {
- $this->charWidths[$char] = round($aw);
- }
- }
- }
+ $scale = 1; // not used
+ $this->getHMTX($numberOfHMetrics, $numGlyphs, $glyphToChar,
$scale);
///////////////////////////////////
// loca - Index to location
///////////////////////////////////
- if ($presubset) {
- $this->seek_table('loca');
- $this->glyphPos = array();
- if ($indexToLocFormat == 0) {
- for ($n=0; $n<=$numGlyphs; $n++) { // 1.02
- $this->glyphPos[] =
($this->read_ushort() * 2);
- }
- }
- else if ($indexToLocFormat == 1) {
- for ($n=0; $n<=$numGlyphs; $n++) { // 1.02
- $this->glyphPos[] =
($this->read_ulong());
- }
- }
- else
- die('Unknown location table format
'.$indexToLocFormat);
- }
- }
+ $this->getLOCA($indexToLocFormat, $numGlyphs);
+ $subsetglyphs = array(0=>0);
+ $subsetCharToGlyph = array();
+ foreach($subset AS $code) {
+ if (isset($this->charToGlyph[$code])) {
+ $subsetglyphs[$this->charToGlyph[$code]] =
$code; // Old Glyph ID => Unicode
+ $subsetCharToGlyph[$code] =
$this->charToGlyph[$code]; // Unicode to old GlyphID
-/////////////////////////////////////////////////////////////////////////////////////////
-
-
- function makeSubset($subset) {
- $this->fh = fopen($this->filename ,'rb') or die('Can\'t open
file ' . $this->filename );
- $this->otables = array();
- $glyphMap = array(0=>0);
- $glyphSet = array(0=>0);
- $codeToGlyph = array();
- foreach($subset AS $code) {
- if (isset($this->charToGlyph[$code]))
- $originalGlyphIdx = $this->charToGlyph[$code];
- else
- $originalGlyphIdx = 0;
- if (!isset($glyphSet[$originalGlyphIdx])) {
- $glyphSet[$originalGlyphIdx] = count($glyphMap);
- $glyphMap[] = $originalGlyphIdx;
}
- $codeToGlyph[$code] = $glyphSet[$originalGlyphIdx];
+ $this->maxUni = max($this->maxUni, $code);
}
list($start,$dummy) = $this->get_table_pos('glyf');
+ $glyphSet = array();
+ ksort($subsetglyphs);
$n = 0;
- while ($n < count($glyphMap)) {
- $originalGlyphIdx = $glyphMap[$n];
- $glyphPos = $this->glyphPos[$originalGlyphIdx];
- $glyphLen = $this->glyphPos[$originalGlyphIdx + 1] -
$glyphPos;
- $n += 1;
- if (!$glyphLen) continue;
- $this->seek($start + $glyphPos);
- $numberOfContours = $this->read_short();
- if ($numberOfContours < 0) {
- $this->skip(8);
- $flags = GF_MORE;
- while ($flags & GF_MORE) {
- $flags = $this->read_ushort();
- $glyphIdx = $this->read_ushort();
- if (!isset($glyphSet[$glyphIdx])) {
- $glyphSet[$glyphIdx] =
count($glyphMap);
- $glyphMap[] = $glyphIdx;
- }
- if ($flags & GF_WORDS)
- $this->skip(4);
- else
- $this->skip(2);
- if ($flags & GF_SCALE)
- $this->skip(2);
- else if ($flags & GF_XYSCALE)
- $this->skip(4);
- else if ($flags & GF_TWOBYTWO)
- $this->skip(8);
- }
- }
+ $fsLastCharIndex = 0; // maximum Unicode index (character
code) in this font, according to the cmap subtable for platform ID 3 and
platform- specific encoding ID 0 or 1.
+ foreach($subsetglyphs AS $originalGlyphIdx => $uni) {
+ $fsLastCharIndex = max($fsLastCharIndex , $uni);
+ $glyphSet[$originalGlyphIdx] = $n; // old glyphID
to new glyphID
+ $n++;
}
- $numGlyphs = $n = count($glyphMap);
- while ($n > 1 && $this->hmetrics[$n][0] == $this->hmetrics[$n -
1][0]) { $n -= 1; }
- $numberOfHMetrics = $n;
+ ksort($subsetCharToGlyph);
+ foreach($subsetCharToGlyph AS $uni => $originalGlyphIdx) {
+ $codeToGlyph[$uni] = $glyphSet[$originalGlyphIdx] ;
+ }
+ $this->codeToGlyph = $codeToGlyph;
+ ksort($subsetglyphs);
+ foreach($subsetglyphs AS $originalGlyphIdx => $uni) {
+ $this->getGlyphs($originalGlyphIdx, $start, $glyphSet,
$subsetglyphs);
+ }
+
+ $numGlyphs = $numberOfHMetrics = count($subsetglyphs );
+
//tables copied from the original
- $tags = array ('name', 'OS/2', 'prep');
+ $tags = array ('name');
foreach($tags AS $tag) { $this->add($tag,
$this->get_table($tag)); }
- $tags = array ('cvt ', 'fpgm');
- foreach($tags AS $tag) { // 1.02
- if (isset($this->table['tag'])) { $this->add($tag,
$this->get_table($tag)); }
+ $tags = array ('cvt ', 'fpgm', 'prep', 'gasp');
+ foreach($tags AS $tag) {
+ if (isset($this->tables[$tag])) { $this->add($tag,
$this->get_table($tag)); }
}
// post - PostScript
@@ -677,75 +623,85 @@
$post = "\x00\x03\x00\x00" . substr($opost,4,12) .
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00";
$this->add('post', $post);
- // hhea - Horizontal Header
- $hhea = $this->get_table('hhea');
- $hhea = $this->_set_ushort($hhea, 34, $numberOfHMetrics);
- $this->add('hhea', $hhea);
+ // Sort CID2GID map into segments of contiguous codes
+ ksort($codeToGlyph);
+ unset($codeToGlyph[0]);
+ //unset($codeToGlyph[65535]);
+ $rangeid = 0;
+ $range = array();
+ $prevcid = -2;
+ $prevglidx = -1;
+ // for each character
+ foreach ($codeToGlyph as $cid => $glidx) {
+ if ($cid == ($prevcid + 1) && $glidx == ($prevglidx +
1)) {
+ $range[$rangeid][] = $glidx;
+ } else {
+ // new range
+ $rangeid = $cid;
+ $range[$rangeid] = array();
+ $range[$rangeid][] = $glidx;
+ }
+ $prevcid = $cid;
+ $prevglidx = $glidx;
+ }
- // maxp - Maximum Profile
- $maxp = $this->get_table('maxp');
- $maxp = $this->_set_ushort($maxp, 4, $numGlyphs);
- $this->add('maxp', $maxp);
-
- // cmap - Character to glyph mapping - Format 6
- $entryCount = count($subset);
- $length = 10 + $entryCount * 2;
- $cmap = array(0, 1, // Index : version, number of subtables
- 1, 0, // Subtable : platform, encoding
- 0, 12, // offset (hi,lo)
- 6, $length, // Format 6 Mapping table: format,
length
- 0, 1, // language, First char code
- $entryCount);
-
- foreach($subset AS $code) { $cmap[] = $codeToGlyph[$code]; }
- $cmapstr = '';
- foreach($cmap AS $cm) { $cmapstr .= pack("n",$cm); }
- $this->add('cmap', $cmapstr);
-
-/*
// cmap - Character to glyph mapping - Format 4 (MS / )
- $segCount = 2;
+ $segCount = count($range) + 1; // + 1 Last segment has missing
character 0xFFFF
$searchRange = 1;
$entrySelector = 0;
while ($searchRange * 2 <= $segCount ) {
$searchRange = $searchRange * 2;
$entrySelector = $entrySelector + 1;
}
- $searchRange = $searchRange * 16;
- $rangeShift = $segCount * 16 - $searchRange;
- $length = 48 + ($numGlyphs-1);
- $cmap = array(0, 1, // Index : version, number of subtables
- 3, 0, // Subtable : platform (MS=3),
encoding
- 0, 12, // Subtable : offset (hi,lo)
- 4, $length, 0, // Format 4 Mapping table: format,
length, language
+ $searchRange = $searchRange * 2;
+ $rangeShift = $segCount * 2 - $searchRange;
+ $length = 16 + (8*$segCount ) + ($numGlyphs+1);
+ $cmap = array(0, 1, // Index : version, number of
encoding subtables
+ 3, 1, // Encoding Subtable :
platform (MS=3), encoding (Unicode)
+ 0, 12, // Encoding Subtable : offset
(hi,lo)
+ 4, $length, 0, // Format 4 Mapping subtable:
format, length, language
$segCount*2,
$searchRange,
$entrySelector,
- $rangeShift,
- $numGlyphs, 0xFFFF, // endCode(s)
- 0,
- 1, 0xFFFF, // startCode(s)
- 0, 1, // idDelta(s) Delta for
all character codes in segment
- 0, 0); // idRangeOffset[segCount]
Offset in bytes to glyph indexArray, or 0
- foreach($subset AS $code) { $cmap[] = $codeToGlyph[$code]; }
- $cmap[] = 0xFFFF;
+ $rangeShift);
+
+ // endCode(s)
+ foreach($range AS $start=>$subrange) {
+ $endCode = $start + (count($subrange)-1);
+ $cmap[] = $endCode; // endCode(s)
+ }
+ $cmap[] = 0xFFFF; // endCode of last Segment
+ $cmap[] = 0; // reservedPad
+
+ // startCode(s)
+ foreach($range AS $start=>$subrange) {
+ $cmap[] = $start; // startCode(s)
+ }
+ $cmap[] = 0xFFFF; // startCode of last Segment
+ // idDelta(s)
+ foreach($range AS $start=>$subrange) {
+ $idDelta = -($start-$subrange[0]);
+ $n += count($subrange);
+ $cmap[] = $idDelta; // idDelta(s)
+ }
+ $cmap[] = 1; // idDelta of last Segment
+ // idRangeOffset(s)
+ foreach($range AS $subrange) {
+ $cmap[] = 0; // idRangeOffset[segCount] Offset
in bytes to glyph indexArray, or 0
+
+ }
+ $cmap[] = 0; // idRangeOffset of last Segment
+ foreach($range AS $subrange) {
+ foreach($subrange AS $glidx) {
+ $cmap[] = $glidx;
+ }
+ }
+ $cmap[] = 0; // Mapping for last character
$cmapstr = '';
foreach($cmap AS $cm) { $cmapstr .= pack("n",$cm); }
$this->add('cmap', $cmapstr);
-*/
- // hmtx - Horizontal Metrics
- $hmtxstr = '';
- for($n=0;$n<$numGlyphs;$n++) {
- $originalGlyphIdx = $glyphMap[$n];
- $aw = $this->hmetrics[$originalGlyphIdx][0];
- $lsb = $this->hmetrics[$originalGlyphIdx][1];
- if ($n < $numberOfHMetrics) { $hmtxstr .=
pack("n",$aw); }
- $hmtxstr .= $this->pack_short($lsb);
- }
- $this->add('hmtx', $hmtxstr);
-
// glyf - Glyph data
list($glyfOffset,$glyfLength) = $this->get_table_pos('glyf');
if ($glyfLength < $this->maxStrLenRead) {
@@ -756,9 +712,29 @@
$glyf = '';
$pos = 0;
- for ($n=0;$n<$numGlyphs;$n++) {
+ $hmtxstr = '';
+ $xMinT = 0;
+ $yMinT = 0;
+ $xMaxT = 0;
+ $yMaxT = 0;
+ $advanceWidthMax = 0;
+ $minLeftSideBearing = 0;
+ $minRightSideBearing = 0;
+ $xMaxExtent = 0;
+ $maxPoints = 0; // points in non-compound glyph
+ $maxContours = 0; // contours in
non-compound glyph
+ $maxComponentPoints = 0; // points in compound glyph
+ $maxComponentContours = 0; // contours in compound glyph
+ $maxComponentElements = 0; // number of glyphs referenced
at top level
+ $maxComponentDepth = 0; // levels of recursion, set to
0 if font has only simple glyphs
+ $this->glyphdata = array();
+
+ foreach($subsetglyphs AS $originalGlyphIdx => $uni) {
+ // hmtx - Horizontal Metrics
+ $hm = $this->getHMetric($orignHmetrics,
$originalGlyphIdx);
+ $hmtxstr .= $hm;
+
$offsets[] = $pos;
- $originalGlyphIdx = $glyphMap[$n];
$glyphPos = $this->glyphPos[$originalGlyphIdx];
$glyphLen = $this->glyphPos[$originalGlyphIdx + 1] -
$glyphPos;
if ($glyfLength < $this->maxStrLenRead) {
@@ -768,15 +744,22 @@
if ($glyphLen > 0) $data =
$this->get_chunk($glyfOffset+$glyphPos,$glyphLen);
else $data = '';
}
- if ($glyphLen > 0) $up = unpack("n", substr($data,0,2));
- if ($glyphLen > 2 && ($up[1] & (1 << 15)) ) {
+
+ if ($glyphLen > 0) {
+ $up = unpack("n", substr($data,0,2));
+ }
+
+ if ($glyphLen > 2 && ($up[1] & (1 << 15)) ) { // If
number of contours <= -1 i.e. composiste glyph
$pos_in_glyph = 10;
$flags = GF_MORE;
+ $nComponentElements = 0;
while ($flags & GF_MORE) {
+ $nComponentElements += 1; //
number of glyphs referenced at top level
$up = unpack("n",
substr($data,$pos_in_glyph,2));
$flags = $up[1];
$up = unpack("n",
substr($data,$pos_in_glyph+2,2));
$glyphIdx = $up[1];
+
$this->glyphdata[$originalGlyphIdx]['compGlyphs'][] = $glyphIdx;
$data = $this->_set_ushort($data,
$pos_in_glyph + 2, $glyphSet[$glyphIdx]);
$pos_in_glyph += 4;
if ($flags & GF_WORDS) { $pos_in_glyph
+= 4; }
@@ -785,7 +768,9 @@
else if ($flags & GF_XYSCALE) {
$pos_in_glyph += 4; }
else if ($flags & GF_TWOBYTWO) {
$pos_in_glyph += 8; }
}
+ $maxComponentElements =
max($maxComponentElements, $nComponentElements);
}
+
$glyf .= $data;
$pos += $glyphLen;
if ($pos % 4 != 0) {
@@ -794,9 +779,13 @@
$pos += $padding;
}
}
+
$offsets[] = $pos;
$this->add('glyf', $glyf);
+ // hmtx - Horizontal Metrics
+ $this->add('hmtx', $hmtxstr);
+
// loca - Index to location
$locastr = '';
if ((($pos + 1) >> 1) > 0xFFFF) {
@@ -814,10 +803,237 @@
$head = $this->_set_ushort($head, 50, $indexToLocFormat);
$this->add('head', $head);
+
+ // hhea - Horizontal Header
+ $hhea = $this->get_table('hhea');
+ $hhea = $this->_set_ushort($hhea, 34, $numberOfHMetrics);
+ $this->add('hhea', $hhea);
+
+ // maxp - Maximum Profile
+ $maxp = $this->get_table('maxp');
+ $maxp = $this->_set_ushort($maxp, 4, $numGlyphs);
+ $this->add('maxp', $maxp);
+
+
+ // OS/2 - OS/2
+ $os2 = $this->get_table('OS/2');
+ $this->add('OS/2', $os2 );
+
fclose($this->fh);
// Put the TTF file together
$stm = '';
+ $this->endTTFile($stm);
+ return $stm ;
+ }
+
+
//////////////////////////////////////////////////////////////////////////////////
+ // Recursively get composite glyph data
+ function getGlyphData($originalGlyphIdx, &$maxdepth, &$depth, &$points,
&$contours) {
+ $depth++;
+ $maxdepth = max($maxdepth, $depth);
+ if (count($this->glyphdata[$originalGlyphIdx]['compGlyphs'])) {
+
foreach($this->glyphdata[$originalGlyphIdx]['compGlyphs'] AS $glyphIdx) {
+ $this->getGlyphData($glyphIdx, $maxdepth,
$depth, $points, $contours);
+ }
+ }
+ else if (($this->glyphdata[$originalGlyphIdx]['nContours'] > 0)
&& $depth > 0) { // simple
+ $contours +=
$this->glyphdata[$originalGlyphIdx]['nContours'];
+ $points +=
$this->glyphdata[$originalGlyphIdx]['nPoints'];
+ }
+ $depth--;
+ }
+
+
+
//////////////////////////////////////////////////////////////////////////////////
+ // Recursively get composite glyphs
+ function getGlyphs($originalGlyphIdx, &$start, &$glyphSet,
&$subsetglyphs) {
+ $glyphPos = $this->glyphPos[$originalGlyphIdx];
+ $glyphLen = $this->glyphPos[$originalGlyphIdx + 1] - $glyphPos;
+ if (!$glyphLen) {
+ return;
+ }
+ $this->seek($start + $glyphPos);
+ $numberOfContours = $this->read_short();
+ if ($numberOfContours < 0) {
+ $this->skip(8);
+ $flags = GF_MORE;
+ while ($flags & GF_MORE) {
+ $flags = $this->read_ushort();
+ $glyphIdx = $this->read_ushort();
+ if (!isset($glyphSet[$glyphIdx])) {
+ $glyphSet[$glyphIdx] =
count($subsetglyphs); // old glyphID to new glyphID
+ $subsetglyphs[$glyphIdx] = true;
+ }
+ $savepos = ftell($this->fh);
+ $this->getGlyphs($glyphIdx, $start, $glyphSet,
$subsetglyphs);
+ $this->seek($savepos);
+ if ($flags & GF_WORDS)
+ $this->skip(4);
+ else
+ $this->skip(2);
+ if ($flags & GF_SCALE)
+ $this->skip(2);
+ else if ($flags & GF_XYSCALE)
+ $this->skip(4);
+ else if ($flags & GF_TWOBYTWO)
+ $this->skip(8);
+ }
+ }
+ }
+
+
//////////////////////////////////////////////////////////////////////////////////
+
+ function getHMTX($numberOfHMetrics, $numGlyphs, &$glyphToChar, $scale) {
+ $start = $this->seek_table("hmtx");
+ $aw = 0;
+ $this->charWidths = str_pad('', 256*256*2, "\x00");
+ $nCharWidths = 0;
+ if (($numberOfHMetrics*4) < $this->maxStrLenRead) {
+ $data = $this->get_chunk($start,($numberOfHMetrics*4));
+ $arr = unpack("n*", $data);
+ }
+ else { $this->seek($start); }
+ for( $glyph=0; $glyph<$numberOfHMetrics; $glyph++) {
+
+ if (($numberOfHMetrics*4) < $this->maxStrLenRead) {
+ $aw = $arr[($glyph*2)+1];
+ }
+ else {
+ $aw = $this->read_ushort();
+ $lsb = $this->read_ushort();
+ }
+ if (isset($glyphToChar[$glyph]) || $glyph == 0) {
+
+ if ($aw >= (1 << 15) ) { $aw = 0; } // 1.03
Some (arabic) fonts have -ve values for width
+ // although should be unsigned value -
comes out as e.g. 65108 (intended -50)
+ if ($glyph == 0) {
+ $this->defaultWidth = $scale*$aw;
+ continue;
+ }
+ foreach($glyphToChar[$glyph] AS $char) {
+ if ($char != 0 && $char != 65535) {
+ $w = intval(round($scale*$aw));
+ if ($w == 0) { $w = 65535; }
+ if ($char < 196608) {
+
$this->charWidths[$char*2] = chr($w >> 8);
+
$this->charWidths[$char*2 + 1] = chr($w & 0xFF);
+ $nCharWidths++;
+ }
+ }
+ }
+ }
+ }
+ $data =
$this->get_chunk(($start+$numberOfHMetrics*4),($numGlyphs*2));
+ $arr = unpack("n*", $data);
+ $diff = $numGlyphs-$numberOfHMetrics;
+ for( $pos=0; $pos<$diff; $pos++) {
+ $glyph = $pos + $numberOfHMetrics;
+ if (isset($glyphToChar[$glyph])) {
+ foreach($glyphToChar[$glyph] AS $char) {
+ if ($char != 0 && $char != 65535) {
+ $w = intval(round($scale*$aw));
+ if ($w == 0) { $w = 65535; }
+ if ($char < 196608) {
+
$this->charWidths[$char*2] = chr($w >> 8);
+
$this->charWidths[$char*2 + 1] = chr($w & 0xFF);
+ $nCharWidths++;
+ }
+ }
+ }
+ }
+ }
+ // NB 65535 is a set width of 0
+ // First bytes define number of chars in font
+ $this->charWidths[0] = chr($nCharWidths >> 8);
+ $this->charWidths[1] = chr($nCharWidths & 0xFF);
+ }
+
+ function getHMetric($numberOfHMetrics, $gid) {
+ $start = $this->seek_table("hmtx");
+ if ($gid < $numberOfHMetrics) {
+ $this->seek($start+($gid*4));
+ $hm = fread($this->fh,4);
+ }
+ else {
+ $this->seek($start+(($numberOfHMetrics-1)*4));
+ $hm = fread($this->fh,2);
+ $this->seek($start+($numberOfHMetrics*2)+($gid*2));
+ $hm .= fread($this->fh,2);
+ }
+ return $hm;
+ }
+
+ function getLOCA($indexToLocFormat, $numGlyphs) {
+ $start = $this->seek_table('loca');
+ $this->glyphPos = array();
+ if ($indexToLocFormat == 0) {
+ $data = $this->get_chunk($start,($numGlyphs*2)+2);
+ $arr = unpack("n*", $data);
+ for ($n=0; $n<=$numGlyphs; $n++) {
+ $this->glyphPos[] = ($arr[$n+1] * 2);
+ }
+ }
+ else if ($indexToLocFormat == 1) {
+ $data = $this->get_chunk($start,($numGlyphs*4)+4);
+ $arr = unpack("N*", $data);
+ for ($n=0; $n<=$numGlyphs; $n++) {
+ $this->glyphPos[] = ($arr[$n+1]);
+ }
+ }
+ else
+ die('Unknown location table format '.$indexToLocFormat);
+ }
+
+
+ // CMAP Format 4
+ function getCMAP4($unicode_cmap_offset, &$glyphToChar, &$charToGlyph ) {
+ $this->maxUniChar = 0;
+ $this->seek($unicode_cmap_offset + 2);
+ $length = $this->read_ushort();
+ $limit = $unicode_cmap_offset + $length;
+ $this->skip(2);
+
+ $segCount = $this->read_ushort() / 2;
+ $this->skip(6);
+ $endCount = array();
+ for($i=0; $i<$segCount; $i++) { $endCount[] =
$this->read_ushort(); }
+ $this->skip(2);
+ $startCount = array();
+ for($i=0; $i<$segCount; $i++) { $startCount[] =
$this->read_ushort(); }
+ $idDelta = array();
+ for($i=0; $i<$segCount; $i++) { $idDelta[] =
$this->read_short(); } // ???? was unsigned short
+ $idRangeOffset_start = $this->_pos;
+ $idRangeOffset = array();
+ for($i=0; $i<$segCount; $i++) { $idRangeOffset[] =
$this->read_ushort(); }
+
+ for ($n=0;$n<$segCount;$n++) {
+ $endpoint = ($endCount[$n] + 1);
+ for
($unichar=$startCount[$n];$unichar<$endpoint;$unichar++) {
+ if ($idRangeOffset[$n] == 0)
+ $glyph = ($unichar + $idDelta[$n]) &
0xFFFF;
+ else {
+ $offset = ($unichar - $startCount[$n])
* 2 + $idRangeOffset[$n];
+ $offset = $idRangeOffset_start + 2 * $n
+ $offset;
+ if ($offset >= $limit)
+ $glyph = 0;
+ else {
+ $glyph =
$this->get_ushort($offset);
+ if ($glyph != 0)
+ $glyph = ($glyph +
$idDelta[$n]) & 0xFFFF;
+ }
+ }
+ $charToGlyph[$unichar] = $glyph;
+ if ($unichar < 196608) { $this->maxUniChar =
max($unichar,$this->maxUniChar); }
+ $glyphToChar[$glyph][] = $unichar;
+ }
+ }
+ }
+
+
+ // Put the TTF file together
+ function endTTFile(&$stm) {
+ $stm = '';
$numTables = count($this->otables);
$searchRange = 1;
$entrySelector = 0;
@@ -829,11 +1045,16 @@
$rangeShift = $numTables * 16 - $searchRange;
// Header
- $stm .= (pack("Nnnnn", 0x74727565, $numTables, $searchRange,
$entrySelector, $rangeShift)); // 0x74727565 "true" for Mac
- // 0x00010000 for Windows
+ if (_TTF_MAC_HEADER) {
+ $stm .= (pack("Nnnnn", 0x74727565, $numTables,
$searchRange, $entrySelector, $rangeShift)); // Mac
+ }
+ else {
+ $stm .= (pack("Nnnnn", 0x00010000 , $numTables,
$searchRange, $entrySelector, $rangeShift)); // Windows
+ }
// Table directory
$tables = $this->otables;
+
ksort ($tables);
$offset = 12 + $numTables * 16;
foreach ($tables AS $tag=>$data) {
@@ -860,6 +1081,8 @@
}
+
+
}
Modified: phpcompta/trunk/include/tfpdf/font/zapfdingbats.php
===================================================================
--- phpcompta/trunk/include/tfpdf/font/zapfdingbats.php 2011-10-14 23:00:50 UTC
(rev 4217)
+++ phpcompta/trunk/include/tfpdf/font/zapfdingbats.php 2011-10-15 01:02:41 UTC
(rev 4218)
@@ -1,5 +1,9 @@
<?php
-$fpdf_charwidths['zapfdingbats']=array(
+$type = 'Core';
+$name = 'ZapfDingbats';
+$up = -100;
+$ut = 50;
+$cw = array(
chr(0)=>0,chr(1)=>0,chr(2)=>0,chr(3)=>0,chr(4)=>0,chr(5)=>0,chr(6)=>0,chr(7)=>0,chr(8)=>0,chr(9)=>0,chr(10)=>0,chr(11)=>0,chr(12)=>0,chr(13)=>0,chr(14)=>0,chr(15)=>0,chr(16)=>0,chr(17)=>0,chr(18)=>0,chr(19)=>0,chr(20)=>0,chr(21)=>0,
chr(22)=>0,chr(23)=>0,chr(24)=>0,chr(25)=>0,chr(26)=>0,chr(27)=>0,chr(28)=>0,chr(29)=>0,chr(30)=>0,chr(31)=>0,'
'=>278,'!'=>974,'"'=>961,'#'=>974,'$'=>980,'%'=>719,'&'=>789,'\''=>790,'('=>791,')'=>690,'*'=>960,'+'=>939,
','=>549,'-'=>855,'.'=>911,'/'=>933,'0'=>911,'1'=>945,'2'=>974,'3'=>755,'4'=>846,'5'=>762,'6'=>761,'7'=>571,'8'=>677,'9'=>763,':'=>760,';'=>759,'<'=>754,'='=>494,'>'=>552,'?'=>537,'@'=>577,'A'=>692,
Modified: phpcompta/trunk/include/tfpdf/tfpdf.php
===================================================================
--- phpcompta/trunk/include/tfpdf/tfpdf.php 2011-10-14 23:00:50 UTC (rev
4217)
+++ phpcompta/trunk/include/tfpdf/tfpdf.php 2011-10-15 01:02:41 UTC (rev
4218)
@@ -1,515 +1,515 @@
<?php
/*******************************************************************************
-* tFPDF (based on FPDF 1.6)
*
+* tFPDF (based on FPDF 1.7)
*
*
*
-* Version: 1.01
*
-* Date: 2010-06-19
*
+* Version: 1.24
*
+* Date: 2011-09-24
*
* Author: Ian Back <address@hidden>
*
* License: LGPL
*
*******************************************************************************/
-define('tFPDF_VERSION','1.01');
+define('tFPDF_VERSION','1.24');
class tFPDF
{
var $unifontSubset;
-var $extraFontSubsets = 0;
-var $t1asm;
+var $page; // current page number
+var $n; // current object number
+var $offsets; // array of object offsets
+var $buffer; // buffer holding in-memory PDF
+var $pages; // array containing pages
+var $state; // current document state
+var $compress; // compression flag
+var $k; // scale factor (number of points in user unit)
+var $DefOrientation; // default orientation
+var $CurOrientation; // current orientation
+var $StdPageSizes; // standard page sizes
+var $DefPageSize; // default page size
+var $CurPageSize; // current page size
+var $PageSizes; // used for pages with non default sizes or
orientations
+var $wPt, $hPt; // dimensions of current page in points
+var $w, $h; // dimensions of current page in user unit
+var $lMargin; // left margin
+var $tMargin; // top margin
+var $rMargin; // right margin
+var $bMargin; // page break margin
+var $cMargin; // cell margin
+var $x, $y; // current position in user unit
+var $lasth; // height of last printed cell
+var $LineWidth; // line width in user unit
+var $fontpath; // path containing fonts
+var $CoreFonts; // array of core font names
+var $fonts; // array of used fonts
+var $FontFiles; // array of font files
+var $diffs; // array of encoding differences
+var $FontFamily; // current font family
+var $FontStyle; // current font style
+var $underline; // underlining flag
+var $CurrentFont; // current font info
+var $FontSizePt; // current font size in points
+var $FontSize; // current font size in user unit
+var $DrawColor; // commands for drawing color
+var $FillColor; // commands for filling color
+var $TextColor; // commands for text color
+var $ColorFlag; // indicates whether fill and text colors are
different
+var $ws; // word spacing
+var $images; // array of used images
+var $PageLinks; // array of links in pages
+var $links; // array of internal links
+var $AutoPageBreak; // automatic page breaking
+var $PageBreakTrigger; // threshold used to trigger page breaks
+var $InHeader; // flag set when processing header
+var $InFooter; // flag set when processing footer
+var $ZoomMode; // zoom display mode
+var $LayoutMode; // layout display mode
+var $title; // title
+var $subject; // subject
+var $author; // author
+var $keywords; // keywords
+var $creator; // creator
+var $AliasNbPages; // alias for total number of pages
+var $PDFVersion; // PDF version number
-var $page; //current page number
-var $n; //current object number
-var $offsets; //array of object offsets
-var $buffer; //buffer holding in-memory PDF
-var $pages; //array containing pages
-var $state; //current document state
-var $compress; //compression flag
-var $k; //scale factor (number of points in user unit)
-var $DefOrientation; //default orientation
-var $CurOrientation; //current orientation
-var $PageFormats; //available page formats
-var $DefPageFormat; //default page format
-var $CurPageFormat; //current page format
-var $PageSizes; //array storing non-default page sizes
-var $wPt,$hPt; //dimensions of current page in points
-var $w,$h; //dimensions of current page in user unit
-var $lMargin; //left margin
-var $tMargin; //top margin
-var $rMargin; //right margin
-var $bMargin; //page break margin
-var $cMargin; //cell margin
-var $x,$y; //current position in user unit
-var $lasth; //height of last printed cell
-var $LineWidth; //line width in user unit
-var $CoreFonts; //array of standard font names
-var $fonts; //array of used fonts
-var $FontFiles; //array of font files
-var $diffs; //array of encoding differences
-var $FontFamily; //current font family
-var $FontStyle; //current font style
-var $underline; //underlining flag
-var $CurrentFont; //current font info
-var $FontSizePt; //current font size in points
-var $FontSize; //current font size in user unit
-var $DrawColor; //commands for drawing color
-var $FillColor; //commands for filling color
-var $TextColor; //commands for text color
-var $ColorFlag; //indicates whether fill and text colors are different
-var $ws; //word spacing
-var $images; //array of used images
-var $PageLinks; //array of links in pages
-var $links; //array of internal links
-var $AutoPageBreak; //automatic page breaking
-var $PageBreakTrigger; //threshold used to trigger page breaks
-var $InHeader; //flag set when processing header
-var $InFooter; //flag set when processing footer
-var $ZoomMode; //zoom display mode
-var $LayoutMode; //layout display mode
-var $title; //title
-var $subject; //subject
-var $author; //author
-var $keywords; //keywords
-var $creator; //creator
-var $AliasNbPages; //alias for total number of pages
-var $PDFVersion; //PDF version number
-
/*******************************************************************************
*
*
* Public methods
*
*
*
*******************************************************************************/
-function tFPDF($orientation='P', $unit='mm', $format='A4')
+function tFPDF($orientation='P', $unit='mm', $size='A4')
{
- //Some checks
+ // Some checks
$this->_dochecks();
- //Initialization of properties
- $this->page=0;
- $this->n=2;
- $this->buffer='';
- $this->pages=array();
- $this->PageSizes=array();
- $this->state=0;
- $this->fonts=array();
- $this->FontFiles=array();
- $this->diffs=array();
- $this->images=array();
- $this->links=array();
- $this->InHeader=false;
- $this->InFooter=false;
- $this->lasth=0;
- $this->FontFamily='';
- $this->FontStyle='';
- $this->FontSizePt=12;
- $this->underline=false;
- $this->DrawColor='0 G';
- $this->FillColor='0 g';
- $this->TextColor='0 g';
- $this->ColorFlag=false;
- $this->ws=0;
- //Standard fonts
- $this->CoreFonts=array('courier'=>'Courier',
'courierB'=>'Courier-Bold', 'courierI'=>'Courier-Oblique',
'courierBI'=>'Courier-BoldOblique',
- 'helvetica'=>'Helvetica', 'helveticaB'=>'Helvetica-Bold',
'helveticaI'=>'Helvetica-Oblique', 'helveticaBI'=>'Helvetica-BoldOblique',
- 'times'=>'Times-Roman', 'timesB'=>'Times-Bold',
'timesI'=>'Times-Italic', 'timesBI'=>'Times-BoldItalic',
- 'symbol'=>'Symbol', 'zapfdingbats'=>'ZapfDingbats');
- //Scale factor
+ // Initialization of properties
+ $this->page = 0;
+ $this->n = 2;
+ $this->buffer = '';
+ $this->pages = array();
+ $this->PageSizes = array();
+ $this->state = 0;
+ $this->fonts = array();
+ $this->FontFiles = array();
+ $this->diffs = array();
+ $this->images = array();
+ $this->links = array();
+ $this->InHeader = false;
+ $this->InFooter = false;
+ $this->lasth = 0;
+ $this->FontFamily = '';
+ $this->FontStyle = '';
+ $this->FontSizePt = 12;
+ $this->underline = false;
+ $this->DrawColor = '0 G';
+ $this->FillColor = '0 g';
+ $this->TextColor = '0 g';
+ $this->ColorFlag = false;
+ $this->ws = 0;
+ // Font path
+ if(defined('FPDF_FONTPATH'))
+ {
+ $this->fontpath = FPDF_FONTPATH;
+ if(substr($this->fontpath,-1)!='/' &&
substr($this->fontpath,-1)!='\\')
+ $this->fontpath .= '/';
+ }
+ elseif(is_dir(dirname(__FILE__).'/font'))
+ $this->fontpath = dirname(__FILE__).'/font/';
+ else
+ $this->fontpath = '';
+ // Core fonts
+ $this->CoreFonts = array('courier', 'helvetica', 'times', 'symbol',
'zapfdingbats');
+ // Scale factor
if($unit=='pt')
- $this->k=1;
+ $this->k = 1;
elseif($unit=='mm')
- $this->k=72/25.4;
+ $this->k = 72/25.4;
elseif($unit=='cm')
- $this->k=72/2.54;
+ $this->k = 72/2.54;
elseif($unit=='in')
- $this->k=72;
+ $this->k = 72;
else
$this->Error('Incorrect unit: '.$unit);
-
- //Page format
- $this->PageFormats=array('a3'=>array(841.89,1190.55),
'a4'=>array(595.28,841.89), 'a5'=>array(420.94,595.28),
+ // Page sizes
+ $this->StdPageSizes = array('a3'=>array(841.89,1190.55),
'a4'=>array(595.28,841.89), 'a5'=>array(420.94,595.28),
'letter'=>array(612,792), 'legal'=>array(612,1008));
- if(is_string($format))
- $format=$this->_getpageformat($format);
- $this->DefPageFormat=$format;
- $this->CurPageFormat=$format;
- //Page orientation
- $orientation=strtolower($orientation);
+ $size = $this->_getpagesize($size);
+ $this->DefPageSize = $size;
+ $this->CurPageSize = $size;
+ // Page orientation
+ $orientation = strtolower($orientation);
if($orientation=='p' || $orientation=='portrait')
{
- $this->DefOrientation='P';
- $this->w=$this->DefPageFormat[0];
- $this->h=$this->DefPageFormat[1];
+ $this->DefOrientation = 'P';
+ $this->w = $size[0];
+ $this->h = $size[1];
}
elseif($orientation=='l' || $orientation=='landscape')
{
- $this->DefOrientation='L';
- $this->w=$this->DefPageFormat[1];
- $this->h=$this->DefPageFormat[0];
+ $this->DefOrientation = 'L';
+ $this->w = $size[1];
+ $this->h = $size[0];
}
else
$this->Error('Incorrect orientation: '.$orientation);
- $this->CurOrientation=$this->DefOrientation;
- $this->wPt=$this->w*$this->k;
- $this->hPt=$this->h*$this->k;
- //Page margins (1 cm)
- $margin=28.35/$this->k;
+ $this->CurOrientation = $this->DefOrientation;
+ $this->wPt = $this->w*$this->k;
+ $this->hPt = $this->h*$this->k;
+ // Page margins (1 cm)
+ $margin = 28.35/$this->k;
$this->SetMargins($margin,$margin);
- //Interior cell margin (1 mm)
- $this->cMargin=$margin/10;
- //Line width (0.2 mm)
- $this->LineWidth=.567/$this->k;
- //Automatic page break
+ // Interior cell margin (1 mm)
+ $this->cMargin = $margin/10;
+ // Line width (0.2 mm)
+ $this->LineWidth = .567/$this->k;
+ // Automatic page break
$this->SetAutoPageBreak(true,2*$margin);
- //Full width display mode
- $this->SetDisplayMode('fullwidth');
- //Enable compression
+ // Default display mode
+ $this->SetDisplayMode('default');
+ // Enable compression
$this->SetCompression(true);
- //Set default PDF version number
- $this->PDFVersion='1.3';
+ // Set default PDF version number
+ $this->PDFVersion = '1.3';
}
function SetMargins($left, $top, $right=null)
{
- //Set left, top and right margins
- $this->lMargin=$left;
- $this->tMargin=$top;
+ // Set left, top and right margins
+ $this->lMargin = $left;
+ $this->tMargin = $top;
if($right===null)
- $right=$left;
- $this->rMargin=$right;
+ $right = $left;
+ $this->rMargin = $right;
}
function SetLeftMargin($margin)
{
- //Set left margin
- $this->lMargin=$margin;
+ // Set left margin
+ $this->lMargin = $margin;
if($this->page>0 && $this->x<$margin)
- $this->x=$margin;
+ $this->x = $margin;
}
function SetTopMargin($margin)
{
- //Set top margin
- $this->tMargin=$margin;
+ // Set top margin
+ $this->tMargin = $margin;
}
function SetRightMargin($margin)
{
- //Set right margin
- $this->rMargin=$margin;
+ // Set right margin
+ $this->rMargin = $margin;
}
function SetAutoPageBreak($auto, $margin=0)
{
- //Set auto page break mode and triggering margin
- $this->AutoPageBreak=$auto;
- $this->bMargin=$margin;
- $this->PageBreakTrigger=$this->h-$margin;
+ // Set auto page break mode and triggering margin
+ $this->AutoPageBreak = $auto;
+ $this->bMargin = $margin;
+ $this->PageBreakTrigger = $this->h-$margin;
}
-function SetDisplayMode($zoom, $layout='continuous')
+function SetDisplayMode($zoom, $layout='default')
{
- //Set display mode in viewer
+ // Set display mode in viewer
if($zoom=='fullpage' || $zoom=='fullwidth' || $zoom=='real' ||
$zoom=='default' || !is_string($zoom))
- $this->ZoomMode=$zoom;
+ $this->ZoomMode = $zoom;
else
$this->Error('Incorrect zoom display mode: '.$zoom);
if($layout=='single' || $layout=='continuous' || $layout=='two' ||
$layout=='default')
- $this->LayoutMode=$layout;
+ $this->LayoutMode = $layout;
else
$this->Error('Incorrect layout display mode: '.$layout);
}
function SetCompression($compress)
{
- //Set page compression
+ // Set page compression
if(function_exists('gzcompress'))
- $this->compress=$compress;
+ $this->compress = $compress;
else
- $this->compress=false;
+ $this->compress = false;
}
function SetTitle($title, $isUTF8=false)
{
- //Title of document
+ // Title of document
if($isUTF8)
- $title=$this->_UTF8toUTF16($title);
- $this->title=$title;
+ $title = $this->_UTF8toUTF16($title);
+ $this->title = $title;
}
function SetSubject($subject, $isUTF8=false)
{
- //Subject of document
+ // Subject of document
if($isUTF8)
- $subject=$this->_UTF8toUTF16($subject);
- $this->subject=$subject;
+ $subject = $this->_UTF8toUTF16($subject);
+ $this->subject = $subject;
}
function SetAuthor($author, $isUTF8=false)
{
- //Author of document
+ // Author of document
if($isUTF8)
- $author=$this->_UTF8toUTF16($author);
- $this->author=$author;
+ $author = $this->_UTF8toUTF16($author);
+ $this->author = $author;
}
function SetKeywords($keywords, $isUTF8=false)
{
- //Keywords of document
+ // Keywords of document
if($isUTF8)
- $keywords=$this->_UTF8toUTF16($keywords);
- $this->keywords=$keywords;
+ $keywords = $this->_UTF8toUTF16($keywords);
+ $this->keywords = $keywords;
}
function SetCreator($creator, $isUTF8=false)
{
- //Creator of document
+ // Creator of document
if($isUTF8)
- $creator=$this->_UTF8toUTF16($creator);
- $this->creator=$creator;
+ $creator = $this->_UTF8toUTF16($creator);
+ $this->creator = $creator;
}
function AliasNbPages($alias='{nb}')
{
- //Define an alias for total number of pages
- $this->AliasNbPages=$alias;
+ // Define an alias for total number of pages
+ $this->AliasNbPages = $alias;
}
function Error($msg)
{
- //Fatal error
- die('<b>tFPDF error:</b> '.$msg);
+ // Fatal error
+ die('<b>FPDF error:</b> '.$msg);
}
function Open()
{
- //Begin document
- $this->state=1;
+ // Begin document
+ $this->state = 1;
}
function Close()
{
- //Terminate document
+ // Terminate document
if($this->state==3)
return;
if($this->page==0)
$this->AddPage();
- //Page footer
- $this->InFooter=true;
+ // Page footer
+ $this->InFooter = true;
$this->Footer();
- $this->InFooter=false;
- //Close page
+ $this->InFooter = false;
+ // Close page
$this->_endpage();
- //Close document
+ // Close document
$this->_enddoc();
}
-function AddPage($orientation='', $format='')
+function AddPage($orientation='', $size='')
{
- //Start a new page
+ // Start a new page
if($this->state==0)
$this->Open();
- $family=$this->FontFamily;
- $style=$this->FontStyle.($this->underline ? 'U' : '');
- $size=$this->FontSizePt;
- $lw=$this->LineWidth;
- $dc=$this->DrawColor;
- $fc=$this->FillColor;
- $tc=$this->TextColor;
- $cf=$this->ColorFlag;
+ $family = $this->FontFamily;
+ $style = $this->FontStyle.($this->underline ? 'U' : '');
+ $fontsize = $this->FontSizePt;
+ $lw = $this->LineWidth;
+ $dc = $this->DrawColor;
+ $fc = $this->FillColor;
+ $tc = $this->TextColor;
+ $cf = $this->ColorFlag;
if($this->page>0)
{
- //Page footer
- $this->InFooter=true;
+ // Page footer
+ $this->InFooter = true;
$this->Footer();
- $this->InFooter=false;
- //Close page
+ $this->InFooter = false;
+ // Close page
$this->_endpage();
}
- //Start new page
- $this->_beginpage($orientation,$format);
- //Set line cap style to square
+ // Start new page
+ $this->_beginpage($orientation,$size);
+ // Set line cap style to square
$this->_out('2 J');
- //Set line width
- $this->LineWidth=$lw;
+ // Set line width
+ $this->LineWidth = $lw;
$this->_out(sprintf('%.2F w',$lw*$this->k));
- //Set font
+ // Set font
if($family)
- $this->SetFont($family,$style,$size);
- //Set colors
- $this->DrawColor=$dc;
+ $this->SetFont($family,$style,$fontsize);
+ // Set colors
+ $this->DrawColor = $dc;
if($dc!='0 G')
$this->_out($dc);
- $this->FillColor=$fc;
+ $this->FillColor = $fc;
if($fc!='0 g')
$this->_out($fc);
- $this->TextColor=$tc;
- $this->ColorFlag=$cf;
- //Page header
- $this->InHeader=true;
+ $this->TextColor = $tc;
+ $this->ColorFlag = $cf;
+ // Page header
+ $this->InHeader = true;
$this->Header();
- $this->InHeader=false;
- //Restore line width
+ $this->InHeader = false;
+ // Restore line width
if($this->LineWidth!=$lw)
{
- $this->LineWidth=$lw;
+ $this->LineWidth = $lw;
$this->_out(sprintf('%.2F w',$lw*$this->k));
}
- //Restore font
+ // Restore font
if($family)
- $this->SetFont($family,$style,$size);
- //Restore colors
+ $this->SetFont($family,$style,$fontsize);
+ // Restore colors
if($this->DrawColor!=$dc)
{
- $this->DrawColor=$dc;
+ $this->DrawColor = $dc;
$this->_out($dc);
}
if($this->FillColor!=$fc)
{
- $this->FillColor=$fc;
+ $this->FillColor = $fc;
$this->_out($fc);
}
- $this->TextColor=$tc;
- $this->ColorFlag=$cf;
+ $this->TextColor = $tc;
+ $this->ColorFlag = $cf;
}
function Header()
{
- //To be implemented in your own inherited class
+ // To be implemented in your own inherited class
}
function Footer()
{
- //To be implemented in your own inherited class
+ // To be implemented in your own inherited class
}
function PageNo()
{
- //Get current page number
+ // Get current page number
return $this->page;
}
function SetDrawColor($r, $g=null, $b=null)
{
- //Set color for all stroking operations
+ // Set color for all stroking operations
if(($r==0 && $g==0 && $b==0) || $g===null)
- $this->DrawColor=sprintf('%.3F G',$r/255);
+ $this->DrawColor = sprintf('%.3F G',$r/255);
else
- $this->DrawColor=sprintf('%.3F %.3F %.3F
RG',$r/255,$g/255,$b/255);
+ $this->DrawColor = sprintf('%.3F %.3F %.3F
RG',$r/255,$g/255,$b/255);
if($this->page>0)
$this->_out($this->DrawColor);
}
function SetFillColor($r, $g=null, $b=null)
{
- //Set color for all filling operations
+ // Set color for all filling operations
if(($r==0 && $g==0 && $b==0) || $g===null)
- $this->FillColor=sprintf('%.3F g',$r/255);
+ $this->FillColor = sprintf('%.3F g',$r/255);
else
- $this->FillColor=sprintf('%.3F %.3F %.3F
rg',$r/255,$g/255,$b/255);
- $this->ColorFlag=($this->FillColor!=$this->TextColor);
+ $this->FillColor = sprintf('%.3F %.3F %.3F
rg',$r/255,$g/255,$b/255);
+ $this->ColorFlag = ($this->FillColor!=$this->TextColor);
if($this->page>0)
$this->_out($this->FillColor);
}
function SetTextColor($r, $g=null, $b=null)
{
- //Set color for text
+ // Set color for text
if(($r==0 && $g==0 && $b==0) || $g===null)
- $this->TextColor=sprintf('%.3F g',$r/255);
+ $this->TextColor = sprintf('%.3F g',$r/255);
else
- $this->TextColor=sprintf('%.3F %.3F %.3F
rg',$r/255,$g/255,$b/255);
- $this->ColorFlag=($this->FillColor!=$this->TextColor);
+ $this->TextColor = sprintf('%.3F %.3F %.3F
rg',$r/255,$g/255,$b/255);
+ $this->ColorFlag = ($this->FillColor!=$this->TextColor);
}
function GetStringWidth($s)
{
- //Get width of a string in the current font
- $s=(string)$s;
- $cw=&$this->CurrentFont['cw'];
+ // Get width of a string in the current font
+ $s = (string)$s;
+ $cw = &$this->CurrentFont['cw'];
$w=0;
if ($this->unifontSubset) {
$unicode = $this->UTF8StringToArray($s);
foreach($unicode as $char) {
- if (isset($cw[$char])) { $w+=$cw[$char]; }
- else if($char>0 && $char<128 && isset($cw[chr($char)]))
{ $w+=$cw[chr($char)]; }
+ if (isset($cw[$char])) { $w += (ord($cw[2*$char])<<8) +
ord($cw[2*$char+1]); }
+ else if($char>0 && $char<128 && isset($cw[chr($char)]))
{ $w += $cw[chr($char)]; }
else
if(isset($this->CurrentFont['desc']['MissingWidth'])) { $w +=
$this->CurrentFont['desc']['MissingWidth']; }
else if(isset($this->CurrentFont['MissingWidth'])) { $w
+= $this->CurrentFont['MissingWidth']; }
else { $w += 500; }
}
}
else {
- $l=strlen($s);
+ $l = strlen($s);
for($i=0;$i<$l;$i++)
- $w+=$cw[$s[$i]];
+ $w += $cw[$s[$i]];
}
return $w*$this->FontSize/1000;
}
function SetLineWidth($width)
{
- //Set line width
- $this->LineWidth=$width;
+ // Set line width
+ $this->LineWidth = $width;
if($this->page>0)
$this->_out(sprintf('%.2F w',$width*$this->k));
}
function Line($x1, $y1, $x2, $y2)
{
- //Draw a line
+ // Draw a line
$this->_out(sprintf('%.2F %.2F m %.2F %.2F l
S',$x1*$this->k,($this->h-$y1)*$this->k,$x2*$this->k,($this->h-$y2)*$this->k));
}
function Rect($x, $y, $w, $h, $style='')
{
- //Draw a rectangle
+ // Draw a rectangle
if($style=='F')
- $op='f';
+ $op = 'f';
elseif($style=='FD' || $style=='DF')
- $op='B';
+ $op = 'B';
else
- $op='S';
+ $op = 'S';
$this->_out(sprintf('%.2F %.2F %.2F %.2F re
%s',$x*$this->k,($this->h-$y)*$this->k,$w*$this->k,-$h*$this->k,$op));
}
-
function AddFont($family, $style='', $file='', $uni=false)
{
- //Add a TrueType or Type1 font
- $family=strtolower($family);
+ // Add a TrueType, OpenType or Type1 font
+ $family = strtolower($family);
+ $style = strtoupper($style);
+ if($style=='IB')
+ $style='BI';
if($file=='') {
if ($uni) {
- $file=str_replace(' ','',$family).strtolower($style).'.ttf';
+ $file = str_replace(' ','',$family).strtolower($style).'.ttf';
}
else {
- $file=str_replace(' ','',$family).strtolower($style).'.php';
+ $file = str_replace(' ','',$family).strtolower($style).'.php';
}
}
- if($family=='arial')
- $family='helvetica';
- $style=strtoupper($style);
- if($style=='IB')
- $style='BI';
- $fontkey=$family.$style;
+ $fontkey = $family.$style;
if(isset($this->fonts[$fontkey]))
return;
if ($uni) {
- if (defined("_SYSTEM_TTFONTS") &&
file_exists(_SYSTEM_TTFONTS.$file )) { $ttfilename = _SYSTEM_TTFONTS.$file ; }
- else { $ttfilename = $this->_getfontpath().'unifont/'.$file ; }
- $filename = $file;
- $filename =str_replace(' ','',$filename );
- $filename =str_replace('-','',$filename );
- $unifilename =
$this->_getfontpath().'unifont/'.strtolower(substr($filename
,0,(strpos($filename ,'.'))));
- $diff = '';
- $enc = '';
+ if (defined("_SYSTEM_TTFONTS") &&
file_exists(_SYSTEM_TTFONTS.$file )) { $ttffilename = _SYSTEM_TTFONTS.$file ; }
+ else { $ttffilename = $this->_getfontpath().'unifont/'.$file ; }
+ $unifilename =
$this->_getfontpath().'unifont/'.strtolower(substr($file ,0,(strpos($file
,'.'))));
+ $name = '';
+ $originalsize = 0;
+ $ttfstat = stat($ttffilename);
if (file_exists($unifilename.'.mtx.php')) {
include($unifilename.'.mtx.php');
}
- if (!isset($type) || $type != "TrueTypesubset") {
-
include_once($this->_getfontpath().'unifont/ttfonts.php');
+ if (!isset($type) || !isset($name) || $originalsize !=
$ttfstat['size']) {
+ $ttffile = $ttffilename;
+
require_once($this->_getfontpath().'unifont/ttfonts.php');
$ttf = new TTFontFile();
- $ttf->getMetrics($ttfilename);
+ $ttf->getMetrics($ttffile);
$cw = $ttf->charWidths;
- $type = "TrueTypesubset";
- $name = preg_replace('/ /','',$ttf->fullName);
+ $name = preg_replace('/[ ()]/','',$ttf->fullName);
+
$desc= array('Ascent'=>round($ttf->ascent),
'Descent'=>round($ttf->descent),
'CapHeight'=>round($ttf->capHeight),
@@ -520,128 +520,117 @@
'MissingWidth'=>round($ttf->defaultWidth));
$up = round($ttf->underlinePosition);
$ut = round($ttf->underlineThickness);
- //Generate metrics .php file
+ $originalsize = $ttfstat['size']+0;
+ $type = 'TTF';
+ // Generate metrics .php file
$s='<?php'."\n";
+ $s.='$name=\''.$name."';\n";
$s.='$type=\''.$type."';\n";
- $s.='$name=\''.$name."';\n";
$s.='$desc='.var_export($desc,true).";\n";
$s.='$up='.$up.";\n";
$s.='$ut='.$ut.";\n";
- $s.='$cw='.var_export($cw,true).";\n";
- $s.="?>\n";
- if (is_writable($this->_getfontpath().'unifont')) {
+ $s.='$ttffile=\''.$ttffile."';\n";
+ $s.='$originalsize='.$originalsize.";\n";
+ $s.='$fontkey=\''.$fontkey."';\n";
+ $s.="?>";
+ if
(is_writable(dirname($this->_getfontpath().'unifont/'.'x'))) {
$fh = fopen($unifilename.'.mtx.php',"w");
fwrite($fh,$s,strlen($s));
fclose($fh);
+ $fh = fopen($unifilename.'.cw.dat',"wb");
+ fwrite($fh,$cw,strlen($cw));
+ fclose($fh);
+ @unlink($unifilename.'.cw127.php');
}
unset($ttf);
}
- if(!isset($name)) {
- $this->Error('Problem with the font definition file');
+ else {
+ $cw = @file_get_contents($unifilename.'.cw.dat');
}
- $i = count($this->fonts)+$this->extraFontSubsets+1;
- $sbarr = range(0,127);
- $this->fonts[$fontkey] = array('i'=>$i, 'type'=>$type,
'name'=>$name, 'desc'=>$desc, 'up'=>$up, 'ut'=>$ut, 'cw'=>$cw, 'enc'=>$enc,
'file'=>$ttfilename, 'subsets'=>array(0=>$sbarr), 'subsetfontids'=>array($i),
'used'=>false);
+ $i = count($this->fonts)+1;
+ if(!empty($this->AliasNbPages))
+ $sbarr = range(0,57);
+ else
+ $sbarr = range(0,32);
+ $this->fonts[$fontkey] = array('i'=>$i, 'type'=>$type,
'name'=>$name, 'desc'=>$desc, 'up'=>$up, 'ut'=>$ut, 'cw'=>$cw,
'ttffile'=>$ttffile, 'fontkey'=>$fontkey, 'subset'=>$sbarr,
'unifilename'=>$unifilename);
+
+ $this->FontFiles[$fontkey]=array('length1'=>$originalsize,
'type'=>"TTF", 'ttffile'=>$ttffile);
+ $this->FontFiles[$file]=array('type'=>"TTF");
unset($cw);
}
else {
- include($this->_getfontpath().$file);
- if(!isset($name))
- $this->Error('Could not include font definition file');
- $i=count($this->fonts)+$this->extraFontSubsets+1;
- $this->fonts[$fontkey]=array('i'=>$i, 'type'=>$type,
'name'=>$name, 'desc'=>$desc, 'up'=>$up, 'ut'=>$ut, 'cw'=>$cw, 'enc'=>$enc,
'file'=>$file);
- }
-
- if($diff)
- {
- //Search existing encodings
- $d=0;
- $nb=count($this->diffs);
- for($i=1;$i<=$nb;$i++)
+ $info = $this->_loadfont($file);
+ $info['i'] = count($this->fonts)+1;
+ if(!empty($info['diff']))
{
- if($this->diffs[$i]==$diff)
+ // Search existing encodings
+ $n = array_search($info['diff'],$this->diffs);
+ if(!$n)
{
- $d=$i;
- break;
+ $n = count($this->diffs)+1;
+ $this->diffs[$n] = $info['diff'];
}
+ $info['diffn'] = $n;
}
- if($d==0)
+ if(!empty($info['file']))
{
- $d=$nb+1;
- $this->diffs[$d]=$diff;
+ // Embedded font
+ if($info['type']=='TrueType')
+ $this->FontFiles[$info['file']] =
array('length1'=>$info['originalsize']);
+ else
+ $this->FontFiles[$info['file']] =
array('length1'=>$info['size1'], 'length2'=>$info['size2']);
}
- $this->fonts[$fontkey]['diff']=$d;
+ $this->fonts[$fontkey] = $info;
}
- if($file)
- {
- if($type=='TrueType')
- $this->FontFiles[$file]=array('length1'=>$originalsize);
- else if ($uni && $type == "TrueTypesubset")
- $this->FontFiles[$file]=array('type'=>"TrueTypesubset");
- else
- $this->FontFiles[$file]=array('length1'=>$size1,
'length2'=>$size2);
- }
}
function SetFont($family, $style='', $size=0)
{
- //Select a font; size given in points
- global $fpdf_charwidths;
-
- $family=strtolower($family);
+ // Select a font; size given in points
if($family=='')
- $family=$this->FontFamily;
- if($family=='arial')
- $family='helvetica';
- elseif($family=='symbol' || $family=='zapfdingbats')
- $style='';
- $style=strtoupper($style);
+ $family = $this->FontFamily;
+ else
+ $family = strtolower($family);
+ $style = strtoupper($style);
if(strpos($style,'U')!==false)
{
- $this->underline=true;
- $style=str_replace('U','',$style);
+ $this->underline = true;
+ $style = str_replace('U','',$style);
}
else
- $this->underline=false;
+ $this->underline = false;
if($style=='IB')
- $style='BI';
+ $style = 'BI';
if($size==0)
- $size=$this->FontSizePt;
- //Test if font is already selected
+ $size = $this->FontSizePt;
+ // Test if font is already selected
if($this->FontFamily==$family && $this->FontStyle==$style &&
$this->FontSizePt==$size)
return;
- //Test if used for the first time
- $fontkey=$family.$style;
+ // Test if font is already loaded
+ $fontkey = $family.$style;
if(!isset($this->fonts[$fontkey]))
{
- //Check if one of the standard fonts
- if(isset($this->CoreFonts[$fontkey]))
+ // Test if one of the core fonts
+ if($family=='arial')
+ $family = 'helvetica';
+ if(in_array($family,$this->CoreFonts))
{
- if(!isset($fpdf_charwidths[$fontkey]))
- {
- //Load metric file
- $file=$family;
- if($family=='times' || $family=='helvetica')
- $file.=strtolower($style);
- include($this->_getfontpath().$file.'.php');
- if(!isset($fpdf_charwidths[$fontkey]))
- $this->Error('Could not include font
metric file');
- }
- $i=count($this->fonts)+1+$this->extraFontSubsets;
- $name=$this->CoreFonts[$fontkey];
- $cw=$fpdf_charwidths[$fontkey];
- $this->fonts[$fontkey]=array('i'=>$i, 'type'=>'core',
'name'=>$name, 'up'=>-100, 'ut'=>50, 'cw'=>$cw);
+ if($family=='symbol' || $family=='zapfdingbats')
+ $style = '';
+ $fontkey = $family.$style;
+ if(!isset($this->fonts[$fontkey]))
+ $this->AddFont($family,$style);
}
else
$this->Error('Undefined font: '.$family.' '.$style);
}
- //Select it
- $this->FontFamily=$family;
- $this->FontStyle=$style;
- $this->FontSizePt=$size;
- $this->FontSize=$size/$this->k;
- $this->CurrentFont=&$this->fonts[$fontkey];
- if ($this->fonts[$fontkey]['type']=='TrueTypesubset') {
$this->unifontSubset = true; }
+ // Select it
+ $this->FontFamily = $family;
+ $this->FontStyle = $style;
+ $this->FontSizePt = $size;
+ $this->FontSize = $size/$this->k;
+ $this->CurrentFont = &$this->fonts[$fontkey];
+ if ($this->fonts[$fontkey]['type']=='TTF') { $this->unifontSubset =
true; }
else { $this->unifontSubset = false; }
if($this->page>0)
$this->_out(sprintf('BT /F%d %.2F Tf
ET',$this->CurrentFont['i'],$this->FontSizePt));
@@ -649,199 +638,230 @@
function SetFontSize($size)
{
- //Set font size in points
+ // Set font size in points
if($this->FontSizePt==$size)
return;
- $this->FontSizePt=$size;
- $this->FontSize=$size/$this->k;
+ $this->FontSizePt = $size;
+ $this->FontSize = $size/$this->k;
if($this->page>0)
$this->_out(sprintf('BT /F%d %.2F Tf
ET',$this->CurrentFont['i'],$this->FontSizePt));
}
function AddLink()
{
- //Create a new internal link
- $n=count($this->links)+1;
- $this->links[$n]=array(0, 0);
+ // Create a new internal link
+ $n = count($this->links)+1;
+ $this->links[$n] = array(0, 0);
return $n;
}
function SetLink($link, $y=0, $page=-1)
{
- //Set destination of internal link
+ // Set destination of internal link
if($y==-1)
- $y=$this->y;
+ $y = $this->y;
if($page==-1)
- $page=$this->page;
- $this->links[$link]=array($page, $y);
+ $page = $this->page;
+ $this->links[$link] = array($page, $y);
}
function Link($x, $y, $w, $h, $link)
{
- //Put a link on the page
- $this->PageLinks[$this->page][]=array($x*$this->k,
$this->hPt-$y*$this->k, $w*$this->k, $h*$this->k, $link);
+ // Put a link on the page
+ $this->PageLinks[$this->page][] = array($x*$this->k,
$this->hPt-$y*$this->k, $w*$this->k, $h*$this->k, $link);
}
function Text($x, $y, $txt)
{
- //Output a string
+ // Output a string
if ($this->unifontSubset)
- $txt2 = $this->UTF8toSubset($txt);
+ {
+ $txt2 = '('.$this->_escape($this->UTF8ToUTF16BE($txt,
false)).')';
+ foreach($this->UTF8StringToArray($txt) as $uni)
+ $this->CurrentFont['subset'][$uni] = $uni;
+ }
else
- $txt2='('.$this->_escape($txt).')';
- $s=sprintf('BT %.2F %.2F Td %s Tj
ET',$x*$this->k,($this->h-$y)*$this->k,$txt2);
+ $txt2 = '('.$this->_escape($txt).')';
+ $s = sprintf('BT %.2F %.2F Td %s Tj
ET',$x*$this->k,($this->h-$y)*$this->k,$txt2);
if($this->underline && $txt!='')
- $s.=' '.$this->_dounderline($x,$y,$txt);
+ $s .= ' '.$this->_dounderline($x,$y,$txt);
if($this->ColorFlag)
- $s='q '.$this->TextColor.' '.$s.' Q';
+ $s = 'q '.$this->TextColor.' '.$s.' Q';
$this->_out($s);
}
function AcceptPageBreak()
{
- //Accept automatic page break or not
+ // Accept automatic page break or not
return $this->AutoPageBreak;
}
function Cell($w, $h=0, $txt='', $border=0, $ln=0, $align='', $fill=false,
$link='')
{
- //Output a cell
- $k=$this->k;
+ // Output a cell
+ $k = $this->k;
if($this->y+$h>$this->PageBreakTrigger && !$this->InHeader &&
!$this->InFooter && $this->AcceptPageBreak())
{
- //Automatic page break
- $x=$this->x;
- $ws=$this->ws;
+ // Automatic page break
+ $x = $this->x;
+ $ws = $this->ws;
if($ws>0)
{
- $this->ws=0;
+ $this->ws = 0;
$this->_out('0 Tw');
}
- $this->AddPage($this->CurOrientation,$this->CurPageFormat);
- $this->x=$x;
+ $this->AddPage($this->CurOrientation,$this->CurPageSize);
+ $this->x = $x;
if($ws>0)
{
- $this->ws=$ws;
+ $this->ws = $ws;
$this->_out(sprintf('%.3F Tw',$ws*$k));
}
}
if($w==0)
- $w=$this->w-$this->rMargin-$this->x;
- $s='';
+ $w = $this->w-$this->rMargin-$this->x;
+ $s = '';
if($fill || $border==1)
{
if($fill)
- $op=($border==1) ? 'B' : 'f';
+ $op = ($border==1) ? 'B' : 'f';
else
- $op='S';
- $s=sprintf('%.2F %.2F %.2F %.2F re %s
',$this->x*$k,($this->h-$this->y)*$k,$w*$k,-$h*$k,$op);
+ $op = 'S';
+ $s = sprintf('%.2F %.2F %.2F %.2F re %s
',$this->x*$k,($this->h-$this->y)*$k,$w*$k,-$h*$k,$op);
}
if(is_string($border))
{
- $x=$this->x;
- $y=$this->y;
+ $x = $this->x;
+ $y = $this->y;
if(strpos($border,'L')!==false)
- $s.=sprintf('%.2F %.2F m %.2F %.2F l S
',$x*$k,($this->h-$y)*$k,$x*$k,($this->h-($y+$h))*$k);
+ $s .= sprintf('%.2F %.2F m %.2F %.2F l S
',$x*$k,($this->h-$y)*$k,$x*$k,($this->h-($y+$h))*$k);
if(strpos($border,'T')!==false)
- $s.=sprintf('%.2F %.2F m %.2F %.2F l S
',$x*$k,($this->h-$y)*$k,($x+$w)*$k,($this->h-$y)*$k);
+ $s .= sprintf('%.2F %.2F m %.2F %.2F l S
',$x*$k,($this->h-$y)*$k,($x+$w)*$k,($this->h-$y)*$k);
if(strpos($border,'R')!==false)
- $s.=sprintf('%.2F %.2F m %.2F %.2F l S
',($x+$w)*$k,($this->h-$y)*$k,($x+$w)*$k,($this->h-($y+$h))*$k);
+ $s .= sprintf('%.2F %.2F m %.2F %.2F l S
',($x+$w)*$k,($this->h-$y)*$k,($x+$w)*$k,($this->h-($y+$h))*$k);
if(strpos($border,'B')!==false)
- $s.=sprintf('%.2F %.2F m %.2F %.2F l S
',$x*$k,($this->h-($y+$h))*$k,($x+$w)*$k,($this->h-($y+$h))*$k);
+ $s .= sprintf('%.2F %.2F m %.2F %.2F l S
',$x*$k,($this->h-($y+$h))*$k,($x+$w)*$k,($this->h-($y+$h))*$k);
}
if($txt!=='')
{
if($align=='R')
- $dx=$w-$this->cMargin-$this->GetStringWidth($txt);
+ $dx = $w-$this->cMargin-$this->GetStringWidth($txt);
elseif($align=='C')
- $dx=($w-$this->GetStringWidth($txt))/2;
+ $dx = ($w-$this->GetStringWidth($txt))/2;
else
- $dx=$this->cMargin;
+ $dx = $this->cMargin;
if($this->ColorFlag)
- $s.='q '.$this->TextColor.' ';
- if ($this->unifontSubset)
- $txt2 = $this->UTF8toSubset($txt);
- else
-
$txt2='('.str_replace(')','\\)',str_replace('(','\\(',str_replace('\\','\\\\',$txt))).')';
- $s.=sprintf('BT %.2F %.2F Td %s Tj
ET',($this->x+$dx)*$k,($this->h-($this->y+.5*$h+.3*$this->FontSize))*$k,$txt2);
+ $s .= 'q '.$this->TextColor.' ';
+
+ // If multibyte, Tw has no effect - do word spacing using an
adjustment before each space
+ if ($this->ws && $this->unifontSubset) {
+ foreach($this->UTF8StringToArray($txt) as $uni)
+ $this->CurrentFont['subset'][$uni] = $uni;
+ $space = $this->_escape($this->UTF8ToUTF16BE(' ',
false));
+ $s .= sprintf('BT 0 Tw %.2F %.2F Td
[',($this->x+$dx)*$k,($this->h-($this->y+.5*$h+.3*$this->FontSize))*$k);
+ $t = explode(' ',$txt);
+ $numt = count($t);
+ for($i=0;$i<$numt;$i++) {
+ $tx = $t[$i];
+ $tx =
'('.$this->_escape($this->UTF8ToUTF16BE($tx, false)).')';
+ $s .= sprintf('%s ',$tx);
+ if (($i+1)<$numt) {
+ $adj =
-($this->ws*$this->k)*1000/$this->FontSizePt;
+ $s .= sprintf('%d(%s) ',$adj,$space);
+ }
+ }
+ $s .= '] TJ';
+ $s .= ' ET';
+ }
+ else {
+ if ($this->unifontSubset)
+ {
+ $txt2 =
'('.$this->_escape($this->UTF8ToUTF16BE($txt, false)).')';
+ foreach($this->UTF8StringToArray($txt) as $uni)
+ $this->CurrentFont['subset'][$uni] =
$uni;
+ }
+ else
+
$txt2='('.str_replace(')','\\)',str_replace('(','\\(',str_replace('\\','\\\\',$txt))).')';
+ $s .= sprintf('BT %.2F %.2F Td %s Tj
ET',($this->x+$dx)*$k,($this->h-($this->y+.5*$h+.3*$this->FontSize))*$k,$txt2);
+ }
if($this->underline)
- $s.='
'.$this->_dounderline($this->x+$dx,$this->y+.5*$h+.3*$this->FontSize,$txt);
+ $s .= '
'.$this->_dounderline($this->x+$dx,$this->y+.5*$h+.3*$this->FontSize,$txt);
if($this->ColorFlag)
- $s.=' Q';
+ $s .= ' Q';
if($link)
$this->Link($this->x+$dx,$this->y+.5*$h-.5*$this->FontSize,$this->GetStringWidth($txt),$this->FontSize,$link);
}
if($s)
$this->_out($s);
- $this->lasth=$h;
+ $this->lasth = $h;
if($ln>0)
{
- //Go to next line
- $this->y+=$h;
+ // Go to next line
+ $this->y += $h;
if($ln==1)
- $this->x=$this->lMargin;
+ $this->x = $this->lMargin;
}
else
- $this->x+=$w;
+ $this->x += $w;
}
function MultiCell($w, $h, $txt, $border=0, $align='J', $fill=false)
{
- //Output text with automatic or explicit line breaks
- $cw=&$this->CurrentFont['cw'];
+ // Output text with automatic or explicit line breaks
+ $cw = &$this->CurrentFont['cw'];
if($w==0)
- $w=$this->w-$this->rMargin-$this->x;
- $wmax=($w-2*$this->cMargin);
- $s=str_replace("\r",'',$txt);
+ $w = $this->w-$this->rMargin-$this->x;
+ $wmax = ($w-2*$this->cMargin);
+ $s = str_replace("\r",'',$txt);
if ($this->unifontSubset) {
$nb=mb_strlen($s, 'utf-8');
while($nb>0 && mb_substr($s,$nb-1,1,'utf-8')=="\n") $nb--;
}
else {
- $nb=strlen($s);
+ $nb = strlen($s);
if($nb>0 && $s[$nb-1]=="\n")
$nb--;
}
-
- $b=0;
+ $b = 0;
if($border)
{
if($border==1)
{
- $border='LTRB';
- $b='LRT';
- $b2='LR';
+ $border = 'LTRB';
+ $b = 'LRT';
+ $b2 = 'LR';
}
else
{
- $b2='';
+ $b2 = '';
if(strpos($border,'L')!==false)
- $b2.='L';
+ $b2 .= 'L';
if(strpos($border,'R')!==false)
- $b2.='R';
- $b=(strpos($border,'T')!==false) ? $b2.'T' : $b2;
+ $b2 .= 'R';
+ $b = (strpos($border,'T')!==false) ? $b2.'T' : $b2;
}
}
- $sep=-1;
- $i=0;
- $j=0;
- $l=0;
- $ns=0;
- $nl=1;
+ $sep = -1;
+ $i = 0;
+ $j = 0;
+ $l = 0;
+ $ns = 0;
+ $nl = 1;
while($i<$nb)
{
- //Get next character
+ // Get next character
if ($this->unifontSubset) {
$c = mb_substr($s,$i,1,'UTF-8');
}
else {
$c=$s[$i];
}
- if($c=="\n") {
- //Explicit line break
+ if($c=="\n")
+ {
+ // Explicit line break
if($this->ws>0)
{
- $this->ws=0;
+ $this->ws = 0;
$this->_out('0 Tw');
}
if ($this->unifontSubset) {
@@ -851,19 +871,19 @@
$this->Cell($w,$h,substr($s,$j,$i-$j),$b,2,$align,$fill);
}
$i++;
- $sep=-1;
- $j=$i;
- $l=0;
- $ns=0;
+ $sep = -1;
+ $j = $i;
+ $l = 0;
+ $ns = 0;
$nl++;
if($border && $nl==2)
- $b=$b2;
+ $b = $b2;
continue;
}
if($c==' ')
{
- $sep=$i;
- $ls=$l;
+ $sep = $i;
+ $ls = $l;
$ns++;
}
@@ -872,14 +892,14 @@
if($l>$wmax)
{
- //Automatic line break
+ // Automatic line break
if($sep==-1)
{
if($i==$j)
$i++;
if($this->ws>0)
{
- $this->ws=0;
+ $this->ws = 0;
$this->_out('0 Tw');
}
if ($this->unifontSubset) {
@@ -893,7 +913,7 @@
{
if($align=='J')
{
- $this->ws=($ns>1) ? ($wmax-$ls)/($ns-1)
: 0;
+ $this->ws = ($ns>1) ?
($wmax-$ls)/($ns-1) : 0;
$this->_out(sprintf('%.3F
Tw',$this->ws*$this->k));
}
if ($this->unifontSubset) {
@@ -902,72 +922,71 @@
else {
$this->Cell($w,$h,substr($s,$j,$sep-$j),$b,2,$align,$fill);
}
- $i=$sep+1;
+ $i = $sep+1;
}
- $sep=-1;
- $j=$i;
- $l=0;
- $ns=0;
+ $sep = -1;
+ $j = $i;
+ $l = 0;
+ $ns = 0;
$nl++;
if($border && $nl==2)
- $b=$b2;
+ $b = $b2;
}
else
$i++;
}
- //Last chunk
+ // Last chunk
if($this->ws>0)
{
- $this->ws=0;
+ $this->ws = 0;
$this->_out('0 Tw');
}
if($border && strpos($border,'B')!==false)
- $b.='B';
+ $b .= 'B';
if ($this->unifontSubset) {
$this->Cell($w,$h,mb_substr($s,$j,$i-$j,'UTF-8'),$b,2,$align,$fill);
}
else {
$this->Cell($w,$h,substr($s,$j,$i-$j),$b,2,$align,$fill);
}
- $this->x=$this->lMargin;
+ $this->x = $this->lMargin;
}
function Write($h, $txt, $link='')
{
- //Output text in flowing mode
- $cw=&$this->CurrentFont['cw'];
- $w=$this->w-$this->rMargin-$this->x;
+ // Output text in flowing mode
+ $cw = &$this->CurrentFont['cw'];
+ $w = $this->w-$this->rMargin-$this->x;
- $wmax=($w-2*$this->cMargin);
- $s=str_replace("\r",'',$txt);
+ $wmax = ($w-2*$this->cMargin);
+ $s = str_replace("\r",'',$txt);
if ($this->unifontSubset) {
- $nb=mb_strlen($s, 'UTF-8');
+ $nb = mb_strlen($s, 'UTF-8');
if($nb==1 && $s==" ") {
$this->x += $this->GetStringWidth($s);
return;
}
}
else {
- $nb=strlen($s);
+ $nb = strlen($s);
}
-
- $sep=-1;
- $i=0;
- $j=0;
- $l=0;
- $nl=1;
+ $sep = -1;
+ $i = 0;
+ $j = 0;
+ $l = 0;
+ $nl = 1;
while($i<$nb)
{
- //Get next character
- //Get next character
+ // Get next character
if ($this->unifontSubset) {
$c = mb_substr($s,$i,1,'UTF-8');
}
else {
- $c=$s[$i];
+ $c = $s[$i];
}
- if($c=="\n") {
- //Explicit line break
+ if($c=="\n")
+ {
+ // Explicit line break
if ($this->unifontSubset) {
$this->Cell($w,$h,mb_substr($s,$j,$i-$j,'UTF-8'),0,2,'',0,$link);
}
@@ -975,36 +994,36 @@
$this->Cell($w,$h,substr($s,$j,$i-$j),0,2,'',0,$link);
}
$i++;
- $sep=-1;
- $j=$i;
- $l=0;
+ $sep = -1;
+ $j = $i;
+ $l = 0;
if($nl==1)
{
- $this->x=$this->lMargin;
- $w=$this->w-$this->rMargin-$this->x;
- $wmax=($w-2*$this->cMargin);
+ $this->x = $this->lMargin;
+ $w = $this->w-$this->rMargin-$this->x;
+ $wmax = ($w-2*$this->cMargin);
}
$nl++;
continue;
}
if($c==' ')
- $sep=$i;
+ $sep = $i;
if ($this->unifontSubset) { $l += $this->GetStringWidth($c); }
else { $l += $cw[$c]*$this->FontSize/1000; }
if($l>$wmax)
{
- //Automatic line break
+ // Automatic line break
if($sep==-1)
{
if($this->x>$this->lMargin)
{
- //Move to next line
- $this->x=$this->lMargin;
- $this->y+=$h;
- $w=$this->w-$this->rMargin-$this->x;
- $wmax=($w-2*$this->cMargin);
+ // Move to next line
+ $this->x = $this->lMargin;
+ $this->y += $h;
+ $w = $this->w-$this->rMargin-$this->x;
+ $wmax = ($w-2*$this->cMargin);
$i++;
$nl++;
continue;
@@ -1026,23 +1045,23 @@
else {
$this->Cell($w,$h,substr($s,$j,$sep-$j),0,2,'',0,$link);
}
- $i=$sep+1;
+ $i = $sep+1;
}
- $sep=-1;
- $j=$i;
- $l=0;
+ $sep = -1;
+ $j = $i;
+ $l = 0;
if($nl==1)
{
- $this->x=$this->lMargin;
- $w=$this->w-$this->rMargin-$this->x;
- $wmax=($w-2*$this->cMargin);
+ $this->x = $this->lMargin;
+ $w = $this->w-$this->rMargin-$this->x;
+ $wmax = ($w-2*$this->cMargin);
}
$nl++;
}
else
$i++;
}
- //Last chunk
+ // Last chunk
if($i!=$j) {
if ($this->unifontSubset) {
$this->Cell($l,$h,mb_substr($s,$j,$i-$j,'UTF-8'),0,0,'',0,$link);
@@ -1055,65 +1074,72 @@
function Ln($h=null)
{
- //Line feed; default value is last cell height
- $this->x=$this->lMargin;
+ // Line feed; default value is last cell height
+ $this->x = $this->lMargin;
if($h===null)
- $this->y+=$this->lasth;
+ $this->y += $this->lasth;
else
- $this->y+=$h;
+ $this->y += $h;
}
function Image($file, $x=null, $y=null, $w=0, $h=0, $type='', $link='')
{
- //Put an image on the page
+ // Put an image on the page
if(!isset($this->images[$file]))
{
- //First use of this image, get info
+ // First use of this image, get info
if($type=='')
{
- $pos=strrpos($file,'.');
+ $pos = strrpos($file,'.');
if(!$pos)
$this->Error('Image file has no extension and
no type was specified: '.$file);
- $type=substr($file,$pos+1);
+ $type = substr($file,$pos+1);
}
- $type=strtolower($type);
+ $type = strtolower($type);
if($type=='jpeg')
- $type='jpg';
- $mtd='_parse'.$type;
+ $type = 'jpg';
+ $mtd = '_parse'.$type;
if(!method_exists($this,$mtd))
$this->Error('Unsupported image type: '.$type);
- $info=$this->$mtd($file);
- $info['i']=count($this->images)+1;
- $this->images[$file]=$info;
+ $info = $this->$mtd($file);
+ $info['i'] = count($this->images)+1;
+ $this->images[$file] = $info;
}
else
- $info=$this->images[$file];
- //Automatic width and height calculation if needed
+ $info = $this->images[$file];
+
+ // Automatic width and height calculation if needed
if($w==0 && $h==0)
{
- //Put image at 72 dpi
- $w=$info['w']/$this->k;
- $h=$info['h']/$this->k;
+ // Put image at 96 dpi
+ $w = -96;
+ $h = -96;
}
- elseif($w==0)
- $w=$h*$info['w']/$info['h'];
- elseif($h==0)
- $h=$w*$info['h']/$info['w'];
- //Flowing mode
+ if($w<0)
+ $w = -$info['w']*72/$w/$this->k;
+ if($h<0)
+ $h = -$info['h']*72/$h/$this->k;
+ if($w==0)
+ $w = $h*$info['w']/$info['h'];
+ if($h==0)
+ $h = $w*$info['h']/$info['w'];
+
+ // Flowing mode
if($y===null)
{
if($this->y+$h>$this->PageBreakTrigger && !$this->InHeader &&
!$this->InFooter && $this->AcceptPageBreak())
{
- //Automatic page break
- $x2=$this->x;
-
$this->AddPage($this->CurOrientation,$this->CurPageFormat);
- $this->x=$x2;
+ // Automatic page break
+ $x2 = $this->x;
+
$this->AddPage($this->CurOrientation,$this->CurPageSize);
+ $this->x = $x2;
}
- $y=$this->y;
- $this->y+=$h;
+ $y = $this->y;
+ $this->y += $h;
}
+
if($x===null)
- $x=$this->x;
+ $x = $this->x;
$this->_out(sprintf('q %.2F 0 0 %.2F %.2F %.2F cm /I%d Do
Q',$w*$this->k,$h*$this->k,$x*$this->k,($this->h-($y+$h))*$this->k,$info['i']));
if($link)
$this->Link($x,$y,$w,$h,$link);
@@ -1121,102 +1147,92 @@
function GetX()
{
- //Get x position
+ // Get x position
return $this->x;
}
function SetX($x)
{
- //Set x position
+ // Set x position
if($x>=0)
- $this->x=$x;
+ $this->x = $x;
else
- $this->x=$this->w+$x;
+ $this->x = $this->w+$x;
}
function GetY()
{
- //Get y position
+ // Get y position
return $this->y;
}
function SetY($y)
{
- //Set y position and reset x
- $this->x=$this->lMargin;
+ // Set y position and reset x
+ $this->x = $this->lMargin;
if($y>=0)
- $this->y=$y;
+ $this->y = $y;
else
- $this->y=$this->h+$y;
+ $this->y = $this->h+$y;
}
function SetXY($x, $y)
{
- //Set x and y positions
+ // Set x and y positions
$this->SetY($y);
$this->SetX($x);
}
function Output($name='', $dest='')
{
- //Output PDF to some destination
+ // Output PDF to some destination
if($this->state<3)
$this->Close();
- $dest=strtoupper($dest);
+ $dest = strtoupper($dest);
if($dest=='')
{
if($name=='')
{
- $name='doc.pdf';
- $dest='I';
+ $name = 'doc.pdf';
+ $dest = 'I';
}
else
- $dest='F';
+ $dest = 'F';
}
switch($dest)
{
case 'I':
- //Send to standard output
- if(ob_get_length())
- $this->Error('Some data has already been
output, can\'t send PDF file');
- if(php_sapi_name()!='cli')
+ // Send to standard output
+ $this->_checkoutput();
+ if(PHP_SAPI!='cli')
{
- //We send to a browser
+ // We send to a browser
header('Content-Type: application/pdf');
- if(headers_sent())
- $this->Error('Some data has already
been output, can\'t send PDF file');
- header('Content-Length:
'.strlen($this->buffer));
header('Content-Disposition: inline;
filename="'.$name.'"');
header('Cache-Control: private, max-age=0,
must-revalidate');
header('Pragma: public');
- ini_set('zlib.output_compression','0');
}
echo $this->buffer;
break;
case 'D':
- //Download file
- if(ob_get_length())
- $this->Error('Some data has already been
output, can\'t send PDF file');
+ // Download file
+ $this->_checkoutput();
header('Content-Type: application/x-download');
- if(headers_sent())
- $this->Error('Some data has already been
output, can\'t send PDF file');
- header('Content-Length: '.strlen($this->buffer));
header('Content-Disposition: attachment;
filename="'.$name.'"');
header('Cache-Control: private, max-age=0,
must-revalidate');
header('Pragma: public');
- ini_set('zlib.output_compression','0');
echo $this->buffer;
break;
case 'F':
- //Save to local file
- $f=fopen($name,'wb');
+ // Save to local file
+ $f = fopen($name,'wb');
if(!$f)
$this->Error('Unable to create output file:
'.$name);
fwrite($f,$this->buffer,strlen($this->buffer));
fclose($f);
break;
case 'S':
- //Return as a string
+ // Return as a string
return $this->buffer;
default:
$this->Error('Incorrect output destination: '.$dest);
@@ -1231,125 +1247,163 @@
*******************************************************************************/
function _dochecks()
{
- //Check availability of %F
+ // Check availability of %F
if(sprintf('%.1F',1.0)!='1.0')
$this->Error('This version of PHP is not supported');
- //Check mbstring overloading
+ // Check availability of mbstring
+ if(!function_exists('mb_strlen'))
+ $this->Error('mbstring extension is not available');
+ // Check mbstring overloading
if(ini_get('mbstring.func_overload') & 2)
$this->Error('mbstring overloading must be disabled');
- //Disable runtime magic quotes
+ // Ensure runtime magic quotes are disabled
if(get_magic_quotes_runtime())
@set_magic_quotes_runtime(0);
}
-function _getpageformat($format)
+function _getfontpath()
{
- $format=strtolower($format);
- if(!isset($this->PageFormats[$format]))
- $this->Error('Unknown page format: '.$format);
- $a=$this->PageFormats[$format];
- return array($a[0]/$this->k, $a[1]/$this->k);
+ return $this->fontpath;
}
-function _getfontpath()
+function _checkoutput()
{
- if(!defined('FPDF_FONTPATH') && is_dir(dirname(__FILE__).'/font'))
- define('FPDF_FONTPATH',dirname(__FILE__).'/font/');
- return defined('FPDF_FONTPATH') ? FPDF_FONTPATH : '';
+ if(PHP_SAPI!='cli')
+ {
+ if(headers_sent($file,$line))
+ $this->Error("Some data has already been output, can't
send PDF file (output started at $file:$line)");
+ }
+ if(ob_get_length())
+ {
+ // The output buffer is not empty
+ if(preg_match('/^(\xEF\xBB\xBF)?\s*$/',ob_get_contents()))
+ {
+ // It contains only a UTF-8 BOM and/or whitespace,
let's clean it
+ ob_clean();
+ }
+ else
+ $this->Error("Some data has already been output, can't
send PDF file");
+ }
}
-function _beginpage($orientation, $format)
+function _getpagesize($size)
{
+ if(is_string($size))
+ {
+ $size = strtolower($size);
+ if(!isset($this->StdPageSizes[$size]))
+ $this->Error('Unknown page size: '.$size);
+ $a = $this->StdPageSizes[$size];
+ return array($a[0]/$this->k, $a[1]/$this->k);
+ }
+ else
+ {
+ if($size[0]>$size[1])
+ return array($size[1], $size[0]);
+ else
+ return $size;
+ }
+}
+
+function _beginpage($orientation, $size)
+{
$this->page++;
- $this->pages[$this->page]='';
- $this->state=2;
- $this->x=$this->lMargin;
- $this->y=$this->tMargin;
- $this->FontFamily='';
- //Check page size
+ $this->pages[$this->page] = '';
+ $this->state = 2;
+ $this->x = $this->lMargin;
+ $this->y = $this->tMargin;
+ $this->FontFamily = '';
+ // Check page size and orientation
if($orientation=='')
- $orientation=$this->DefOrientation;
+ $orientation = $this->DefOrientation;
else
- $orientation=strtoupper($orientation[0]);
- if($format=='')
- $format=$this->DefPageFormat;
+ $orientation = strtoupper($orientation[0]);
+ if($size=='')
+ $size = $this->DefPageSize;
else
+ $size = $this->_getpagesize($size);
+ if($orientation!=$this->CurOrientation ||
$size[0]!=$this->CurPageSize[0] || $size[1]!=$this->CurPageSize[1])
{
- if(is_string($format))
- $format=$this->_getpageformat($format);
- }
- if($orientation!=$this->CurOrientation ||
$format[0]!=$this->CurPageFormat[0] || $format[1]!=$this->CurPageFormat[1])
- {
- //New size
+ // New size or orientation
if($orientation=='P')
{
- $this->w=$format[0];
- $this->h=$format[1];
+ $this->w = $size[0];
+ $this->h = $size[1];
}
else
{
- $this->w=$format[1];
- $this->h=$format[0];
+ $this->w = $size[1];
+ $this->h = $size[0];
}
- $this->wPt=$this->w*$this->k;
- $this->hPt=$this->h*$this->k;
- $this->PageBreakTrigger=$this->h-$this->bMargin;
- $this->CurOrientation=$orientation;
- $this->CurPageFormat=$format;
+ $this->wPt = $this->w*$this->k;
+ $this->hPt = $this->h*$this->k;
+ $this->PageBreakTrigger = $this->h-$this->bMargin;
+ $this->CurOrientation = $orientation;
+ $this->CurPageSize = $size;
}
- if($orientation!=$this->DefOrientation ||
$format[0]!=$this->DefPageFormat[0] || $format[1]!=$this->DefPageFormat[1])
- $this->PageSizes[$this->page]=array($this->wPt, $this->hPt);
+ if($orientation!=$this->DefOrientation ||
$size[0]!=$this->DefPageSize[0] || $size[1]!=$this->DefPageSize[1])
+ $this->PageSizes[$this->page] = array($this->wPt, $this->hPt);
}
function _endpage()
{
- $this->state=1;
+ $this->state = 1;
}
+function _loadfont($font)
+{
+ // Load a font definition file from the font directory
+ include($this->fontpath.$font);
+ $a = get_defined_vars();
+ if(!isset($a['name']))
+ $this->Error('Could not include font definition file');
+ return $a;
+}
+
function _escape($s)
{
- //Escape special characters in strings
- $s=str_replace('\\','\\\\',$s);
- $s=str_replace('(','\\(',$s);
- $s=str_replace(')','\\)',$s);
- $s=str_replace("\r",'\\r',$s);
+ // Escape special characters in strings
+ $s = str_replace('\\','\\\\',$s);
+ $s = str_replace('(','\\(',$s);
+ $s = str_replace(')','\\)',$s);
+ $s = str_replace("\r",'\\r',$s);
return $s;
}
function _textstring($s)
{
- //Format a text string
+ // Format a text string
return '('.$this->_escape($s).')';
}
function _UTF8toUTF16($s)
{
- //Convert UTF-8 to UTF-16BE with BOM
- $res="\xFE\xFF";
- $nb=strlen($s);
- $i=0;
+ // Convert UTF-8 to UTF-16BE with BOM
+ $res = "\xFE\xFF";
+ $nb = strlen($s);
+ $i = 0;
while($i<$nb)
{
- $c1=ord($s[$i++]);
+ $c1 = ord($s[$i++]);
if($c1>=224)
{
- //3-byte character
- $c2=ord($s[$i++]);
- $c3=ord($s[$i++]);
- $res.=chr((($c1 & 0x0F)<<4) + (($c2 & 0x3C)>>2));
- $res.=chr((($c2 & 0x03)<<6) + ($c3 & 0x3F));
+ // 3-byte character
+ $c2 = ord($s[$i++]);
+ $c3 = ord($s[$i++]);
+ $res .= chr((($c1 & 0x0F)<<4) + (($c2 & 0x3C)>>2));
+ $res .= chr((($c2 & 0x03)<<6) + ($c3 & 0x3F));
}
elseif($c1>=192)
{
- //2-byte character
- $c2=ord($s[$i++]);
- $res.=chr(($c1 & 0x1C)>>2);
- $res.=chr((($c1 & 0x03)<<6) + ($c2 & 0x3F));
+ // 2-byte character
+ $c2 = ord($s[$i++]);
+ $res .= chr(($c1 & 0x1C)>>2);
+ $res .= chr((($c1 & 0x03)<<6) + ($c2 & 0x3F));
}
else
{
- //Single-byte character
- $res.="\0".chr($c1);
+ // Single-byte character
+ $res .= "\0".chr($c1);
}
}
return $res;
@@ -1357,64 +1411,67 @@
function _dounderline($x, $y, $txt)
{
- //Underline text
- $up=$this->CurrentFont['up'];
- $ut=$this->CurrentFont['ut'];
- $w=$this->GetStringWidth($txt)+$this->ws*substr_count($txt,' ');
+ // Underline text
+ $up = $this->CurrentFont['up'];
+ $ut = $this->CurrentFont['ut'];
+ $w = $this->GetStringWidth($txt)+$this->ws*substr_count($txt,' ');
return sprintf('%.2F %.2F %.2F %.2F re
f',$x*$this->k,($this->h-($y-$up/1000*$this->FontSize))*$this->k,$w*$this->k,-$ut/1000*$this->FontSizePt);
}
function _parsejpg($file)
{
- //Extract info from a JPEG file
- $a=GetImageSize($file);
+ // Extract info from a JPEG file
+ $a = getimagesize($file);
if(!$a)
$this->Error('Missing or incorrect image file: '.$file);
if($a[2]!=2)
$this->Error('Not a JPEG file: '.$file);
if(!isset($a['channels']) || $a['channels']==3)
- $colspace='DeviceRGB';
+ $colspace = 'DeviceRGB';
elseif($a['channels']==4)
- $colspace='DeviceCMYK';
+ $colspace = 'DeviceCMYK';
else
- $colspace='DeviceGray';
- $bpc=isset($a['bits']) ? $a['bits'] : 8;
- //Read whole file
- $f=fopen($file,'rb');
- $data='';
- while(!feof($f))
- $data.=fread($f,8192);
- fclose($f);
+ $colspace = 'DeviceGray';
+ $bpc = isset($a['bits']) ? $a['bits'] : 8;
+ $data = file_get_contents($file);
return array('w'=>$a[0], 'h'=>$a[1], 'cs'=>$colspace, 'bpc'=>$bpc,
'f'=>'DCTDecode', 'data'=>$data);
}
function _parsepng($file)
{
- //Extract info from a PNG file
- $f=fopen($file,'rb');
+ // Extract info from a PNG file
+ $f = fopen($file,'rb');
if(!$f)
$this->Error('Can\'t open image file: '.$file);
- //Check signature
+ $info = $this->_parsepngstream($f,$file);
+ fclose($f);
+ return $info;
+}
+
+function _parsepngstream($f, $file)
+{
+ // Check signature
if($this->_readstream($f,8)!=chr(137).'PNG'.chr(13).chr(10).chr(26).chr(10))
$this->Error('Not a PNG file: '.$file);
- //Read header chunk
+
+ // Read header chunk
$this->_readstream($f,4);
if($this->_readstream($f,4)!='IHDR')
$this->Error('Incorrect PNG file: '.$file);
- $w=$this->_readint($f);
- $h=$this->_readint($f);
- $bpc=ord($this->_readstream($f,1));
+ $w = $this->_readint($f);
+ $h = $this->_readint($f);
+ $bpc = ord($this->_readstream($f,1));
if($bpc>8)
$this->Error('16-bit depth not supported: '.$file);
- $ct=ord($this->_readstream($f,1));
- if($ct==0)
- $colspace='DeviceGray';
- elseif($ct==2)
- $colspace='DeviceRGB';
+ $ct = ord($this->_readstream($f,1));
+ if($ct==0 || $ct==4)
+ $colspace = 'DeviceGray';
+ elseif($ct==2 || $ct==6)
+ $colspace = 'DeviceRGB';
elseif($ct==3)
- $colspace='Indexed';
+ $colspace = 'Indexed';
else
- $this->Error('Alpha channel not supported: '.$file);
+ $this->Error('Unknown color type: '.$file);
if(ord($this->_readstream($f,1))!=0)
$this->Error('Unknown compression method: '.$file);
if(ord($this->_readstream($f,1))!=0)
@@ -1422,41 +1479,42 @@
if(ord($this->_readstream($f,1))!=0)
$this->Error('Interlacing not supported: '.$file);
$this->_readstream($f,4);
- $parms='/DecodeParms <</Predictor 15 /Colors '.($ct==2 ? 3 : 1).'
/BitsPerComponent '.$bpc.' /Columns '.$w.'>>';
- //Scan chunks looking for palette, transparency and image data
- $pal='';
- $trns='';
- $data='';
+ $dp = '/Predictor 15 /Colors '.($colspace=='DeviceRGB' ? 3 : 1).'
/BitsPerComponent '.$bpc.' /Columns '.$w;
+
+ // Scan chunks looking for palette, transparency and image data
+ $pal = '';
+ $trns = '';
+ $data = '';
do
{
- $n=$this->_readint($f);
- $type=$this->_readstream($f,4);
+ $n = $this->_readint($f);
+ $type = $this->_readstream($f,4);
if($type=='PLTE')
{
- //Read palette
- $pal=$this->_readstream($f,$n);
+ // Read palette
+ $pal = $this->_readstream($f,$n);
$this->_readstream($f,4);
}
elseif($type=='tRNS')
{
- //Read transparency info
- $t=$this->_readstream($f,$n);
+ // Read transparency info
+ $t = $this->_readstream($f,$n);
if($ct==0)
- $trns=array(ord(substr($t,1,1)));
+ $trns = array(ord(substr($t,1,1)));
elseif($ct==2)
- $trns=array(ord(substr($t,1,1)),
ord(substr($t,3,1)), ord(substr($t,5,1)));
+ $trns = array(ord(substr($t,1,1)),
ord(substr($t,3,1)), ord(substr($t,5,1)));
else
{
- $pos=strpos($t,chr(0));
+ $pos = strpos($t,chr(0));
if($pos!==false)
- $trns=array($pos);
+ $trns = array($pos);
}
$this->_readstream($f,4);
}
elseif($type=='IDAT')
{
- //Read image data block
- $data.=$this->_readstream($f,$n);
+ // Read image data block
+ $data .= $this->_readstream($f,$n);
$this->_readstream($f,4);
}
elseif($type=='IEND')
@@ -1465,23 +1523,67 @@
$this->_readstream($f,$n+4);
}
while($n);
+
if($colspace=='Indexed' && empty($pal))
$this->Error('Missing palette in '.$file);
- fclose($f);
- return array('w'=>$w, 'h'=>$h, 'cs'=>$colspace, 'bpc'=>$bpc,
'f'=>'FlateDecode', 'parms'=>$parms, 'pal'=>$pal, 'trns'=>$trns, 'data'=>$data);
+ $info = array('w'=>$w, 'h'=>$h, 'cs'=>$colspace, 'bpc'=>$bpc,
'f'=>'FlateDecode', 'dp'=>$dp, 'pal'=>$pal, 'trns'=>$trns);
+ if($ct>=4)
+ {
+ // Extract alpha channel
+ if(!function_exists('gzuncompress'))
+ $this->Error('Zlib not available, can\'t handle alpha
channel: '.$file);
+ $data = gzuncompress($data);
+ $color = '';
+ $alpha = '';
+ if($ct==4)
+ {
+ // Gray image
+ $len = 2*$w;
+ for($i=0;$i<$h;$i++)
+ {
+ $pos = (1+$len)*$i;
+ $color .= $data[$pos];
+ $alpha .= $data[$pos];
+ $line = substr($data,$pos+1,$len);
+ $color .= preg_replace('/(.)./s','$1',$line);
+ $alpha .= preg_replace('/.(.)/s','$1',$line);
+ }
+ }
+ else
+ {
+ // RGB image
+ $len = 4*$w;
+ for($i=0;$i<$h;$i++)
+ {
+ $pos = (1+$len)*$i;
+ $color .= $data[$pos];
+ $alpha .= $data[$pos];
+ $line = substr($data,$pos+1,$len);
+ $color .= preg_replace('/(.{3})./s','$1',$line);
+ $alpha .= preg_replace('/.{3}(.)/s','$1',$line);
+ }
+ }
+ unset($data);
+ $data = gzcompress($color);
+ $info['smask'] = gzcompress($alpha);
+ if($this->PDFVersion<'1.4')
+ $this->PDFVersion = '1.4';
+ }
+ $info['data'] = $data;
+ return $info;
}
function _readstream($f, $n)
{
- //Read n bytes from stream
- $res='';
+ // Read n bytes from stream
+ $res = '';
while($n>0 && !feof($f))
{
- $s=fread($f,$n);
+ $s = fread($f,$n);
if($s===false)
$this->Error('Error while reading stream');
- $n-=strlen($s);
- $res.=$s;
+ $n -= strlen($s);
+ $res .= $s;
}
if($n>0)
$this->Error('Unexpected end of stream');
@@ -1490,38 +1592,55 @@
function _readint($f)
{
- //Read a 4-byte integer from stream
- $a=unpack('Ni',$this->_readstream($f,4));
+ // Read a 4-byte integer from stream
+ $a = unpack('Ni',$this->_readstream($f,4));
return $a['i'];
}
function _parsegif($file)
{
- //Extract info from a GIF file (via PNG conversion)
+ // Extract info from a GIF file (via PNG conversion)
if(!function_exists('imagepng'))
$this->Error('GD extension is required for GIF support');
if(!function_exists('imagecreatefromgif'))
$this->Error('GD has no GIF read support');
- $im=imagecreatefromgif($file);
+ $im = imagecreatefromgif($file);
if(!$im)
$this->Error('Missing or incorrect image file: '.$file);
imageinterlace($im,0);
- $tmp=tempnam('.','gif');
- if(!$tmp)
- $this->Error('Unable to create a temporary file');
- if(!imagepng($im,$tmp))
- $this->Error('Error while saving to temporary file');
- imagedestroy($im);
- $info=$this->_parsepng($tmp);
- unlink($tmp);
+ $f = @fopen('php://temp','rb+');
+ if($f)
+ {
+ // Perform conversion in memory
+ ob_start();
+ imagepng($im);
+ $data = ob_get_clean();
+ imagedestroy($im);
+ fwrite($f,$data);
+ rewind($f);
+ $info = $this->_parsepngstream($f,$file);
+ fclose($f);
+ }
+ else
+ {
+ // Use temporary file
+ $tmp = tempnam('.','gif');
+ if(!$tmp)
+ $this->Error('Unable to create a temporary file');
+ if(!imagepng($im,$tmp))
+ $this->Error('Error while saving to temporary file');
+ imagedestroy($im);
+ $info = $this->_parsepng($tmp);
+ unlink($tmp);
+ }
return $info;
}
function _newobj()
{
- //Begin a new object
+ // Begin a new object
$this->n++;
- $this->offsets[$this->n]=strlen($this->buffer);
+ $this->offsets[$this->n] = strlen($this->buffer);
$this->_out($this->n.' 0 obj');
}
@@ -1534,46 +1653,41 @@
function _out($s)
{
- //Add a line to the document
+ // Add a line to the document
if($this->state==2)
- $this->pages[$this->page].=$s."\n";
+ $this->pages[$this->page] .= $s."\n";
else
- $this->buffer.=$s."\n";
+ $this->buffer .= $s."\n";
}
function _putpages()
{
- $nb=$this->page;
+ $nb = $this->page;
if(!empty($this->AliasNbPages))
{
- //Replace number of pages in fonts using subsets
- $r = '';
- $nstr = "$nb";
- for($i=0;$i<strlen($nstr);$i++) {
- $r .= sprintf("%02s",
strtoupper(dechex(intval($nstr{$i})+33)));
- }
+ // Replace number of pages in fonts using subsets
+ $alias = $this->UTF8ToUTF16BE($this->AliasNbPages, false);
+ $r = $this->UTF8ToUTF16BE("$nb", false);
for($n=1;$n<=$nb;$n++)
-
$this->pages[$n]=str_replace('`'.$this->AliasNbPages.'`',$r,$this->pages[$n]);
+ $this->pages[$n] =
str_replace($alias,$r,$this->pages[$n]);
// Now repeat for no pages in non-subset fonts
- $r = $nb;
- //Replace number of pages
for($n=1;$n<=$nb;$n++)
-
$this->pages[$n]=str_replace($this->AliasNbPages,$r,$this->pages[$n]);
+ $this->pages[$n] =
str_replace($this->AliasNbPages,$nb,$this->pages[$n]);
}
if($this->DefOrientation=='P')
{
- $wPt=$this->DefPageFormat[0]*$this->k;
- $hPt=$this->DefPageFormat[1]*$this->k;
+ $wPt = $this->DefPageSize[0]*$this->k;
+ $hPt = $this->DefPageSize[1]*$this->k;
}
else
{
- $wPt=$this->DefPageFormat[1]*$this->k;
- $hPt=$this->DefPageFormat[0]*$this->k;
+ $wPt = $this->DefPageSize[1]*$this->k;
+ $hPt = $this->DefPageSize[0]*$this->k;
}
- $filter=($this->compress) ? '/Filter /FlateDecode ' : '';
+ $filter = ($this->compress) ? '/Filter /FlateDecode ' : '';
for($n=1;$n<=$nb;$n++)
{
- //Page
+ // Page
$this->_newobj();
$this->_out('<</Type /Page');
$this->_out('/Parent 1 0 R');
@@ -1582,39 +1696,41 @@
$this->_out('/Resources 2 0 R');
if(isset($this->PageLinks[$n]))
{
- //Links
- $annots='/Annots [';
+ // Links
+ $annots = '/Annots [';
foreach($this->PageLinks[$n] as $pl)
{
- $rect=sprintf('%.2F %.2F %.2F
%.2F',$pl[0],$pl[1],$pl[0]+$pl[2],$pl[1]-$pl[3]);
- $annots.='<</Type /Annot /Subtype /Link /Rect
['.$rect.'] /Border [0 0 0] ';
+ $rect = sprintf('%.2F %.2F %.2F
%.2F',$pl[0],$pl[1],$pl[0]+$pl[2],$pl[1]-$pl[3]);
+ $annots .= '<</Type /Annot /Subtype /Link /Rect
['.$rect.'] /Border [0 0 0] ';
if(is_string($pl[4]))
- $annots.='/A <</S /URI /URI
'.$this->_textstring($pl[4]).'>>>>';
+ $annots .= '/A <</S /URI /URI
'.$this->_textstring($pl[4]).'>>>>';
else
{
- $l=$this->links[$pl[4]];
- $h=isset($this->PageSizes[$l[0]]) ?
$this->PageSizes[$l[0]][1] : $hPt;
- $annots.=sprintf('/Dest [%d 0 R /XYZ 0
%.2F null]>>',1+2*$l[0],$h-$l[1]*$this->k);
+ $l = $this->links[$pl[4]];
+ $h = isset($this->PageSizes[$l[0]]) ?
$this->PageSizes[$l[0]][1] : $hPt;
+ $annots .= sprintf('/Dest [%d 0 R /XYZ
0 %.2F null]>>',1+2*$l[0],$h-$l[1]*$this->k);
}
}
$this->_out($annots.']');
}
+ if($this->PDFVersion>'1.3')
+ $this->_out('/Group <</Type /Group /S /Transparency /CS
/DeviceRGB>>');
$this->_out('/Contents '.($this->n+1).' 0 R>>');
$this->_out('endobj');
- //Page content
- $p=($this->compress) ? gzcompress($this->pages[$n]) :
$this->pages[$n];
+ // Page content
+ $p = ($this->compress) ? gzcompress($this->pages[$n]) :
$this->pages[$n];
$this->_newobj();
$this->_out('<<'.$filter.'/Length '.strlen($p).'>>');
$this->_putstream($p);
$this->_out('endobj');
}
- //Pages root
- $this->offsets[1]=strlen($this->buffer);
+ // Pages root
+ $this->offsets[1] = strlen($this->buffer);
$this->_out('1 0 obj');
$this->_out('<</Type /Pages');
- $kids='/Kids [';
+ $kids = '/Kids [';
for($i=0;$i<$nb;$i++)
- $kids.=(3+2*$i).' 0 R ';
+ $kids .= (3+2*$i).' 0 R ';
$this->_out($kids.']');
$this->_out('/Count '.$nb);
$this->_out(sprintf('/MediaBox [0 0 %.2F %.2F]',$wPt,$hPt));
@@ -1627,15 +1743,15 @@
$nf=$this->n;
foreach($this->diffs as $diff)
{
- //Encodings
+ // Encodings
$this->_newobj();
$this->_out('<</Type /Encoding /BaseEncoding /WinAnsiEncoding
/Differences ['.$diff.']>>');
$this->_out('endobj');
}
foreach($this->FontFiles as $file=>$info)
{
- if (!isset($info['type']) || $info['type']!='TrueTypesubset') {
- //Font file embedding
+ if (!isset($info['type']) || $info['type']!='TTF') {
+ // Font file embedding
$this->_newobj();
$this->FontFiles[$file]['n']=$this->n;
$font='';
@@ -1651,12 +1767,12 @@
$header=(ord($font[0])==128);
if($header)
{
- //Strip first binary header
+ // Strip first binary header
$font=substr($font,6);
}
if($header && ord($font[$info['length1']])==128)
{
- //Strip second binary header
+ // Strip second binary header
$font=substr($font,0,$info['length1']).substr($font,$info['length1']+6);
}
}
@@ -1673,13 +1789,13 @@
}
foreach($this->fonts as $k=>$font)
{
- //Font objects
+ // Font objects
//$this->fonts[$k]['n']=$this->n+1;
- $type=$font['type'];
- $name=$font['name'];
- if($type=='core')
+ $type = $font['type'];
+ $name = $font['name'];
+ if($type=='Core')
{
- //Standard font
+ // Standard font
$this->fonts[$k]['n']=$this->n+1;
$this->_newobj();
$this->_out('<</Type /Font');
@@ -1692,7 +1808,7 @@
}
elseif($type=='Type1' || $type=='TrueType')
{
- //Additional Type1 or TrueType font
+ // Additional Type1 or TrueType font
$this->fonts[$k]['n']=$this->n+1;
$this->_newobj();
$this->_out('<</Type /Font');
@@ -1710,7 +1826,7 @@
}
$this->_out('>>');
$this->_out('endobj');
- //Widths
+ // Widths
$this->_newobj();
$cw=&$font['cw'];
$s='[';
@@ -1718,7 +1834,7 @@
$s.=$cw[chr($i)].' ';
$this->_out($s.']');
$this->_out('endobj');
- //Descriptor
+ // Descriptor
$this->_newobj();
$s='<</Type /FontDescriptor /FontName /'.$name;
foreach($font['desc'] as $k=>$v)
@@ -1729,65 +1845,53 @@
$this->_out($s.'>>');
$this->_out('endobj');
}
- // TrueType embedded SUBSETS
- else if ($type=='TrueTypesubset') {
- $ssfaid="A";
- include_once($this->_getfontpath().'unifont/ttfonts.php');
- $ttf = new TTFontFile();
- $ttf->getMetrics($font['file'], 1);
- for($sfid=0;$sfid<count($font['subsetfontids']);$sfid++) {
- $this->fonts[$k]['n'][$sfid]=$this->n+1;
// NB an array for subset
- $subsetname = 'MPDFA'.$ssfaid.'+'.$font['name'];
- $ssfaid++;
- $subset = $font['subsets'][$sfid];
+ // TrueType embedded SUBSETS or FULL
+ else if ($type=='TTF') {
+ $this->fonts[$k]['n']=$this->n+1;
+
require_once($this->_getfontpath().'unifont/ttfonts.php');
+ $ttf = new TTFontFile();
+ $fontname = 'MPDFAA'.'+'.$font['name'];
+ $subset = $font['subset'];
unset($subset[0]);
- $ttfontstream = $ttf->makeSubset($subset);
+ $ttfontstream = $ttf->makeSubset($font['ttffile'],
$subset);
$ttfontsize = strlen($ttfontstream);
$fontstream = gzcompress($ttfontstream);
- $widthstring = '';
- $toUnistring = '';
- foreach($font['subsets'][$sfid] AS $cp=>$u) {
- if (isset($font['cw'][$u])) {
- $widthstring .= $font['cw'][$u].' ';
- }
- else {
- $widthstring .= $ttf->defaultWidth.' ';
- }
- $toUnistring .= sprintf("<%02s> <%04s>\n",
strtoupper(dechex($cp)), strtoupper(dechex($u)));
- }
+ $codeToGlyph = $ttf->codeToGlyph;
+ unset($codeToGlyph[0]);
- //Additional Type1 or TrueType font
+ // Type0 Font
+ // A composite font - a font composed of other fonts,
organized hierarchically
$this->_newobj();
$this->_out('<</Type /Font');
- $this->_out('/BaseFont /'.$subsetname);
- $this->_out('/Subtype /TrueType');
- $this->_out('/FirstChar 0 /LastChar
'.(count($font['subsets'][$sfid])));
- $this->_out('/Widths '.($this->n+1).' 0 R');
- $this->_out('/FontDescriptor '.($this->n+2).' 0 R');
- $this->_out('/ToUnicode '.($this->n + 3).' 0 R');
+ $this->_out('/Subtype /Type0');
+ $this->_out('/BaseFont /'.$fontname.'');
+ $this->_out('/Encoding /Identity-H');
+ $this->_out('/DescendantFonts ['.($this->n + 1).' 0
R]');
+ $this->_out('/ToUnicode '.($this->n + 2).' 0 R');
$this->_out('>>');
$this->_out('endobj');
- //Widths
+ // CIDFontType2
+ // A CIDFont whose glyph descriptions are based on
TrueType font technology
$this->_newobj();
- $this->_out('['.$widthstring.']');
- $this->_out('endobj');
-
- //Descriptor
- $this->_newobj();
- $s='<</Type /FontDescriptor /FontName
/'.$subsetname."\n";
- foreach($font['desc'] as $kd=>$v) {
- if ($kd == 'Flags') { $v = $v | 4; $v = $v &
~32; }
- $s.=' /'.$kd.' '.$v."\n";
+ $this->_out('<</Type /Font');
+ $this->_out('/Subtype /CIDFontType2');
+ $this->_out('/BaseFont /'.$fontname.'');
+ $this->_out('/CIDSystemInfo '.($this->n + 2).' 0 R');
+ $this->_out('/FontDescriptor '.($this->n + 3).' 0 R');
+ if (isset($font['desc']['MissingWidth'])){
+ $this->_out('/DW
'.$font['desc']['MissingWidth'].'');
}
- $s.='/FontFile2 '.($this->n + 2).' 0 R';
- $this->_out($s.'>>');
+ $this->_putTTfontwidths($font, $ttf->maxUni);
+
+ $this->_out('/CIDToGIDMap '.($this->n + 4).' 0 R');
+ $this->_out('>>');
$this->_out('endobj');
// ToUnicode
- $toUni = "stream\n";
- $toUni .= "/CIDInit /ProcSet findresource begin\n";
+ $this->_newobj();
+ $toUni = "/CIDInit /ProcSet findresource begin\n";
$toUni .= "12 dict begin\n";
$toUni .= "begincmap\n";
$toUni .= "/CIDSystemInfo\n";
@@ -1798,21 +1902,55 @@
$toUni .= "/CMapName /Adobe-Identity-UCS def\n";
$toUni .= "/CMapType 2 def\n";
$toUni .= "1 begincodespacerange\n";
- $toUni .= "<00> <FF>\n";
+ $toUni .= "<0000> <FFFF>\n";
$toUni .= "endcodespacerange\n";
- $toUni .= count($font['subsets'][$sfid])."
beginbfchar\n";
- $toUni .= $toUnistring;
- $toUni .= "endbfchar\n";
+ $toUni .= "1 beginbfrange\n";
+ $toUni .= "<0000> <FFFF> <0000>\n";
+ $toUni .= "endbfrange\n";
$toUni .= "endcmap\n";
$toUni .= "CMapName currentdict /CMap defineresource
pop\n";
$toUni .= "end\n";
- $toUni .= "end\n";
- $toUni .= "endstream\n";
- $toUni .= "endobj";
+ $toUni .= "end";
+ $this->_out('<</Length '.(strlen($toUni)).'>>');
+ $this->_putstream($toUni);
+ $this->_out('endobj');
+
+ // CIDSystemInfo dictionary
$this->_newobj();
- $this->_out('<</Length '.(strlen($toUni)-24).'>>');
- $this->_out($toUni);
+ $this->_out('<</Registry (Adobe)');
+ $this->_out('/Ordering (UCS)');
+ $this->_out('/Supplement 0');
+ $this->_out('>>');
+ $this->_out('endobj');
+ // Font descriptor
+ $this->_newobj();
+ $this->_out('<</Type /FontDescriptor');
+ $this->_out('/FontName /'.$fontname);
+ foreach($font['desc'] as $kd=>$v) {
+ if ($kd == 'Flags') { $v = $v | 4; $v = $v &
~32; } // SYMBOLIC font flag
+ $this->_out(' /'.$kd.' '.$v);
+ }
+ $this->_out('/FontFile2 '.($this->n + 2).' 0 R');
+ $this->_out('>>');
+ $this->_out('endobj');
+
+ // Embed CIDToGIDMap
+ // A specification of the mapping from CIDs to glyph
indices
+ $cidtogidmap = '';
+ $cidtogidmap = str_pad('', 256*256*2, "\x00");
+ foreach($codeToGlyph as $cc=>$glyph) {
+ $cidtogidmap[$cc*2] = chr($glyph >> 8);
+ $cidtogidmap[$cc*2 + 1] = chr($glyph & 0xFF);
+ }
+ $cidtogidmap = gzcompress($cidtogidmap);
+ $this->_newobj();
+ $this->_out('<</Length '.strlen($cidtogidmap).'');
+ $this->_out('/Filter /FlateDecode');
+ $this->_out('>>');
+ $this->_putstream($cidtogidmap);
+ $this->_out('endobj');
+
//Font file
$this->_newobj();
$this->_out('<</Length '.strlen($fontstream));
@@ -1821,13 +1959,12 @@
$this->_out('>>');
$this->_putstream($fontstream);
$this->_out('endobj');
- }
- unset($ttf);
+ unset($ttf);
}
else
{
- //Allow for additional types
- $this->fonts[$k]['n']=$this->n+1;
+ // Allow for additional types
+ $this->fonts[$k]['n'] = $this->n+1;
$mtd='_put'.strtolower($type);
if(!method_exists($this,$mtd))
$this->Error('Unsupported font type: '.$type);
@@ -1836,51 +1973,165 @@
}
}
+function _putTTfontwidths(&$font, $maxUni) {
+ if (file_exists($font['unifilename'].'.cw127.php')) {
+ include($font['unifilename'].'.cw127.php') ;
+ $startcid = 128;
+ }
+ else {
+ $rangeid = 0;
+ $range = array();
+ $prevcid = -2;
+ $prevwidth = -1;
+ $interval = false;
+ $startcid = 1;
+ }
+ $cwlen = $maxUni + 1;
+
+ // for each character
+ for ($cid=$startcid; $cid<$cwlen; $cid++) {
+ if ($cid==128 &&
(!file_exists($font['unifilename'].'.cw127.php'))) {
+ if
(is_writable(dirname($this->_getfontpath().'unifont/x'))) {
+ $fh =
fopen($font['unifilename'].'.cw127.php',"wb");
+ $cw127='<?php'."\n";
+ $cw127.='$rangeid='.$rangeid.";\n";
+ $cw127.='$prevcid='.$prevcid.";\n";
+ $cw127.='$prevwidth='.$prevwidth.";\n";
+ if ($interval) {
$cw127.='$interval=true'.";\n"; }
+ else { $cw127.='$interval=false'.";\n"; }
+ $cw127.='$range='.var_export($range,true).";\n";
+ $cw127.="?>";
+ fwrite($fh,$cw127,strlen($cw127));
+ fclose($fh);
+ }
+ }
+ if ($font['cw'][$cid*2] == "\00" && $font['cw'][$cid*2+1] ==
"\00") { continue; }
+ $width = (ord($font['cw'][$cid*2]) << 8) +
ord($font['cw'][$cid*2+1]);
+ if ($width == 65535) { $width = 0; }
+ if ($cid > 255 && (!isset($font['subset'][$cid]) ||
!$font['subset'][$cid])) { continue; }
+ if (!isset($font['dw']) || (isset($font['dw']) && $width !=
$font['dw'])) {
+ if ($cid == ($prevcid + 1)) {
+ if ($width == $prevwidth) {
+ if ($width == $range[$rangeid][0]) {
+ $range[$rangeid][] = $width;
+ }
+ else {
+ array_pop($range[$rangeid]);
+ // new range
+ $rangeid = $prevcid;
+ $range[$rangeid] = array();
+ $range[$rangeid][] = $prevwidth;
+ $range[$rangeid][] = $width;
+ }
+ $interval = true;
+ $range[$rangeid]['interval'] = true;
+ } else {
+ if ($interval) {
+ // new range
+ $rangeid = $cid;
+ $range[$rangeid] = array();
+ $range[$rangeid][] = $width;
+ }
+ else { $range[$rangeid][] = $width; }
+ $interval = false;
+ }
+ } else {
+ $rangeid = $cid;
+ $range[$rangeid] = array();
+ $range[$rangeid][] = $width;
+ $interval = false;
+ }
+ $prevcid = $cid;
+ $prevwidth = $width;
+ }
+ }
+ $prevk = -1;
+ $nextk = -1;
+ $prevint = false;
+ foreach ($range as $k => $ws) {
+ $cws = count($ws);
+ if (($k == $nextk) AND (!$prevint) AND
((!isset($ws['interval'])) OR ($cws < 4))) {
+ if (isset($range[$k]['interval'])) {
unset($range[$k]['interval']); }
+ $range[$prevk] = array_merge($range[$prevk],
$range[$k]);
+ unset($range[$k]);
+ }
+ else { $prevk = $k; }
+ $nextk = $k + $cws;
+ if (isset($ws['interval'])) {
+ if ($cws > 3) { $prevint = true; }
+ else { $prevint = false; }
+ unset($range[$k]['interval']);
+ --$nextk;
+ }
+ else { $prevint = false; }
+ }
+ $w = '';
+ foreach ($range as $k => $ws) {
+ if (count(array_count_values($ws)) == 1) { $w .= ' '.$k.' '.($k
+ count($ws) - 1).' '.$ws[0]; }
+ else { $w .= ' '.$k.' [ '.implode(' ', $ws).' ]' . "\n"; }
+ }
+ $this->_out('/W ['.$w.' ]');
+}
+
function _putimages()
{
- $filter=($this->compress) ? '/Filter /FlateDecode ' : '';
- reset($this->images);
- while(list($file,$info)=each($this->images))
+ foreach(array_keys($this->images) as $file)
{
+ $this->_putimage($this->images[$file]);
+ unset($this->images[$file]['data']);
+ unset($this->images[$file]['smask']);
+ }
+}
+
+function _putimage(&$info)
+{
+ $this->_newobj();
+ $info['n'] = $this->n;
+ $this->_out('<</Type /XObject');
+ $this->_out('/Subtype /Image');
+ $this->_out('/Width '.$info['w']);
+ $this->_out('/Height '.$info['h']);
+ if($info['cs']=='Indexed')
+ $this->_out('/ColorSpace [/Indexed /DeviceRGB
'.(strlen($info['pal'])/3-1).' '.($this->n+1).' 0 R]');
+ else
+ {
+ $this->_out('/ColorSpace /'.$info['cs']);
+ if($info['cs']=='DeviceCMYK')
+ $this->_out('/Decode [1 0 1 0 1 0 1 0]');
+ }
+ $this->_out('/BitsPerComponent '.$info['bpc']);
+ if(isset($info['f']))
+ $this->_out('/Filter /'.$info['f']);
+ if(isset($info['dp']))
+ $this->_out('/DecodeParms <<'.$info['dp'].'>>');
+ if(isset($info['trns']) && is_array($info['trns']))
+ {
+ $trns = '';
+ for($i=0;$i<count($info['trns']);$i++)
+ $trns .= $info['trns'][$i].' '.$info['trns'][$i].' ';
+ $this->_out('/Mask ['.$trns.']');
+ }
+ if(isset($info['smask']))
+ $this->_out('/SMask '.($this->n+1).' 0 R');
+ $this->_out('/Length '.strlen($info['data']).'>>');
+ $this->_putstream($info['data']);
+ $this->_out('endobj');
+ // Soft mask
+ if(isset($info['smask']))
+ {
+ $dp = '/Predictor 15 /Colors 1 /BitsPerComponent 8 /Columns
'.$info['w'];
+ $smask = array('w'=>$info['w'], 'h'=>$info['h'],
'cs'=>'DeviceGray', 'bpc'=>8, 'f'=>$info['f'], 'dp'=>$dp,
'data'=>$info['smask']);
+ $this->_putimage($smask);
+ }
+ // Palette
+ if($info['cs']=='Indexed')
+ {
+ $filter = ($this->compress) ? '/Filter /FlateDecode ' : '';
+ $pal = ($this->compress) ? gzcompress($info['pal']) :
$info['pal'];
$this->_newobj();
- $this->images[$file]['n']=$this->n;
- $this->_out('<</Type /XObject');
- $this->_out('/Subtype /Image');
- $this->_out('/Width '.$info['w']);
- $this->_out('/Height '.$info['h']);
- if($info['cs']=='Indexed')
- $this->_out('/ColorSpace [/Indexed /DeviceRGB
'.(strlen($info['pal'])/3-1).' '.($this->n+1).' 0 R]');
- else
- {
- $this->_out('/ColorSpace /'.$info['cs']);
- if($info['cs']=='DeviceCMYK')
- $this->_out('/Decode [1 0 1 0 1 0 1 0]');
- }
- $this->_out('/BitsPerComponent '.$info['bpc']);
- if(isset($info['f']))
- $this->_out('/Filter /'.$info['f']);
- if(isset($info['parms']))
- $this->_out($info['parms']);
- if(isset($info['trns']) && is_array($info['trns']))
- {
- $trns='';
- for($i=0;$i<count($info['trns']);$i++)
- $trns.=$info['trns'][$i].'
'.$info['trns'][$i].' ';
- $this->_out('/Mask ['.$trns.']');
- }
- $this->_out('/Length '.strlen($info['data']).'>>');
- $this->_putstream($info['data']);
- unset($this->images[$file]['data']);
+ $this->_out('<<'.$filter.'/Length '.strlen($pal).'>>');
+ $this->_putstream($pal);
$this->_out('endobj');
- //Palette
- if($info['cs']=='Indexed')
- {
- $this->_newobj();
- $pal=($this->compress) ? gzcompress($info['pal']) :
$info['pal'];
- $this->_out('<<'.$filter.'/Length '.strlen($pal).'>>');
- $this->_putstream($pal);
- $this->_out('endobj');
- }
}
}
@@ -1895,14 +2146,7 @@
$this->_out('/ProcSet [/PDF /Text /ImageB /ImageC /ImageI]');
$this->_out('/Font <<');
foreach($this->fonts as $font) {
- if ($font['type']=='TrueTypesubset') {
- foreach($font['n'] AS $k => $fid) {
- $this->_out('/F'.$font['subsetfontids'][$k].'
'.$font['n'][$k].' 0 R');
- }
- }
- else {
- $this->_out('/F'.$font['i'].' '.$font['n'].' 0 R');
- }
+ $this->_out('/F'.$font['i'].' '.$font['n'].' 0 R');
}
$this->_out('>>');
$this->_out('/XObject <<');
@@ -1914,8 +2158,8 @@
{
$this->_putfonts();
$this->_putimages();
- //Resource dictionary
- $this->offsets[2]=strlen($this->buffer);
+ // Resource dictionary
+ $this->offsets[2] = strlen($this->buffer);
$this->_out('2 0 obj');
$this->_out('<<');
$this->_putresourcedict();
@@ -1950,7 +2194,7 @@
elseif($this->ZoomMode=='real')
$this->_out('/OpenAction [3 0 R /XYZ null null 1]');
elseif(!is_string($this->ZoomMode))
- $this->_out('/OpenAction [3 0 R /XYZ null null
'.($this->ZoomMode/100).']');
+ $this->_out('/OpenAction [3 0 R /XYZ null null
'.sprintf('%.2F',$this->ZoomMode/100).']');
if($this->LayoutMode=='single')
$this->_out('/PageLayout /SinglePage');
elseif($this->LayoutMode=='continuous')
@@ -1976,26 +2220,26 @@
$this->_putheader();
$this->_putpages();
$this->_putresources();
- //Info
+ // Info
$this->_newobj();
$this->_out('<<');
$this->_putinfo();
$this->_out('>>');
$this->_out('endobj');
- //Catalog
+ // Catalog
$this->_newobj();
$this->_out('<<');
$this->_putcatalog();
$this->_out('>>');
$this->_out('endobj');
- //Cross-ref
- $o=strlen($this->buffer);
+ // Cross-ref
+ $o = strlen($this->buffer);
$this->_out('xref');
$this->_out('0 '.($this->n+1));
$this->_out('0000000000 65535 f ');
for($i=1;$i<=$this->n;$i++)
$this->_out(sprintf('%010d 00000 n ',$this->offsets[$i]));
- //Trailer
+ // Trailer
$this->_out('trailer');
$this->_out('<<');
$this->_puttrailer();
@@ -2003,62 +2247,18 @@
$this->_out('startxref');
$this->_out($o);
$this->_out('%%EOF');
- $this->state=3;
+ $this->state = 3;
}
// ********* NEW FUNCTIONS *********
-
-// Convert utf-8 string to <HHHHHH> for Font Subsets
-function UTF8toSubset($str) {
- $ret = '<';
- if ($this->AliasNbPages)
- $str =
preg_replace('/'.preg_quote($this->AliasNbPages,'/').'/', chr(7), $str );
- $unicode = $this->UTF8StringToArray($str);
- $orig_fid = $this->CurrentFont['subsetfontids'][0];
- $last_fid = $this->CurrentFont['subsetfontids'][0];
- foreach($unicode as $c) {
- if ($c == 7) {
- if ($orig_fid != $last_fid) {
- $ret .= '> Tj /F'.$orig_fid.'
'.$this->FontSizePt.' Tf <';
- $last_fid = $orig_fid;
- }
- $ret .= '`'.$this->AliasNbPages.'`';
- continue;
- }
- for ($i=0; $i<99; $i++) {
- // return c as decimal char
- $init = array_search($c, $this->CurrentFont['subsets'][$i]);
- if ($init!==false) {
- if ($this->CurrentFont['subsetfontids'][$i] !=
$last_fid) {
- $ret .= '> Tj
/F'.$this->CurrentFont['subsetfontids'][$i].' '.$this->FontSizePt.' Tf <';
- $last_fid =
$this->CurrentFont['subsetfontids'][$i];
- }
- $ret .= sprintf("%02s", strtoupper(dechex($init)));
- break;
- }
- else if (count($this->CurrentFont['subsets'][$i]) < 255) {
- $n = count($this->CurrentFont['subsets'][$i]);
- $this->CurrentFont['subsets'][$i][$n] = $c;
- if ($this->CurrentFont['subsetfontids'][$i] !=
$last_fid) {
- $ret .= '> Tj
/F'.$this->CurrentFont['subsetfontids'][$i].' '.$this->FontSizePt.' Tf <';
- $last_fid =
$this->CurrentFont['subsetfontids'][$i];
- }
- $ret .= sprintf("%02s", strtoupper(dechex($n)));
- break;
- }
- else if (!isset($this->CurrentFont['subsets'][($i+1)])) {
- $this->CurrentFont['subsets'][($i+1)] = array(0=>0);
- $new_fid =
count($this->fonts)+$this->extraFontSubsets+1;
- $this->CurrentFont['subsetfontids'][($i+1)] = $new_fid;
- $this->extraFontSubsets++;
- }
- }
+// Converts UTF-8 strings to UTF16-BE.
+function UTF8ToUTF16BE($str, $setbom=true) {
+ $outstr = "";
+ if ($setbom) {
+ $outstr .= "\xFE\xFF"; // Byte Order Mark (BOM)
}
- $ret .= '>';
- if ($last_fid != $orig_fid) {
- $ret .= ' Tj /F'.$orig_fid.' '.$this->FontSizePt.' Tf <> ';
- }
- return $ret;
+ $outstr .= mb_convert_encoding($str, 'UTF-16BE', 'UTF-8');
+ return $outstr;
}
// Converts UTF-8 strings to codepoints array
@@ -2066,33 +2266,37 @@
$out = array();
$len = strlen($str);
for ($i = 0; $i < $len; $i++) {
+ $uni = -1;
$h = ord($str[$i]);
if ( $h <= 0x7F )
- $out[] = $h;
+ $uni = $h;
elseif ( $h >= 0xC2 ) {
if ( ($h <= 0xDF) && ($i < $len -1) )
- $out[] = ($h & 0x1F) << 6 | (ord($str{++$i}) & 0x3F);
+ $uni = ($h & 0x1F) << 6 | (ord($str[++$i]) & 0x3F);
elseif ( ($h <= 0xEF) && ($i < $len -2) )
- $out[] = ($h & 0x0F) << 12 | (ord($str{++$i}) & 0x3F) << 6
- | (ord($str{++$i}) & 0x3F);
+ $uni = ($h & 0x0F) << 12 | (ord($str[++$i]) & 0x3F) << 6
+ | (ord($str[++$i]) & 0x3F);
elseif ( ($h <= 0xF4) && ($i < $len -3) )
- $out[] = ($h & 0x0F) << 18 | (ord($str{++$i}) & 0x3F) << 12
- | (ord($str{++$i}) & 0x3F) << 6
- | (ord($str{++$i}) & 0x3F);
+ $uni = ($h & 0x0F) << 18 | (ord($str[++$i]) & 0x3F) << 12
+ | (ord($str[++$i]) & 0x3F) << 6
+ | (ord($str[++$i]) & 0x3F);
}
+ if ($uni >= 0) {
+ $out[] = $uni;
+ }
}
return $out;
}
-//End of class
+// End of class
}
-//Handle special IE contype request
+// Handle special IE contype request
if(isset($_SERVER['HTTP_USER_AGENT']) &&
$_SERVER['HTTP_USER_AGENT']=='contype')
{
header('Content-Type: application/pdf');
exit;
}
-?>
\ No newline at end of file
+?>
[Prev in Thread] |
Current Thread |
[Next in Thread] |
- [Phpcompta-dev] r4218 - in phpcompta/trunk/include/tfpdf: . font font/unifont,
phpcompta-dev <=