Fixed quote escaping problem in view.php.
[openemr.git] / library / classes / class.pdf.php
blob2ea21fccbe9a4e224fb37373f34e0e7c5f621c50
1 <?php
2 /**
3 * Cpdf
5 * http://www.ros.co.nz/pdf
7 * A PHP class to provide the basic functionality to create a pdf document without
8 * any requirement for additional modules.
10 * Note that they companion class CezPdf can be used to extend this class and dramatically
11 * simplify the creation of documents.
13 * IMPORTANT NOTE
14 * there is no warranty, implied or otherwise with this software.
16 * LICENCE
17 * This code has been placed in the Public Domain for all to enjoy.
19 * @author Wayne Munro <pdf@ros.co.nz>
20 * @version 009
21 * @package Cpdf
23 class Cpdf {
25 /**
26 * the current number of pdf objects in the document
28 var $numObj=0;
29 /**
30 * this array contains all of the pdf objects, ready for final assembly
32 var $objects = array();
33 /**
34 * the objectId (number within the objects array) of the document catalog
36 var $catalogId;
37 /**
38 * array carrying information about the fonts that the system currently knows about
39 * used to ensure that a font is not loaded twice, among other things
41 var $fonts=array();
42 /**
43 * a record of the current font
45 var $currentFont='';
46 /**
47 * the current base font
49 var $currentBaseFont='';
50 /**
51 * the number of the current font within the font array
53 var $currentFontNum=0;
54 /**
57 var $currentNode;
58 /**
59 * object number of the current page
61 var $currentPage;
62 /**
63 * object number of the currently active contents block
65 var $currentContents;
66 /**
67 * number of fonts within the system
69 var $numFonts=0;
70 /**
71 * current colour for fill operations, defaults to inactive value, all three components should be between 0 and 1 inclusive when active
73 var $currentColour=array('r'=>-1,'g'=>-1,'b'=>-1);
74 /**
75 * current colour for stroke operations (lines etc.)
77 var $currentStrokeColour=array('r'=>-1,'g'=>-1,'b'=>-1);
78 /**
79 * current style that lines are drawn in
81 var $currentLineStyle='';
82 /**
83 * an array which is used to save the state of the document, mainly the colours and styles
84 * it is used to temporarily change to another state, the change back to what it was before
86 var $stateStack = array();
87 /**
88 * number of elements within the state stack
90 var $nStateStack = 0;
91 /**
92 * number of page objects within the document
94 var $numPages=0;
95 /**
96 * object Id storage stack
98 var $stack=array();
99 /**
100 * number of elements within the object Id storage stack
102 var $nStack=0;
104 * an array which contains information about the objects which are not firmly attached to pages
105 * these have been added with the addObject function
107 var $looseObjects=array();
109 * array contains infomation about how the loose objects are to be added to the document
111 var $addLooseObjects=array();
113 * the objectId of the information object for the document
114 * this contains authorship, title etc.
116 var $infoObject=0;
118 * number of images being tracked within the document
120 var $numImages=0;
122 * an array containing options about the document
123 * it defaults to turning on the compression of the objects
125 var $options=array('compression'=>1);
127 * the objectId of the first page of the document
129 var $firstPageId;
131 * used to track the last used value of the inter-word spacing, this is so that it is known
132 * when the spacing is changed.
134 var $wordSpaceAdjust=0;
136 * the object Id of the procset object
138 var $procsetObjectId;
140 * store the information about the relationship between font families
141 * this used so that the code knows which font is the bold version of another font, etc.
142 * the value of this array is initialised in the constuctor function.
144 var $fontFamilies = array();
146 * track if the current font is bolded or italicised
148 var $currentTextState = '';
150 * messages are stored here during processing, these can be selected afterwards to give some useful debug information
152 var $messages='';
154 * the ancryption array for the document encryption is stored here
156 var $arc4='';
158 * the object Id of the encryption information
160 var $arc4_objnum=0;
162 * the file identifier, used to uniquely identify a pdf document
164 var $fileIdentifier='';
166 * a flag to say if a document is to be encrypted or not
168 var $encrypted=0;
170 * the ancryption key for the encryption of all the document content (structure is not encrypted)
172 var $encryptionKey='';
174 * array which forms a stack to keep track of nested callback functions
176 var $callback = array();
178 * the number of callback functions in the callback array
180 var $nCallback = 0;
182 * store label->id pairs for named destinations, these will be used to replace internal links
183 * done this way so that destinations can be defined after the location that links to them
185 var $destinations = array();
187 * store the stack for the transaction commands, each item in here is a record of the values of all the
188 * variables within the class, so that the user can rollback at will (from each 'start' command)
189 * note that this includes the objects array, so these can be large.
191 var $checkpoint = '';
193 * class constructor
194 * this will start a new document
195 * @var array array of 4 numbers, defining the bottom left and upper right corner of the page. first two are normally zero.
197 function Cpdf ($pageSize=array(0,0,612,792)){
198 $this->newDocument($pageSize);
200 // also initialize the font families that are known about already
201 $this->setFontFamily('init');
202 // $this->fileIdentifier = md5('xxxxxxxx'.time());
207 * Document object methods (internal use only)
209 * There is about one object method for each type of object in the pdf document
210 * Each function has the same call list ($id,$action,$options).
211 * $id = the object ID of the object, or what it is to be if it is being created
212 * $action = a string specifying the action to be performed, though ALL must support:
213 * 'new' - create the object with the id $id
214 * 'out' - produce the output for the pdf object
215 * $options = optional, a string or array containing the various parameters for the object
217 * These, in conjunction with the output function are the ONLY way for output to be produced
218 * within the pdf 'file'.
222 *destination object, used to specify the location for the user to jump to, presently on opening
224 function o_destination($id,$action,$options=''){
225 if ($action!='new'){
226 $o =& $this->objects[$id];
228 switch($action){
229 case 'new':
230 $this->objects[$id]=array('t'=>'destination','info'=>array());
231 $tmp = '';
232 switch ($options['type']){
233 case 'XYZ':
234 case 'FitR':
235 $tmp = ' '.$options['p3'].$tmp;
236 case 'FitH':
237 case 'FitV':
238 case 'FitBH':
239 case 'FitBV':
240 $tmp = ' '.$options['p1'].' '.$options['p2'].$tmp;
241 case 'Fit':
242 case 'FitB':
243 $tmp = $options['type'].$tmp;
244 $this->objects[$id]['info']['string']=$tmp;
245 $this->objects[$id]['info']['page']=$options['page'];
247 break;
248 case 'out':
249 $tmp = $o['info'];
250 $res="\n".$id." 0 obj\n".'['.$tmp['page'].' 0 R /'.$tmp['string']."]\nendobj\n";
251 return $res;
252 break;
257 * set the viewer preferences
259 function o_viewerPreferences($id,$action,$options=''){
260 if ($action!='new'){
261 $o =& $this->objects[$id];
263 switch ($action){
264 case 'new':
265 $this->objects[$id]=array('t'=>'viewerPreferences','info'=>array());
266 break;
267 case 'add':
268 foreach($options as $k=>$v){
269 switch ($k){
270 case 'HideToolbar':
271 case 'HideMenubar':
272 case 'HideWindowUI':
273 case 'FitWindow':
274 case 'CenterWindow':
275 case 'NonFullScreenPageMode':
276 case 'Direction':
277 $o['info'][$k]=$v;
278 break;
281 break;
282 case 'out':
284 $res="\n".$id." 0 obj\n".'<< ';
285 foreach($o['info'] as $k=>$v){
286 $res.="\n/".$k.' '.$v;
288 $res.="\n>>\n";
289 return $res;
290 break;
295 * define the document catalog, the overall controller for the document
297 function o_catalog($id,$action,$options=''){
298 if ($action!='new'){
299 $o =& $this->objects[$id];
301 switch ($action){
302 case 'new':
303 $this->objects[$id]=array('t'=>'catalog','info'=>array());
304 $this->catalogId=$id;
305 break;
306 case 'outlines':
307 case 'pages':
308 case 'openHere':
309 $o['info'][$action]=$options;
310 break;
311 case 'viewerPreferences':
312 if (!isset($o['info']['viewerPreferences'])){
313 $this->numObj++;
314 $this->o_viewerPreferences($this->numObj,'new');
315 $o['info']['viewerPreferences']=$this->numObj;
317 $vp = $o['info']['viewerPreferences'];
318 $this->o_viewerPreferences($vp,'add',$options);
319 break;
320 case 'out':
321 $res="\n".$id." 0 obj\n".'<< /Type /Catalog';
322 foreach($o['info'] as $k=>$v){
323 switch($k){
324 case 'outlines':
325 $res.="\n".'/Outlines '.$v.' 0 R';
326 break;
327 case 'pages':
328 $res.="\n".'/Pages '.$v.' 0 R';
329 break;
330 case 'viewerPreferences':
331 $res.="\n".'/ViewerPreferences '.$o['info']['viewerPreferences'].' 0 R';
332 break;
333 case 'openHere':
334 $res.="\n".'/OpenAction '.$o['info']['openHere'].' 0 R';
335 break;
338 $res.=" >>\nendobj";
339 return $res;
340 break;
345 * object which is a parent to the pages in the document
347 function o_pages($id,$action,$options=''){
348 if ($action!='new'){
349 $o =& $this->objects[$id];
351 switch ($action){
352 case 'new':
353 $this->objects[$id]=array('t'=>'pages','info'=>array());
354 $this->o_catalog($this->catalogId,'pages',$id);
355 break;
356 case 'page':
357 if (!is_array($options)){
358 // then it will just be the id of the new page
359 $o['info']['pages'][]=$options;
360 } else {
361 // then it should be an array having 'id','rid','pos', where rid=the page to which this one will be placed relative
362 // and pos is either 'before' or 'after', saying where this page will fit.
363 if (isset($options['id']) && isset($options['rid']) && isset($options['pos'])){
364 $i = array_search($options['rid'],$o['info']['pages']);
365 if (isset($o['info']['pages'][$i]) && $o['info']['pages'][$i]==$options['rid']){
366 // then there is a match
367 // make a space
368 switch ($options['pos']){
369 case 'before':
370 $k = $i;
371 break;
372 case 'after':
373 $k=$i+1;
374 break;
375 default:
376 $k=-1;
377 break;
379 if ($k>=0){
380 for ($j=count($o['info']['pages'])-1;$j>=$k;$j--){
381 $o['info']['pages'][$j+1]=$o['info']['pages'][$j];
383 $o['info']['pages'][$k]=$options['id'];
388 break;
389 case 'procset':
390 $o['info']['procset']=$options;
391 break;
392 case 'mediaBox':
393 $o['info']['mediaBox']=$options; // which should be an array of 4 numbers
394 break;
395 case 'font':
396 $o['info']['fonts'][]=array('objNum'=>$options['objNum'],'fontNum'=>$options['fontNum']);
397 break;
398 case 'xObject':
399 $o['info']['xObjects'][]=array('objNum'=>$options['objNum'],'label'=>$options['label']);
400 break;
401 case 'out':
402 if (count($o['info']['pages'])){
403 $res="\n".$id." 0 obj\n<< /Type /Pages\n/Kids [";
404 foreach($o['info']['pages'] as $k=>$v){
405 $res.=$v." 0 R\n";
407 $res.="]\n/Count ".count($this->objects[$id]['info']['pages']);
408 if ((isset($o['info']['fonts']) && count($o['info']['fonts'])) || isset($o['info']['procset'])){
409 $res.="\n/Resources <<";
410 if (isset($o['info']['procset'])){
411 $res.="\n/ProcSet ".$o['info']['procset']." 0 R";
413 if (isset($o['info']['fonts']) && count($o['info']['fonts'])){
414 $res.="\n/Font << ";
415 foreach($o['info']['fonts'] as $finfo){
416 $res.="\n/F".$finfo['fontNum']." ".$finfo['objNum']." 0 R";
418 $res.=" >>";
420 if (isset($o['info']['xObjects']) && count($o['info']['xObjects'])){
421 $res.="\n/XObject << ";
422 foreach($o['info']['xObjects'] as $finfo){
423 $res.="\n/".$finfo['label']." ".$finfo['objNum']." 0 R";
425 $res.=" >>";
427 $res.="\n>>";
428 if (isset($o['info']['mediaBox'])){
429 $tmp=$o['info']['mediaBox'];
430 $res.="\n/MediaBox [".sprintf('%.3f',$tmp[0]).' '.sprintf('%.3f',$tmp[1]).' '.sprintf('%.3f',$tmp[2]).' '.sprintf('%.3f',$tmp[3]).']';
433 $res.="\n >>\nendobj";
434 } else {
435 $res="\n".$id." 0 obj\n<< /Type /Pages\n/Count 0\n>>\nendobj";
437 return $res;
438 break;
443 * define the outlines in the doc, empty for now
445 function o_outlines($id,$action,$options=''){
446 if ($action!='new'){
447 $o =& $this->objects[$id];
449 switch ($action){
450 case 'new':
451 $this->objects[$id]=array('t'=>'outlines','info'=>array('outlines'=>array()));
452 $this->o_catalog($this->catalogId,'outlines',$id);
453 break;
454 case 'outline':
455 $o['info']['outlines'][]=$options;
456 break;
457 case 'out':
458 if (count($o['info']['outlines'])){
459 $res="\n".$id." 0 obj\n<< /Type /Outlines /Kids [";
460 foreach($o['info']['outlines'] as $k=>$v){
461 $res.=$v." 0 R ";
463 $res.="] /Count ".count($o['info']['outlines'])." >>\nendobj";
464 } else {
465 $res="\n".$id." 0 obj\n<< /Type /Outlines /Count 0 >>\nendobj";
467 return $res;
468 break;
473 * an object to hold the font description
475 function o_font($id,$action,$options=''){
476 if ($action!='new'){
477 $o =& $this->objects[$id];
479 switch ($action){
480 case 'new':
481 $this->objects[$id]=array('t'=>'font','info'=>array('name'=>$options['name'],'SubType'=>'Type1'));
482 $fontNum=$this->numFonts;
483 $this->objects[$id]['info']['fontNum']=$fontNum;
484 // deal with the encoding and the differences
485 if (isset($options['differences'])){
486 // then we'll need an encoding dictionary
487 $this->numObj++;
488 $this->o_fontEncoding($this->numObj,'new',$options);
489 $this->objects[$id]['info']['encodingDictionary']=$this->numObj;
490 } else if (isset($options['encoding'])){
491 // we can specify encoding here
492 switch($options['encoding']){
493 case 'WinAnsiEncoding':
494 case 'MacRomanEncoding':
495 case 'MacExpertEncoding':
496 $this->objects[$id]['info']['encoding']=$options['encoding'];
497 break;
498 case 'none':
499 break;
500 default:
501 $this->objects[$id]['info']['encoding']='WinAnsiEncoding';
502 break;
504 } else {
505 $this->objects[$id]['info']['encoding']='WinAnsiEncoding';
507 // also tell the pages node about the new font
508 $this->o_pages($this->currentNode,'font',array('fontNum'=>$fontNum,'objNum'=>$id));
509 break;
510 case 'add':
511 foreach ($options as $k=>$v){
512 switch ($k){
513 case 'BaseFont':
514 $o['info']['name'] = $v;
515 break;
516 case 'FirstChar':
517 case 'LastChar':
518 case 'Widths':
519 case 'FontDescriptor':
520 case 'SubType':
521 $this->addMessage('o_font '.$k." : ".$v);
522 $o['info'][$k] = $v;
523 break;
526 break;
527 case 'out':
528 $res="\n".$id." 0 obj\n<< /Type /Font\n/Subtype /".$o['info']['SubType']."\n";
529 $res.="/Name /F".$o['info']['fontNum']."\n";
530 $res.="/BaseFont /".$o['info']['name']."\n";
531 if (isset($o['info']['encodingDictionary'])){
532 // then place a reference to the dictionary
533 $res.="/Encoding ".$o['info']['encodingDictionary']." 0 R\n";
534 } else if (isset($o['info']['encoding'])){
535 // use the specified encoding
536 $res.="/Encoding /".$o['info']['encoding']."\n";
538 if (isset($o['info']['FirstChar'])){
539 $res.="/FirstChar ".$o['info']['FirstChar']."\n";
541 if (isset($o['info']['LastChar'])){
542 $res.="/LastChar ".$o['info']['LastChar']."\n";
544 if (isset($o['info']['Widths'])){
545 $res.="/Widths ".$o['info']['Widths']." 0 R\n";
547 if (isset($o['info']['FontDescriptor'])){
548 $res.="/FontDescriptor ".$o['info']['FontDescriptor']." 0 R\n";
550 $res.=">>\nendobj";
551 return $res;
552 break;
557 * a font descriptor, needed for including additional fonts
559 function o_fontDescriptor($id,$action,$options=''){
560 if ($action!='new'){
561 $o =& $this->objects[$id];
563 switch ($action){
564 case 'new':
565 $this->objects[$id]=array('t'=>'fontDescriptor','info'=>$options);
566 break;
567 case 'out':
568 $res="\n".$id." 0 obj\n<< /Type /FontDescriptor\n";
569 foreach ($o['info'] as $label => $value){
570 switch ($label){
571 case 'Ascent':
572 case 'CapHeight':
573 case 'Descent':
574 case 'Flags':
575 case 'ItalicAngle':
576 case 'StemV':
577 case 'AvgWidth':
578 case 'Leading':
579 case 'MaxWidth':
580 case 'MissingWidth':
581 case 'StemH':
582 case 'XHeight':
583 case 'CharSet':
584 if (strlen($value)){
585 $res.='/'.$label.' '.$value."\n";
587 break;
588 case 'FontFile':
589 case 'FontFile2':
590 case 'FontFile3':
591 $res.='/'.$label.' '.$value." 0 R\n";
592 break;
593 case 'FontBBox':
594 $res.='/'.$label.' ['.$value[0].' '.$value[1].' '.$value[2].' '.$value[3]."]\n";
595 break;
596 case 'FontName':
597 $res.='/'.$label.' /'.$value."\n";
598 break;
601 $res.=">>\nendobj";
602 return $res;
603 break;
608 * the font encoding
610 function o_fontEncoding($id,$action,$options=''){
611 if ($action!='new'){
612 $o =& $this->objects[$id];
614 switch ($action){
615 case 'new':
616 // the options array should contain 'differences' and maybe 'encoding'
617 $this->objects[$id]=array('t'=>'fontEncoding','info'=>$options);
618 break;
619 case 'out':
620 $res="\n".$id." 0 obj\n<< /Type /Encoding\n";
621 if (!isset($o['info']['encoding'])){
622 $o['info']['encoding']='WinAnsiEncoding';
624 if ($o['info']['encoding']!='none'){
625 $res.="/BaseEncoding /".$o['info']['encoding']."\n";
627 $res.="/Differences \n[";
628 $onum=-100;
629 foreach($o['info']['differences'] as $num=>$label){
630 if ($num!=$onum+1){
631 // we cannot make use of consecutive numbering
632 $res.= "\n".$num." /".$label;
633 } else {
634 $res.= " /".$label;
636 $onum=$num;
638 $res.="\n]\n>>\nendobj";
639 return $res;
640 break;
645 * the document procset, solves some problems with printing to old PS printers
647 function o_procset($id,$action,$options=''){
648 if ($action!='new'){
649 $o =& $this->objects[$id];
651 switch ($action){
652 case 'new':
653 $this->objects[$id]=array('t'=>'procset','info'=>array('PDF'=>1,'Text'=>1));
654 $this->o_pages($this->currentNode,'procset',$id);
655 $this->procsetObjectId=$id;
656 break;
657 case 'add':
658 // this is to add new items to the procset list, despite the fact that this is considered
659 // obselete, the items are required for printing to some postscript printers
660 switch ($options) {
661 case 'ImageB':
662 case 'ImageC':
663 case 'ImageI':
664 $o['info'][$options]=1;
665 break;
667 break;
668 case 'out':
669 $res="\n".$id." 0 obj\n[";
670 foreach ($o['info'] as $label=>$val){
671 $res.='/'.$label.' ';
673 $res.="]\nendobj";
674 return $res;
675 break;
680 * define the document information
682 function o_info($id,$action,$options=''){
683 if ($action!='new'){
684 $o =& $this->objects[$id];
686 switch ($action){
687 case 'new':
688 $this->infoObject=$id;
689 $date='D:'.date('Ymd');
690 $this->objects[$id]=array('t'=>'info','info'=>array('Creator'=>'R and OS php pdf writer, http://www.ros.co.nz','CreationDate'=>$date));
691 break;
692 case 'Title':
693 case 'Author':
694 case 'Subject':
695 case 'Keywords':
696 case 'Creator':
697 case 'Producer':
698 case 'CreationDate':
699 case 'ModDate':
700 case 'Trapped':
701 $o['info'][$action]=$options;
702 break;
703 case 'out':
704 if ($this->encrypted){
705 $this->encryptInit($id);
707 $res="\n".$id." 0 obj\n<<\n";
708 foreach ($o['info'] as $k=>$v){
709 $res.='/'.$k.' (';
710 if ($this->encrypted){
711 $res.=$this->filterText($this->ARC4($v));
712 } else {
713 $res.=$this->filterText($v);
715 $res.=")\n";
717 $res.=">>\nendobj";
718 return $res;
719 break;
724 * an action object, used to link to URLS initially
726 function o_action($id,$action,$options=''){
727 if ($action!='new'){
728 $o =& $this->objects[$id];
730 switch ($action){
731 case 'new':
732 if (is_array($options)){
733 $this->objects[$id]=array('t'=>'action','info'=>$options,'type'=>$options['type']);
734 } else {
735 // then assume a URI action
736 $this->objects[$id]=array('t'=>'action','info'=>$options,'type'=>'URI');
738 break;
739 case 'out':
740 if ($this->encrypted){
741 $this->encryptInit($id);
743 $res="\n".$id." 0 obj\n<< /Type /Action";
744 switch($o['type']){
745 case 'ilink':
746 // there will be an 'label' setting, this is the name of the destination
747 $res.="\n/S /GoTo\n/D ".$this->destinations[(string)$o['info']['label']]." 0 R";
748 break;
749 case 'URI':
750 $res.="\n/S /URI\n/URI (";
751 if ($this->encrypted){
752 $res.=$this->filterText($this->ARC4($o['info']));
753 } else {
754 $res.=$this->filterText($o['info']);
756 $res.=")";
757 break;
759 $res.="\n>>\nendobj";
760 return $res;
761 break;
766 * an annotation object, this will add an annotation to the current page.
767 * initially will support just link annotations
769 function o_annotation($id,$action,$options=''){
770 if ($action!='new'){
771 $o =& $this->objects[$id];
773 switch ($action){
774 case 'new':
775 // add the annotation to the current page
776 $pageId = $this->currentPage;
777 $this->o_page($pageId,'annot',$id);
778 // and add the action object which is going to be required
779 switch($options['type']){
780 case 'link':
781 $this->objects[$id]=array('t'=>'annotation','info'=>$options);
782 $this->numObj++;
783 $this->o_action($this->numObj,'new',$options['url']);
784 $this->objects[$id]['info']['actionId']=$this->numObj;
785 break;
786 case 'ilink':
787 // this is to a named internal link
788 $label = $options['label'];
789 $this->objects[$id]=array('t'=>'annotation','info'=>$options);
790 $this->numObj++;
791 $this->o_action($this->numObj,'new',array('type'=>'ilink','label'=>$label));
792 $this->objects[$id]['info']['actionId']=$this->numObj;
793 break;
795 break;
796 case 'out':
797 $res="\n".$id." 0 obj\n<< /Type /Annot";
798 switch($o['info']['type']){
799 case 'link':
800 case 'ilink':
801 $res.= "\n/Subtype /Link";
802 break;
804 $res.="\n/A ".$o['info']['actionId']." 0 R";
805 $res.="\n/Border [0 0 0]";
806 $res.="\n/H /I";
807 $res.="\n/Rect [ ";
808 foreach($o['info']['rect'] as $v){
809 $res.= sprintf("%.4f ",$v);
811 $res.="]";
812 $res.="\n>>\nendobj";
813 return $res;
814 break;
819 * a page object, it also creates a contents object to hold its contents
821 function o_page($id,$action,$options=''){
822 if ($action!='new'){
823 $o =& $this->objects[$id];
825 switch ($action){
826 case 'new':
827 $this->numPages++;
828 $this->objects[$id]=array('t'=>'page','info'=>array('parent'=>$this->currentNode,'pageNum'=>$this->numPages));
829 if (is_array($options)){
830 // then this must be a page insertion, array shoudl contain 'rid','pos'=[before|after]
831 $options['id']=$id;
832 $this->o_pages($this->currentNode,'page',$options);
833 } else {
834 $this->o_pages($this->currentNode,'page',$id);
836 $this->currentPage=$id;
837 //make a contents object to go with this page
838 $this->numObj++;
839 $this->o_contents($this->numObj,'new',$id);
840 $this->currentContents=$this->numObj;
841 $this->objects[$id]['info']['contents']=array();
842 $this->objects[$id]['info']['contents'][]=$this->numObj;
843 $match = ($this->numPages%2 ? 'odd' : 'even');
844 foreach($this->addLooseObjects as $oId=>$target){
845 if ($target=='all' || $match==$target){
846 $this->objects[$id]['info']['contents'][]=$oId;
849 break;
850 case 'content':
851 $o['info']['contents'][]=$options;
852 break;
853 case 'annot':
854 // add an annotation to this page
855 if (!isset($o['info']['annot'])){
856 $o['info']['annot']=array();
858 // $options should contain the id of the annotation dictionary
859 $o['info']['annot'][]=$options;
860 break;
861 case 'out':
862 $res="\n".$id." 0 obj\n<< /Type /Page";
863 $res.="\n/Parent ".$o['info']['parent']." 0 R";
864 if (isset($o['info']['annot'])){
865 $res.="\n/Annots [";
866 foreach($o['info']['annot'] as $aId){
867 $res.=" ".$aId." 0 R";
869 $res.=" ]";
871 $count = count($o['info']['contents']);
872 if ($count==1){
873 $res.="\n/Contents ".$o['info']['contents'][0]." 0 R";
874 } else if ($count>1){
875 $res.="\n/Contents [\n";
876 foreach ($o['info']['contents'] as $cId){
877 $res.=$cId." 0 R\n";
879 $res.="]";
881 $res.="\n>>\nendobj";
882 return $res;
883 break;
888 * the contents objects hold all of the content which appears on pages
890 function o_contents($id,$action,$options=''){
891 if ($action!='new'){
892 $o =& $this->objects[$id];
894 switch ($action){
895 case 'new':
896 $this->objects[$id]=array('t'=>'contents','c'=>'','info'=>array());
897 if (strlen($options) && intval($options)){
898 // then this contents is the primary for a page
899 $this->objects[$id]['onPage']=$options;
900 } else if ($options=='raw'){
901 // then this page contains some other type of system object
902 $this->objects[$id]['raw']=1;
904 break;
905 case 'add':
906 // add more options to the decleration
907 foreach ($options as $k=>$v){
908 $o['info'][$k]=$v;
910 case 'out':
911 $tmp=$o['c'];
912 $res= "\n".$id." 0 obj\n";
913 if (isset($this->objects[$id]['raw'])){
914 $res.=$tmp;
915 } else {
916 $res.= "<<";
917 if (function_exists('gzcompress') && $this->options['compression']){
918 // then implement ZLIB based compression on this content stream
919 $res.=" /Filter /FlateDecode";
920 $tmp = gzcompress($tmp);
922 if ($this->encrypted){
923 $this->encryptInit($id);
924 $tmp = $this->ARC4($tmp);
926 foreach($o['info'] as $k=>$v){
927 $res .= "\n/".$k.' '.$v;
929 $res.="\n/Length ".strlen($tmp)." >>\nstream\n".$tmp."\nendstream";
931 $res.="\nendobj\n";
932 return $res;
933 break;
938 * an image object, will be an XObject in the document, includes description and data
940 function o_image($id,$action,$options=''){
941 if ($action!='new'){
942 $o =& $this->objects[$id];
944 switch($action){
945 case 'new':
946 // make the new object
947 $this->objects[$id]=array('t'=>'image','data'=>$options['data'],'info'=>array());
948 $this->objects[$id]['info']['Type']='/XObject';
949 $this->objects[$id]['info']['Subtype']='/Image';
950 $this->objects[$id]['info']['Width']=$options['iw'];
951 $this->objects[$id]['info']['Height']=$options['ih'];
952 if (!isset($options['type']) || $options['type']=='jpg'){
953 if (!isset($options['channels'])){
954 $options['channels']=3;
956 switch($options['channels']){
957 case 1:
958 $this->objects[$id]['info']['ColorSpace']='/DeviceGray';
959 break;
960 default:
961 $this->objects[$id]['info']['ColorSpace']='/DeviceRGB';
962 break;
964 $this->objects[$id]['info']['Filter']='/DCTDecode';
965 $this->objects[$id]['info']['BitsPerComponent']=8;
966 } else if ($options['type']=='png'){
967 $this->objects[$id]['info']['Filter']='/FlateDecode';
968 $this->objects[$id]['info']['DecodeParms']='<< /Predictor 15 /Colors '.$options['ncolor'].' /Columns '.$options['iw'].' /BitsPerComponent '.$options['bitsPerComponent'].'>>';
969 if (strlen($options['pdata'])){
970 $tmp = ' [ /Indexed /DeviceRGB '.(strlen($options['pdata'])/3-1).' ';
971 $this->numObj++;
972 $this->o_contents($this->numObj,'new');
973 $this->objects[$this->numObj]['c']=$options['pdata'];
974 $tmp.=$this->numObj.' 0 R';
975 $tmp .=' ]';
976 $this->objects[$id]['info']['ColorSpace'] = $tmp;
977 if (isset($options['transparency'])){
978 switch($options['transparency']['type']){
979 case 'indexed':
980 $tmp=' [ '.$options['transparency']['data'].' '.$options['transparency']['data'].'] ';
981 $this->objects[$id]['info']['Mask'] = $tmp;
982 break;
985 } else {
986 $this->objects[$id]['info']['ColorSpace']='/'.$options['color'];
988 $this->objects[$id]['info']['BitsPerComponent']=$options['bitsPerComponent'];
990 // assign it a place in the named resource dictionary as an external object, according to
991 // the label passed in with it.
992 $this->o_pages($this->currentNode,'xObject',array('label'=>$options['label'],'objNum'=>$id));
993 // also make sure that we have the right procset object for it.
994 $this->o_procset($this->procsetObjectId,'add','ImageC');
995 break;
996 case 'out':
997 $tmp=$o['data'];
998 $res= "\n".$id." 0 obj\n<<";
999 foreach($o['info'] as $k=>$v){
1000 $res.="\n/".$k.' '.$v;
1002 if ($this->encrypted){
1003 $this->encryptInit($id);
1004 $tmp = $this->ARC4($tmp);
1006 $res.="\n/Length ".strlen($tmp)." >>\nstream\n".$tmp."\nendstream\nendobj\n";
1007 return $res;
1008 break;
1013 * encryption object.
1015 function o_encryption($id,$action,$options=''){
1016 if ($action!='new'){
1017 $o =& $this->objects[$id];
1019 switch($action){
1020 case 'new':
1021 // make the new object
1022 $this->objects[$id]=array('t'=>'encryption','info'=>$options);
1023 $this->arc4_objnum=$id;
1024 // figure out the additional paramaters required
1025 $pad = chr(0x28).chr(0xBF).chr(0x4E).chr(0x5E).chr(0x4E).chr(0x75).chr(0x8A).chr(0x41).chr(0x64).chr(0x00).chr(0x4E).chr(0x56).chr(0xFF).chr(0xFA).chr(0x01).chr(0x08).chr(0x2E).chr(0x2E).chr(0x00).chr(0xB6).chr(0xD0).chr(0x68).chr(0x3E).chr(0x80).chr(0x2F).chr(0x0C).chr(0xA9).chr(0xFE).chr(0x64).chr(0x53).chr(0x69).chr(0x7A);
1026 $len = strlen($options['owner']);
1027 if ($len>32){
1028 $owner = substr($options['owner'],0,32);
1029 } else if ($len<32){
1030 $owner = $options['owner'].substr($pad,0,32-$len);
1031 } else {
1032 $owner = $options['owner'];
1034 $len = strlen($options['user']);
1035 if ($len>32){
1036 $user = substr($options['user'],0,32);
1037 } else if ($len<32){
1038 $user = $options['user'].substr($pad,0,32-$len);
1039 } else {
1040 $user = $options['user'];
1042 $tmp = $this->md5_16($owner);
1043 $okey = substr($tmp,0,5);
1044 $this->ARC4_init($okey);
1045 $ovalue=$this->ARC4($user);
1046 $this->objects[$id]['info']['O']=$ovalue;
1047 // now make the u value, phew.
1048 $tmp = $this->md5_16($user.$ovalue.chr($options['p']).chr(255).chr(255).chr(255).$this->fileIdentifier);
1049 $ukey = substr($tmp,0,5);
1051 $this->ARC4_init($ukey);
1052 $this->encryptionKey = $ukey;
1053 $this->encrypted=1;
1054 $uvalue=$this->ARC4($pad);
1056 $this->objects[$id]['info']['U']=$uvalue;
1057 $this->encryptionKey=$ukey;
1059 // initialize the arc4 array
1060 break;
1061 case 'out':
1062 $res= "\n".$id." 0 obj\n<<";
1063 $res.="\n/Filter /Standard";
1064 $res.="\n/V 1";
1065 $res.="\n/R 2";
1066 $res.="\n/O (".$this->filterText($o['info']['O']).')';
1067 $res.="\n/U (".$this->filterText($o['info']['U']).')';
1068 // and the p-value needs to be converted to account for the twos-complement approach
1069 $o['info']['p'] = (($o['info']['p']^255)+1)*-1;
1070 $res.="\n/P ".($o['info']['p']);
1071 $res.="\n>>\nendobj\n";
1073 return $res;
1074 break;
1079 * ARC4 functions
1080 * A series of function to implement ARC4 encoding in PHP
1084 * calculate the 16 byte version of the 128 bit md5 digest of the string
1086 function md5_16($string){
1087 $tmp = md5($string);
1088 $out='';
1089 for ($i=0;$i<=30;$i=$i+2){
1090 $out.=chr(hexdec(substr($tmp,$i,2)));
1092 return $out;
1096 * initialize the encryption for processing a particular object
1098 function encryptInit($id){
1099 $tmp = $this->encryptionKey;
1100 $hex = dechex($id);
1101 if (strlen($hex)<6){
1102 $hex = substr('000000',0,6-strlen($hex)).$hex;
1104 $tmp.= chr(hexdec(substr($hex,4,2))).chr(hexdec(substr($hex,2,2))).chr(hexdec(substr($hex,0,2))).chr(0).chr(0);
1105 $key = $this->md5_16($tmp);
1106 $this->ARC4_init(substr($key,0,10));
1110 * initialize the ARC4 encryption
1112 function ARC4_init($key=''){
1113 $this->arc4 = '';
1114 // setup the control array
1115 if (strlen($key)==0){
1116 return;
1118 $k = '';
1119 while(strlen($k)<256){
1120 $k.=$key;
1122 $k=substr($k,0,256);
1123 for ($i=0;$i<256;$i++){
1124 $this->arc4 .= chr($i);
1126 $j=0;
1127 for ($i=0;$i<256;$i++){
1128 $t = $this->arc4[$i];
1129 $j = ($j + ord($t) + ord($k[$i]))%256;
1130 $this->arc4[$i]=$this->arc4[$j];
1131 $this->arc4[$j]=$t;
1136 * ARC4 encrypt a text string
1138 function ARC4($text){
1139 $len=strlen($text);
1140 $a=0;
1141 $b=0;
1142 $c = $this->arc4;
1143 $out='';
1144 for ($i=0;$i<$len;$i++){
1145 $a = ($a+1)%256;
1146 $t= $c[$a];
1147 $b = ($b+ord($t))%256;
1148 $c[$a]=$c[$b];
1149 $c[$b]=$t;
1150 $k = ord($c[(ord($c[$a])+ord($c[$b]))%256]);
1151 $out.=chr(ord($text[$i]) ^ $k);
1154 return $out;
1158 * functions which can be called to adjust or add to the document
1162 * add a link in the document to an external URL
1164 function addLink($url,$x0,$y0,$x1,$y1){
1165 $this->numObj++;
1166 $info = array('type'=>'link','url'=>$url,'rect'=>array($x0,$y0,$x1,$y1));
1167 $this->o_annotation($this->numObj,'new',$info);
1171 * add a link in the document to an internal destination (ie. within the document)
1173 function addInternalLink($label,$x0,$y0,$x1,$y1){
1174 $this->numObj++;
1175 $info = array('type'=>'ilink','label'=>$label,'rect'=>array($x0,$y0,$x1,$y1));
1176 $this->o_annotation($this->numObj,'new',$info);
1180 * set the encryption of the document
1181 * can be used to turn it on and/or set the passwords which it will have.
1182 * also the functions that the user will have are set here, such as print, modify, add
1184 function setEncryption($userPass='',$ownerPass='',$pc=array()){
1185 $p=bindec(11000000);
1187 $options = array(
1188 'print'=>4
1189 ,'modify'=>8
1190 ,'copy'=>16
1191 ,'add'=>32
1193 foreach($pc as $k=>$v){
1194 if ($v && isset($options[$k])){
1195 $p+=$options[$k];
1196 } else if (isset($options[$v])){
1197 $p+=$options[$v];
1200 // implement encryption on the document
1201 if ($this->arc4_objnum == 0){
1202 // then the block does not exist already, add it.
1203 $this->numObj++;
1204 if (strlen($ownerPass)==0){
1205 $ownerPass=$userPass;
1207 $this->o_encryption($this->numObj,'new',array('user'=>$userPass,'owner'=>$ownerPass,'p'=>$p));
1212 * should be used for internal checks, not implemented as yet
1214 function checkAllHere(){
1218 * return the pdf stream as a string returned from the function
1220 function output($debug=0){
1222 if ($debug){
1223 // turn compression off
1224 $this->options['compression']=0;
1227 if ($this->arc4_objnum){
1228 $this->ARC4_init($this->encryptionKey);
1231 $this->checkAllHere();
1233 $xref=array();
1234 $content="%PDF-1.3\n%��\n";
1235 // $content="%PDF-1.3\n";
1236 $pos=strlen($content);
1237 foreach($this->objects as $k=>$v){
1238 $tmp='o_'.$v['t'];
1239 $cont=$this->$tmp($k,'out');
1240 $content.=$cont;
1241 $xref[]=$pos;
1242 $pos+=strlen($cont);
1244 $content.="\nxref\n0 ".(count($xref)+1)."\n0000000000 65535 f \n";
1245 foreach($xref as $p){
1246 $content.=substr('0000000000',0,10-strlen($p)).$p." 00000 n \n";
1248 $content.="\ntrailer\n << /Size ".(count($xref)+1)."\n /Root 1 0 R\n /Info ".$this->infoObject." 0 R\n";
1249 // if encryption has been applied to this document then add the marker for this dictionary
1250 if ($this->arc4_objnum > 0){
1251 $content .= "/Encrypt ".$this->arc4_objnum." 0 R\n";
1253 if (strlen($this->fileIdentifier)){
1254 $content .= "/ID[<".$this->fileIdentifier."><".$this->fileIdentifier.">]\n";
1256 $content .= " >>\nstartxref\n".$pos."\n%%EOF\n";
1257 return $content;
1261 * intialize a new document
1262 * if this is called on an existing document results may be unpredictable, but the existing document would be lost at minimum
1263 * this function is called automatically by the constructor function
1265 * @access private
1267 function newDocument($pageSize=array(0,0,612,792)){
1268 $this->numObj=0;
1269 $this->objects = array();
1271 $this->numObj++;
1272 $this->o_catalog($this->numObj,'new');
1274 $this->numObj++;
1275 $this->o_outlines($this->numObj,'new');
1277 $this->numObj++;
1278 $this->o_pages($this->numObj,'new');
1280 $this->o_pages($this->numObj,'mediaBox',$pageSize);
1281 $this->currentNode = 3;
1283 $this->numObj++;
1284 $this->o_procset($this->numObj,'new');
1286 $this->numObj++;
1287 $this->o_info($this->numObj,'new');
1289 $this->numObj++;
1290 $this->o_page($this->numObj,'new');
1292 // need to store the first page id as there is no way to get it to the user during
1293 // startup
1294 $this->firstPageId = $this->currentContents;
1298 * open the font file and return a php structure containing it.
1299 * first check if this one has been done before and saved in a form more suited to php
1300 * note that if a php serialized version does not exist it will try and make one, but will
1301 * require write access to the directory to do it... it is MUCH faster to have these serialized
1302 * files.
1304 * @access private
1306 function openFont($font){
1307 // assume that $font contains both the path and perhaps the extension to the file, split them
1308 $pos=strrpos($font,'/');
1309 if ($pos===false){
1310 $dir = './';
1311 $name = $font;
1312 } else {
1313 $dir=substr($font,0,$pos+1);
1314 $name=substr($font,$pos+1);
1317 if (substr($name,-4)=='.afm'){
1318 $name=substr($name,0,strlen($name)-4);
1320 $this->addMessage('openFont: '.$font.' - '.$name);
1321 if (file_exists($dir.'php_'.$name.'.afm')){
1322 $this->addMessage('openFont: php file exists '.$dir.'php_'.$name.'.afm');
1323 $tmp = file($dir.'php_'.$name.'.afm');
1324 $this->fonts[$font]=unserialize($tmp[0]);
1325 if (!isset($this->fonts[$font]['_version_']) || $this->fonts[$font]['_version_']<1){
1326 // if the font file is old, then clear it out and prepare for re-creation
1327 $this->addMessage('openFont: clear out, make way for new version.');
1328 unset($this->fonts[$font]);
1331 if (!isset($this->fonts[$font]) && file_exists($dir.$name.'.afm')){
1332 // then rebuild the php_<font>.afm file from the <font>.afm file
1333 $this->addMessage('openFont: build php file from '.$dir.$name.'.afm');
1334 $data = array();
1335 $file = file($dir.$name.'.afm');
1336 foreach ($file as $rowA){
1337 $row=trim($rowA);
1338 $pos=strpos($row,' ');
1339 if ($pos){
1340 // then there must be some keyword
1341 $key = substr($row,0,$pos);
1342 switch ($key){
1343 case 'FontName':
1344 case 'FullName':
1345 case 'FamilyName':
1346 case 'Weight':
1347 case 'ItalicAngle':
1348 case 'IsFixedPitch':
1349 case 'CharacterSet':
1350 case 'UnderlinePosition':
1351 case 'UnderlineThickness':
1352 case 'Version':
1353 case 'EncodingScheme':
1354 case 'CapHeight':
1355 case 'XHeight':
1356 case 'Ascender':
1357 case 'Descender':
1358 case 'StdHW':
1359 case 'StdVW':
1360 case 'StartCharMetrics':
1361 $data[$key]=trim(substr($row,$pos));
1362 break;
1363 case 'FontBBox':
1364 $data[$key]=explode(' ',trim(substr($row,$pos)));
1365 break;
1366 case 'C':
1367 //C 39 ; WX 222 ; N quoteright ; B 53 463 157 718 ;
1368 $bits=explode(';',trim($row));
1369 $dtmp=array();
1370 foreach($bits as $bit){
1371 $bits2 = explode(' ',trim($bit));
1372 if (strlen($bits2[0])){
1373 if (count($bits2)>2){
1374 $dtmp[$bits2[0]]=array();
1375 for ($i=1;$i<count($bits2);$i++){
1376 $dtmp[$bits2[0]][]=$bits2[$i];
1378 } else if (count($bits2)==2){
1379 $dtmp[$bits2[0]]=$bits2[1];
1383 if ($dtmp['C']>=0){
1384 $data['C'][$dtmp['C']]=$dtmp;
1385 $data['C'][$dtmp['N']]=$dtmp;
1386 } else {
1387 $data['C'][$dtmp['N']]=$dtmp;
1389 break;
1390 case 'KPX':
1391 //KPX Adieresis yacute -40
1392 $bits=explode(' ',trim($row));
1393 $data['KPX'][$bits[1]][$bits[2]]=$bits[3];
1394 break;
1398 $data['_version_']=1;
1399 $this->fonts[$font]=$data;
1400 touch($dir.'php_'.$name.'.afm'); // php bug
1401 $fp = fopen($dir.'php_'.$name.'.afm','w');
1402 fwrite($fp,serialize($data));
1403 fclose($fp);
1404 } else if (!isset($this->fonts[$font])){
1405 $this->addMessage('openFont: no font file found');
1406 // echo 'Font not Found '.$font;
1411 * if the font is not loaded then load it and make the required object
1412 * else just make it the current font
1413 * the encoding array can contain 'encoding'=> 'none','WinAnsiEncoding','MacRomanEncoding' or 'MacExpertEncoding'
1414 * note that encoding='none' will need to be used for symbolic fonts
1415 * and 'differences' => an array of mappings between numbers 0->255 and character names.
1418 function selectFont($fontName,$encoding='',$set=1){
1419 if (!isset($this->fonts[$fontName])){
1420 // load the file
1421 $this->openFont($fontName);
1422 if (isset($this->fonts[$fontName])){
1423 $this->numObj++;
1424 $this->numFonts++;
1425 $pos=strrpos($fontName,'/');
1426 // $dir=substr($fontName,0,$pos+1);
1427 $name=substr($fontName,$pos+1);
1428 if (substr($name,-4)=='.afm'){
1429 $name=substr($name,0,strlen($name)-4);
1431 $options=array('name'=>$name);
1432 if (is_array($encoding)){
1433 // then encoding and differences might be set
1434 if (isset($encoding['encoding'])){
1435 $options['encoding']=$encoding['encoding'];
1437 if (isset($encoding['differences'])){
1438 $options['differences']=$encoding['differences'];
1440 } else if (strlen($encoding)){
1441 // then perhaps only the encoding has been set
1442 $options['encoding']=$encoding;
1444 $fontObj = $this->numObj;
1445 $this->o_font($this->numObj,'new',$options);
1446 $this->fonts[$fontName]['fontNum']=$this->numFonts;
1447 // if this is a '.afm' font, and there is a '.pfa' file to go with it ( as there
1448 // should be for all non-basic fonts), then load it into an object and put the
1449 // references into the font object
1450 $basefile = substr($fontName,0,strlen($fontName)-4);
1451 if (file_exists($basefile.'.pfb')){
1452 $fbtype = 'pfb';
1453 } else if (file_exists($basefile.'.ttf')){
1454 $fbtype = 'ttf';
1455 } else {
1456 $fbtype='';
1458 $fbfile = $basefile.'.'.$fbtype;
1460 // $pfbfile = substr($fontName,0,strlen($fontName)-4).'.pfb';
1461 // $ttffile = substr($fontName,0,strlen($fontName)-4).'.ttf';
1462 $this->addMessage('selectFont: checking for - '.$fbfile);
1463 if (substr($fontName,-4)=='.afm' && strlen($fbtype) ){
1464 $adobeFontName = $this->fonts[$fontName]['FontName'];
1465 // $fontObj = $this->numObj;
1466 $this->addMessage('selectFont: adding font file - '.$fbfile.' - '.$adobeFontName);
1467 // find the array of fond widths, and put that into an object.
1468 $firstChar = -1;
1469 $lastChar = 0;
1470 $widths = array();
1471 foreach ($this->fonts[$fontName]['C'] as $num=>$d){
1472 if (intval($num)>0 || $num=='0'){
1473 if ($lastChar>0 && $num>$lastChar+1){
1474 for($i=$lastChar+1;$i<$num;$i++){
1475 $widths[] = 0;
1478 $widths[] = $d['WX'];
1479 if ($firstChar==-1){
1480 $firstChar = $num;
1482 $lastChar = $num;
1485 // also need to adjust the widths for the differences array
1486 if (isset($options['differences'])){
1487 foreach($options['differences'] as $charNum=>$charName){
1488 if ($charNum>$lastChar){
1489 for($i=$lastChar+1;$i<=$charNum;$i++){
1490 $widths[]=0;
1492 $lastChar=$charNum;
1494 if (isset($this->fonts[$fontName]['C'][$charName])){
1495 $widths[$charNum-$firstChar]=$this->fonts[$fontName]['C'][$charName]['WX'];
1499 $this->addMessage('selectFont: FirstChar='.$firstChar);
1500 $this->addMessage('selectFont: LastChar='.$lastChar);
1501 $this->numObj++;
1502 $this->o_contents($this->numObj,'new','raw');
1503 $this->objects[$this->numObj]['c'].='[';
1504 foreach($widths as $width){
1505 $this->objects[$this->numObj]['c'].=' '.$width;
1507 $this->objects[$this->numObj]['c'].=' ]';
1508 $widthid = $this->numObj;
1510 // load the pfb file, and put that into an object too.
1511 // note that pdf supports only binary format type 1 font files, though there is a
1512 // simple utility to convert them from pfa to pfb.
1513 $fp = fopen($fbfile,'rb');
1514 $tmp = get_magic_quotes_runtime();
1515 set_magic_quotes_runtime(0);
1516 $data = fread($fp,filesize($fbfile));
1517 set_magic_quotes_runtime($tmp);
1518 fclose($fp);
1520 // create the font descriptor
1521 $this->numObj++;
1522 $fontDescriptorId = $this->numObj;
1523 $this->numObj++;
1524 $pfbid = $this->numObj;
1525 // determine flags (more than a little flakey, hopefully will not matter much)
1526 $flags=0;
1527 if ($this->fonts[$fontName]['ItalicAngle']!=0){ $flags+=pow(2,6); }
1528 if ($this->fonts[$fontName]['IsFixedPitch']=='true'){ $flags+=1; }
1529 $flags+=pow(2,5); // assume non-sybolic
1531 $list = array('Ascent'=>'Ascender','CapHeight'=>'CapHeight','Descent'=>'Descender','FontBBox'=>'FontBBox','ItalicAngle'=>'ItalicAngle');
1532 $fdopt = array(
1533 'Flags'=>$flags
1534 ,'FontName'=>$adobeFontName
1535 ,'StemV'=>100 // don't know what the value for this should be!
1537 foreach($list as $k=>$v){
1538 if (isset($this->fonts[$fontName][$v])){
1539 $fdopt[$k]=$this->fonts[$fontName][$v];
1543 if ($fbtype=='pfb'){
1544 $fdopt['FontFile']=$pfbid;
1545 } else if ($fbtype=='ttf'){
1546 $fdopt['FontFile2']=$pfbid;
1548 $this->o_fontDescriptor($fontDescriptorId,'new',$fdopt);
1550 // embed the font program
1551 $this->o_contents($this->numObj,'new');
1552 $this->objects[$pfbid]['c'].=$data;
1553 // determine the cruicial lengths within this file
1554 if ($fbtype=='pfb'){
1555 $l1 = strpos($data,'eexec')+6;
1556 $l2 = strpos($data,'00000000')-$l1;
1557 $l3 = strlen($data)-$l2-$l1;
1558 $this->o_contents($this->numObj,'add',array('Length1'=>$l1,'Length2'=>$l2,'Length3'=>$l3));
1559 } else if ($fbtype=='ttf'){
1560 $l1 = strlen($data);
1561 $this->o_contents($this->numObj,'add',array('Length1'=>$l1));
1565 // tell the font object about all this new stuff
1566 $tmp = array('BaseFont'=>$adobeFontName,'Widths'=>$widthid
1567 ,'FirstChar'=>$firstChar,'LastChar'=>$lastChar
1568 ,'FontDescriptor'=>$fontDescriptorId);
1569 if ($fbtype=='ttf'){
1570 $tmp['SubType']='TrueType';
1572 $this->addMessage('adding extra info to font.('.$fontObj.')');
1573 foreach($tmp as $fk=>$fv){
1574 $this->addMessage($fk." : ".$fv);
1576 $this->o_font($fontObj,'add',$tmp);
1578 } else {
1579 $this->addMessage('selectFont: pfb or ttf file not found, ok if this is one of the 14 standard fonts');
1583 // also set the differences here, note that this means that these will take effect only the
1584 //first time that a font is selected, else they are ignored
1585 if (isset($options['differences'])){
1586 $this->fonts[$fontName]['differences']=$options['differences'];
1590 if ($set && isset($this->fonts[$fontName])){
1591 // so if for some reason the font was not set in the last one then it will not be selected
1592 $this->currentBaseFont=$fontName;
1593 // the next line means that if a new font is selected, then the current text state will be
1594 // applied to it as well.
1595 $this->setCurrentFont();
1597 return $this->currentFontNum;
1601 * sets up the current font, based on the font families, and the current text state
1602 * note that this system is quite flexible, a <b><i> font can be completely different to a
1603 * <i><b> font, and even <b><b> will have to be defined within the family to have meaning
1604 * This function is to be called whenever the currentTextState is changed, it will update
1605 * the currentFont setting to whatever the appropriatte family one is.
1606 * If the user calls selectFont themselves then that will reset the currentBaseFont, and the currentFont
1607 * This function will change the currentFont to whatever it should be, but will not change the
1608 * currentBaseFont.
1610 * @access private
1612 function setCurrentFont(){
1613 if (strlen($this->currentBaseFont)==0){
1614 // then assume an initial font
1615 $this->selectFont('./fonts/Helvetica.afm');
1617 $cf = substr($this->currentBaseFont,strrpos($this->currentBaseFont,'/')+1);
1618 if (strlen($this->currentTextState)
1619 && isset($this->fontFamilies[$cf])
1620 && isset($this->fontFamilies[$cf][$this->currentTextState])){
1621 // then we are in some state or another
1622 // and this font has a family, and the current setting exists within it
1623 // select the font, then return it
1624 $nf = substr($this->currentBaseFont,0,strrpos($this->currentBaseFont,'/')+1).$this->fontFamilies[$cf][$this->currentTextState];
1625 $this->selectFont($nf,'',0);
1626 $this->currentFont = $nf;
1627 $this->currentFontNum = $this->fonts[$nf]['fontNum'];
1628 } else {
1629 // the this font must not have the right family member for the current state
1630 // simply assume the base font
1631 $this->currentFont = $this->currentBaseFont;
1632 $this->currentFontNum = $this->fonts[$this->currentFont]['fontNum'];
1637 * function for the user to find out what the ID is of the first page that was created during
1638 * startup - useful if they wish to add something to it later.
1640 function getFirstPageId(){
1641 return $this->firstPageId;
1645 * add content to the currently active object
1647 * @access private
1649 function addContent($content){
1650 $this->objects[$this->currentContents]['c'].=$content;
1654 * sets the colour for fill operations
1656 function setColor($r,$g,$b,$force=0){
1657 if ($r>=0 && ($force || $r!=$this->currentColour['r'] || $g!=$this->currentColour['g'] || $b!=$this->currentColour['b'])){
1658 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$r).' '.sprintf('%.3f',$g).' '.sprintf('%.3f',$b).' rg';
1659 $this->currentColour=array('r'=>$r,'g'=>$g,'b'=>$b);
1664 * sets the colour for stroke operations
1666 function setStrokeColor($r,$g,$b,$force=0){
1667 if ($r>=0 && ($force || $r!=$this->currentStrokeColour['r'] || $g!=$this->currentStrokeColour['g'] || $b!=$this->currentStrokeColour['b'])){
1668 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$r).' '.sprintf('%.3f',$g).' '.sprintf('%.3f',$b).' RG';
1669 $this->currentStrokeColour=array('r'=>$r,'g'=>$g,'b'=>$b);
1674 * draw a line from one set of coordinates to another
1676 function line($x1,$y1,$x2,$y2){
1677 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$x1).' '.sprintf('%.3f',$y1).' m '.sprintf('%.3f',$x2).' '.sprintf('%.3f',$y2).' l S';
1681 * draw a bezier curve based on 4 control points
1683 function curve($x0,$y0,$x1,$y1,$x2,$y2,$x3,$y3){
1684 // in the current line style, draw a bezier curve from (x0,y0) to (x3,y3) using the other two points
1685 // as the control points for the curve.
1686 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$x0).' '.sprintf('%.3f',$y0).' m '.sprintf('%.3f',$x1).' '.sprintf('%.3f',$y1);
1687 $this->objects[$this->currentContents]['c'].= ' '.sprintf('%.3f',$x2).' '.sprintf('%.3f',$y2).' '.sprintf('%.3f',$x3).' '.sprintf('%.3f',$y3).' c S';
1691 * draw a part of an ellipse
1693 function partEllipse($x0,$y0,$astart,$afinish,$r1,$r2=0,$angle=0,$nSeg=8){
1694 $this->ellipse($x0,$y0,$r1,$r2,$angle,$nSeg,$astart,$afinish,0);
1698 * draw a filled ellipse
1700 function filledEllipse($x0,$y0,$r1,$r2=0,$angle=0,$nSeg=8,$astart=0,$afinish=360){
1701 return $this->ellipse($x0,$y0,$r1,$r2=0,$angle,$nSeg,$astart,$afinish,1,1);
1705 * draw an ellipse
1706 * note that the part and filled ellipse are just special cases of this function
1708 * draws an ellipse in the current line style
1709 * centered at $x0,$y0, radii $r1,$r2
1710 * if $r2 is not set, then a circle is drawn
1711 * nSeg is not allowed to be less than 2, as this will simply draw a line (and will even draw a
1712 * pretty crappy shape at 2, as we are approximating with bezier curves.
1714 function ellipse($x0,$y0,$r1,$r2=0,$angle=0,$nSeg=8,$astart=0,$afinish=360,$close=1,$fill=0){
1715 if ($r1==0){
1716 return;
1718 if ($r2==0){
1719 $r2=$r1;
1721 if ($nSeg<2){
1722 $nSeg=2;
1725 $astart = deg2rad((float)$astart);
1726 $afinish = deg2rad((float)$afinish);
1727 $totalAngle =$afinish-$astart;
1729 $dt = $totalAngle/$nSeg;
1730 $dtm = $dt/3;
1732 if ($angle != 0){
1733 $a = -1*deg2rad((float)$angle);
1734 $tmp = "\n q ";
1735 $tmp .= sprintf('%.3f',cos($a)).' '.sprintf('%.3f',(-1.0*sin($a))).' '.sprintf('%.3f',sin($a)).' '.sprintf('%.3f',cos($a)).' ';
1736 $tmp .= sprintf('%.3f',$x0).' '.sprintf('%.3f',$y0).' cm';
1737 $this->objects[$this->currentContents]['c'].= $tmp;
1738 $x0=0;
1739 $y0=0;
1742 $t1 = $astart;
1743 $a0 = $x0+$r1*cos($t1);
1744 $b0 = $y0+$r2*sin($t1);
1745 $c0 = -$r1*sin($t1);
1746 $d0 = $r2*cos($t1);
1748 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$a0).' '.sprintf('%.3f',$b0).' m ';
1749 for ($i=1;$i<=$nSeg;$i++){
1750 // draw this bit of the total curve
1751 $t1 = $i*$dt+$astart;
1752 $a1 = $x0+$r1*cos($t1);
1753 $b1 = $y0+$r2*sin($t1);
1754 $c1 = -$r1*sin($t1);
1755 $d1 = $r2*cos($t1);
1756 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',($a0+$c0*$dtm)).' '.sprintf('%.3f',($b0+$d0*$dtm));
1757 $this->objects[$this->currentContents]['c'].= ' '.sprintf('%.3f',($a1-$c1*$dtm)).' '.sprintf('%.3f',($b1-$d1*$dtm)).' '.sprintf('%.3f',$a1).' '.sprintf('%.3f',$b1).' c';
1758 $a0=$a1;
1759 $b0=$b1;
1760 $c0=$c1;
1761 $d0=$d1;
1763 if ($fill){
1764 $this->objects[$this->currentContents]['c'].=' f';
1765 } else {
1766 if ($close){
1767 $this->objects[$this->currentContents]['c'].=' s'; // small 's' signifies closing the path as well
1768 } else {
1769 $this->objects[$this->currentContents]['c'].=' S';
1772 if ($angle !=0){
1773 $this->objects[$this->currentContents]['c'].=' Q';
1778 * this sets the line drawing style.
1779 * width, is the thickness of the line in user units
1780 * cap is the type of cap to put on the line, values can be 'butt','round','square'
1781 * where the diffference between 'square' and 'butt' is that 'square' projects a flat end past the
1782 * end of the line.
1783 * join can be 'miter', 'round', 'bevel'
1784 * dash is an array which sets the dash pattern, is a series of length values, which are the lengths of the
1785 * on and off dashes.
1786 * (2) represents 2 on, 2 off, 2 on , 2 off ...
1787 * (2,1) is 2 on, 1 off, 2 on, 1 off.. etc
1788 * phase is a modifier on the dash pattern which is used to shift the point at which the pattern starts.
1790 function setLineStyle($width=1,$cap='',$join='',$dash='',$phase=0){
1792 // this is quite inefficient in that it sets all the parameters whenever 1 is changed, but will fix another day
1793 $string = '';
1794 if ($width>0){
1795 $string.= $width.' w';
1797 $ca = array('butt'=>0,'round'=>1,'square'=>2);
1798 if (isset($ca[$cap])){
1799 $string.= ' '.$ca[$cap].' J';
1801 $ja = array('miter'=>0,'round'=>1,'bevel'=>2);
1802 if (isset($ja[$join])){
1803 $string.= ' '.$ja[$join].' j';
1805 if (is_array($dash)){
1806 $string.= ' [';
1807 foreach ($dash as $len){
1808 $string.=' '.$len;
1810 $string.= ' ] '.$phase.' d';
1812 $this->currentLineStyle = $string;
1813 $this->objects[$this->currentContents]['c'].="\n".$string;
1817 * draw a polygon, the syntax for this is similar to the GD polygon command
1819 function polygon($p,$np,$f=0){
1820 $this->objects[$this->currentContents]['c'].="\n";
1821 $this->objects[$this->currentContents]['c'].=sprintf('%.3f',$p[0]).' '.sprintf('%.3f',$p[1]).' m ';
1822 for ($i=2;$i<$np*2;$i=$i+2){
1823 $this->objects[$this->currentContents]['c'].= sprintf('%.3f',$p[$i]).' '.sprintf('%.3f',$p[$i+1]).' l ';
1825 if ($f==1){
1826 $this->objects[$this->currentContents]['c'].=' f';
1827 } else {
1828 $this->objects[$this->currentContents]['c'].=' S';
1833 * a filled rectangle, note that it is the width and height of the rectangle which are the secondary paramaters, not
1834 * the coordinates of the upper-right corner
1836 function filledRectangle($x1,$y1,$width,$height){
1837 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$x1).' '.sprintf('%.3f',$y1).' '.sprintf('%.3f',$width).' '.sprintf('%.3f',$height).' re f';
1841 * draw a rectangle, note that it is the width and height of the rectangle which are the secondary paramaters, not
1842 * the coordinates of the upper-right corner
1844 function rectangle($x1,$y1,$width,$height){
1845 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$x1).' '.sprintf('%.3f',$y1).' '.sprintf('%.3f',$width).' '.sprintf('%.3f',$height).' re S';
1849 * add a new page to the document
1850 * this also makes the new page the current active object
1852 function newPage($insert=0,$id=0,$pos='after'){
1854 // if there is a state saved, then go up the stack closing them
1855 // then on the new page, re-open them with the right setings
1857 if ($this->nStateStack){
1858 for ($i=$this->nStateStack;$i>=1;$i--){
1859 $this->restoreState($i);
1863 $this->numObj++;
1864 if ($insert){
1865 // the id from the ezPdf class is the od of the contents of the page, not the page object itself
1866 // query that object to find the parent
1867 $rid = $this->objects[$id]['onPage'];
1868 $opt= array('rid'=>$rid,'pos'=>$pos);
1869 $this->o_page($this->numObj,'new',$opt);
1870 } else {
1871 $this->o_page($this->numObj,'new');
1873 // if there is a stack saved, then put that onto the page
1874 if ($this->nStateStack){
1875 for ($i=1;$i<=$this->nStateStack;$i++){
1876 $this->saveState($i);
1879 // and if there has been a stroke or fill colour set, then transfer them
1880 if ($this->currentColour['r']>=0){
1881 $this->setColor($this->currentColour['r'],$this->currentColour['g'],$this->currentColour['b'],1);
1883 if ($this->currentStrokeColour['r']>=0){
1884 $this->setStrokeColor($this->currentStrokeColour['r'],$this->currentStrokeColour['g'],$this->currentStrokeColour['b'],1);
1887 // if there is a line style set, then put this in too
1888 if (strlen($this->currentLineStyle)){
1889 $this->objects[$this->currentContents]['c'].="\n".$this->currentLineStyle;
1892 // the call to the o_page object set currentContents to the present page, so this can be returned as the page id
1893 return $this->currentContents;
1897 * output the pdf code, streaming it to the browser
1898 * the relevant headers are set so that hopefully the browser will recognise it
1900 function stream($options=''){
1901 // setting the options allows the adjustment of the headers
1902 // values at the moment are:
1903 // 'Content-Disposition'=>'filename' - sets the filename, though not too sure how well this will
1904 // work as in my trial the browser seems to use the filename of the php file with .pdf on the end
1905 // 'Accept-Ranges'=>1 or 0 - if this is not set to 1, then this header is not included, off by default
1906 // this header seems to have caused some problems despite tha fact that it is supposed to solve
1907 // them, so I am leaving it off by default.
1908 // 'compress'=> 1 or 0 - apply content stream compression, this is on (1) by default
1909 if (!is_array($options)){
1910 $options=array();
1912 if ( isset($options['compress']) && $options['compress']==0){
1913 $tmp = $this->output(1);
1914 } else {
1915 $tmp = $this->output();
1918 // Rod's mods below are based on this tip:
1919 // http://sourceforge.net/forum/forum.php?thread_id=1420028&forum_id=147987
1920 header('Cache-Control:'); // added by Rod
1921 header('Pragma:'); // added by Rod
1922 header("Content-type: application/pdf");
1923 header("Content-Length: ".strlen(ltrim($tmp)));
1924 $fileName = (isset($options['Content-Disposition'])?$options['Content-Disposition']:'file.pdf');
1925 // header("Content-Disposition: inline; filename=".$fileName);
1926 header("Content-Disposition: attachment; filename=".$fileName); // modded by Rod
1927 // End of Rod's mods.
1929 if (isset($options['Accept-Ranges']) && $options['Accept-Ranges']==1){
1930 header("Accept-Ranges: ".strlen(ltrim($tmp)));
1932 echo ltrim($tmp);
1936 * return the height in units of the current font in the given size
1938 function getFontHeight($size){
1939 if (!$this->numFonts){
1940 $this->selectFont('./fonts/Helvetica');
1942 // for the current font, and the given size, what is the height of the font in user units
1943 $h = $this->fonts[$this->currentFont]['FontBBox'][3]-$this->fonts[$this->currentFont]['FontBBox'][1];
1944 return $size*$h/1000;
1948 * return the font decender, this will normally return a negative number
1949 * if you add this number to the baseline, you get the level of the bottom of the font
1950 * it is in the pdf user units
1952 function getFontDecender($size){
1953 // note that this will most likely return a negative value
1954 if (!$this->numFonts){
1955 $this->selectFont('./fonts/Helvetica');
1957 $h = $this->fonts[$this->currentFont]['FontBBox'][1];
1958 return $size*$h/1000;
1962 * filter the text, this is applied to all text just before being inserted into the pdf document
1963 * it escapes the various things that need to be escaped, and so on
1965 * @access private
1967 function filterText($text){
1968 $text = str_replace('\\','\\\\',$text);
1969 $text = str_replace('(','\(',$text);
1970 $text = str_replace(')','\)',$text);
1971 $text = str_replace('&lt;','<',$text);
1972 $text = str_replace('&gt;','>',$text);
1973 $text = str_replace('&#039;','\'',$text);
1974 $text = str_replace('&quot;','"',$text);
1975 $text = str_replace('&amp;','&',$text);
1977 return $text;
1981 * given a start position and information about how text is to be laid out, calculate where
1982 * on the page the text will end
1984 * @access private
1986 function PRVTgetTextPosition($x,$y,$angle,$size,$wa,$text){
1987 // given this information return an array containing x and y for the end position as elements 0 and 1
1988 $w = $this->getTextWidth($size,$text);
1989 // need to adjust for the number of spaces in this text
1990 $words = explode(' ',$text);
1991 $nspaces=count($words)-1;
1992 $w += $wa*$nspaces;
1993 $a = deg2rad((float)$angle);
1994 return array(cos($a)*$w+$x,-sin($a)*$w+$y);
1998 * wrapper function for PRVTcheckTextDirective1
2000 * @access private
2002 function PRVTcheckTextDirective(&$text,$i,&$f){
2003 $x=0;
2004 $y=0;
2005 return $this->PRVTcheckTextDirective1($text,$i,$f,0,$x,$y);
2009 * checks if the text stream contains a control directive
2010 * if so then makes some changes and returns the number of characters involved in the directive
2011 * this has been re-worked to include everything neccesary to fins the current writing point, so that
2012 * the location can be sent to the callback function if required
2013 * if the directive does not require a font change, then $f should be set to 0
2015 * @access private
2017 function PRVTcheckTextDirective1(&$text,$i,&$f,$final,&$x,&$y,$size=0,$angle=0,$wordSpaceAdjust=0){
2018 $directive = 0;
2019 $j=$i;
2020 if ($text[$j]=='<'){
2021 $j++;
2022 switch($text[$j]){
2023 case '/':
2024 $j++;
2025 if (strlen($text) <= $j){
2026 return $directive;
2028 switch($text[$j]){
2029 case 'b':
2030 case 'i':
2031 $j++;
2032 if ($text[$j]=='>'){
2033 $p = strrpos($this->currentTextState,$text[$j-1]);
2034 if ($p !== false){
2035 // then there is one to remove
2036 $this->currentTextState = substr($this->currentTextState,0,$p).substr($this->currentTextState,$p+1);
2038 $directive=$j-$i+1;
2040 break;
2041 case 'c':
2042 // this this might be a callback function
2043 $j++;
2044 $k = strpos($text,'>',$j);
2045 if ($k!==false && $text[$j]==':'){
2046 // then this will be treated as a callback directive
2047 $directive = $k-$i+1;
2048 $f=0;
2049 // split the remainder on colons to get the function name and the paramater
2050 $tmp = substr($text,$j+1,$k-$j-1);
2051 $b1 = strpos($tmp,':');
2052 if ($b1!==false){
2053 $func = substr($tmp,0,$b1);
2054 $parm = substr($tmp,$b1+1);
2055 } else {
2056 $func=$tmp;
2057 $parm='';
2059 if (!isset($func) || !strlen(trim($func))){
2060 $directive=0;
2061 } else {
2062 // only call the function if this is the final call
2063 if ($final){
2064 // need to assess the text position, calculate the text width to this point
2065 // can use getTextWidth to find the text width I think
2066 $tmp = $this->PRVTgetTextPosition($x,$y,$angle,$size,$wordSpaceAdjust,substr($text,0,$i));
2067 $info = array('x'=>$tmp[0],'y'=>$tmp[1],'angle'=>$angle,'status'=>'end','p'=>$parm,'nCallback'=>$this->nCallback);
2068 $x=$tmp[0];
2069 $y=$tmp[1];
2070 $ret = $this->$func($info);
2071 if (is_array($ret)){
2072 // then the return from the callback function could set the position, to start with, later will do font colour, and font
2073 foreach($ret as $rk=>$rv){
2074 switch($rk){
2075 case 'x':
2076 case 'y':
2077 $$rk=$rv;
2078 break;
2082 // also remove from to the stack
2083 // for simplicity, just take from the end, fix this another day
2084 $this->nCallback--;
2085 if ($this->nCallback<0){
2086 $this->nCallBack=0;
2091 break;
2093 break;
2094 case 'b':
2095 case 'i':
2096 $j++;
2097 if ($text[$j]=='>'){
2098 $this->currentTextState.=$text[$j-1];
2099 $directive=$j-$i+1;
2101 break;
2102 case 'C':
2103 $noClose=1;
2104 case 'c':
2105 // this this might be a callback function
2106 $j++;
2107 $k = strpos($text,'>',$j);
2108 if ($k!==false && $text[$j]==':'){
2109 // then this will be treated as a callback directive
2110 $directive = $k-$i+1;
2111 $f=0;
2112 // split the remainder on colons to get the function name and the paramater
2113 // $bits = explode(':',substr($text,$j+1,$k-$j-1));
2114 $tmp = substr($text,$j+1,$k-$j-1);
2115 $b1 = strpos($tmp,':');
2116 if ($b1!==false){
2117 $func = substr($tmp,0,$b1);
2118 $parm = substr($tmp,$b1+1);
2119 } else {
2120 $func=$tmp;
2121 $parm='';
2123 if (!isset($func) || !strlen(trim($func))){
2124 $directive=0;
2125 } else {
2126 // only call the function if this is the final call, ie, the one actually doing printing, not measurement
2127 if ($final){
2128 // need to assess the text position, calculate the text width to this point
2129 // can use getTextWidth to find the text width I think
2130 // also add the text height and decender
2131 $tmp = $this->PRVTgetTextPosition($x,$y,$angle,$size,$wordSpaceAdjust,substr($text,0,$i));
2132 $info = array('x'=>$tmp[0],'y'=>$tmp[1],'angle'=>$angle,'status'=>'start','p'=>$parm,'f'=>$func,'height'=>$this->getFontHeight($size),'decender'=>$this->getFontDecender($size));
2133 $x=$tmp[0];
2134 $y=$tmp[1];
2135 if (!isset($noClose) || !$noClose){
2136 // only add to the stack if this is a small 'c', therefore is a start-stop pair
2137 $this->nCallback++;
2138 $info['nCallback']=$this->nCallback;
2139 $this->callback[$this->nCallback]=$info;
2141 $ret = $this->$func($info);
2142 if (is_array($ret)){
2143 // then the return from the callback function could set the position, to start with, later will do font colour, and font
2144 foreach($ret as $rk=>$rv){
2145 switch($rk){
2146 case 'x':
2147 case 'y':
2148 $$rk=$rv;
2149 break;
2156 break;
2159 return $directive;
2163 * add text to the document, at a specified location, size and angle on the page
2165 function addText($x,$y,$size,$text,$angle=0,$wordSpaceAdjust=0){
2166 if (!$this->numFonts){$this->selectFont('./fonts/Helvetica');}
2168 // if there are any open callbacks, then they should be called, to show the start of the line
2169 if ($this->nCallback>0){
2170 for ($i=$this->nCallback;$i>0;$i--){
2171 // call each function
2172 $info = array('x'=>$x,'y'=>$y,'angle'=>$angle,'status'=>'sol','p'=>$this->callback[$i]['p'],'nCallback'=>$this->callback[$i]['nCallback'],'height'=>$this->callback[$i]['height'],'decender'=>$this->callback[$i]['decender']);
2173 $func = $this->callback[$i]['f'];
2174 $this->$func($info);
2177 if ($angle==0){
2178 $this->objects[$this->currentContents]['c'].="\n".'BT '.sprintf('%.3f',$x).' '.sprintf('%.3f',$y).' Td';
2179 } else {
2180 $a = deg2rad((float)$angle);
2181 $tmp = "\n".'BT ';
2182 $tmp .= sprintf('%.3f',cos($a)).' '.sprintf('%.3f',(-1.0*sin($a))).' '.sprintf('%.3f',sin($a)).' '.sprintf('%.3f',cos($a)).' ';
2183 $tmp .= sprintf('%.3f',$x).' '.sprintf('%.3f',$y).' Tm';
2184 $this->objects[$this->currentContents]['c'] .= $tmp;
2186 if ($wordSpaceAdjust!=0 || $wordSpaceAdjust != $this->wordSpaceAdjust){
2187 $this->wordSpaceAdjust=$wordSpaceAdjust;
2188 $this->objects[$this->currentContents]['c'].=' '.sprintf('%.3f',$wordSpaceAdjust).' Tw';
2190 $len=strlen($text);
2191 $start=0;
2192 for ($i=0;$i<$len;$i++){
2193 $f=1;
2194 $directive = $this->PRVTcheckTextDirective($text,$i,$f);
2195 if ($directive){
2196 // then we should write what we need to
2197 if ($i>$start){
2198 $part = substr($text,$start,$i-$start);
2199 $this->objects[$this->currentContents]['c'].=' /F'.$this->currentFontNum.' '.sprintf('%.1f',$size).' Tf ';
2200 $this->objects[$this->currentContents]['c'].=' ('.$this->filterText($part).') Tj';
2202 if ($f){
2203 // then there was nothing drastic done here, restore the contents
2204 $this->setCurrentFont();
2205 } else {
2206 $this->objects[$this->currentContents]['c'] .= ' ET';
2207 $f=1;
2208 $xp=$x;
2209 $yp=$y;
2210 $directive = $this->PRVTcheckTextDirective1($text,$i,$f,1,$xp,$yp,$size,$angle,$wordSpaceAdjust);
2212 // restart the text object
2213 if ($angle==0){
2214 $this->objects[$this->currentContents]['c'].="\n".'BT '.sprintf('%.3f',$xp).' '.sprintf('%.3f',$yp).' Td';
2215 } else {
2216 $a = deg2rad((float)$angle);
2217 $tmp = "\n".'BT ';
2218 $tmp .= sprintf('%.3f',cos($a)).' '.sprintf('%.3f',(-1.0*sin($a))).' '.sprintf('%.3f',sin($a)).' '.sprintf('%.3f',cos($a)).' ';
2219 $tmp .= sprintf('%.3f',$xp).' '.sprintf('%.3f',$yp).' Tm';
2220 $this->objects[$this->currentContents]['c'] .= $tmp;
2222 if ($wordSpaceAdjust!=0 || $wordSpaceAdjust != $this->wordSpaceAdjust){
2223 $this->wordSpaceAdjust=$wordSpaceAdjust;
2224 $this->objects[$this->currentContents]['c'].=' '.sprintf('%.3f',$wordSpaceAdjust).' Tw';
2227 // and move the writing point to the next piece of text
2228 $i=$i+$directive-1;
2229 $start=$i+1;
2233 if ($start<$len){
2234 $part = substr($text,$start);
2235 $this->objects[$this->currentContents]['c'].=' /F'.$this->currentFontNum.' '.sprintf('%.1f',$size).' Tf ';
2236 $this->objects[$this->currentContents]['c'].=' ('.$this->filterText($part).') Tj';
2238 $this->objects[$this->currentContents]['c'].=' ET';
2240 // if there are any open callbacks, then they should be called, to show the end of the line
2241 if ($this->nCallback>0){
2242 for ($i=$this->nCallback;$i>0;$i--){
2243 // call each function
2244 $tmp = $this->PRVTgetTextPosition($x,$y,$angle,$size,$wordSpaceAdjust,$text);
2245 $info = array('x'=>$tmp[0],'y'=>$tmp[1],'angle'=>$angle,'status'=>'eol','p'=>$this->callback[$i]['p'],'nCallback'=>$this->callback[$i]['nCallback'],'height'=>$this->callback[$i]['height'],'decender'=>$this->callback[$i]['decender']);
2246 $func = $this->callback[$i]['f'];
2247 $this->$func($info);
2254 * calculate how wide a given text string will be on a page, at a given size.
2255 * this can be called externally, but is alse used by the other class functions
2257 function getTextWidth($size,$text){
2258 // this function should not change any of the settings, though it will need to
2259 // track any directives which change during calculation, so copy them at the start
2260 // and put them back at the end.
2261 $store_currentTextState = $this->currentTextState;
2263 if (!$this->numFonts){
2264 $this->selectFont('./fonts/Helvetica');
2267 // converts a number or a float to a string so it can get the width
2268 $text = "$text";
2270 // hmm, this is where it all starts to get tricky - use the font information to
2271 // calculate the width of each character, add them up and convert to user units
2272 $w=0;
2273 $len=strlen($text);
2274 $cf = $this->currentFont;
2275 for ($i=0;$i<$len;$i++){
2276 $f=1;
2277 $directive = $this->PRVTcheckTextDirective($text,$i,$f);
2278 if ($directive){
2279 if ($f){
2280 $this->setCurrentFont();
2281 $cf = $this->currentFont;
2283 $i=$i+$directive-1;
2284 } else {
2285 $char=ord($text[$i]);
2286 if (isset($this->fonts[$cf]['differences'][$char])){
2287 // then this character is being replaced by another
2288 $name = $this->fonts[$cf]['differences'][$char];
2289 if (isset($this->fonts[$cf]['C'][$name]['WX'])){
2290 $w+=$this->fonts[$cf]['C'][$name]['WX'];
2292 } else if (isset($this->fonts[$cf]['C'][$char]['WX'])){
2293 $w+=$this->fonts[$cf]['C'][$char]['WX'];
2298 $this->currentTextState = $store_currentTextState;
2299 $this->setCurrentFont();
2301 return $w*$size/1000;
2305 * do a part of the calculation for sorting out the justification of the text
2307 * @access private
2309 function PRVTadjustWrapText($text,$actual,$width,&$x,&$adjust,$justification){
2310 switch ($justification){
2311 case 'left':
2312 return;
2313 break;
2314 case 'right':
2315 $x+=$width-$actual;
2316 break;
2317 case 'center':
2318 case 'centre':
2319 $x+=($width-$actual)/2;
2320 break;
2321 case 'full':
2322 // count the number of words
2323 $words = explode(' ',$text);
2324 $nspaces=count($words)-1;
2325 if ($nspaces>0){
2326 $adjust = ($width-$actual)/$nspaces;
2327 } else {
2328 $adjust=0;
2330 break;
2335 * add text to the page, but ensure that it fits within a certain width
2336 * if it does not fit then put in as much as possible, splitting at word boundaries
2337 * and return the remainder.
2338 * justification and angle can also be specified for the text
2340 function addTextWrap($x,$y,$width,$size,$text,$justification='left',$angle=0,$test=0){
2341 // this will display the text, and if it goes beyond the width $width, will backtrack to the
2342 // previous space or hyphen, and return the remainder of the text.
2344 // $justification can be set to 'left','right','center','centre','full'
2346 // need to store the initial text state, as this will change during the width calculation
2347 // but will need to be re-set before printing, so that the chars work out right
2348 $store_currentTextState = $this->currentTextState;
2350 if (!$this->numFonts){$this->selectFont('./fonts/Helvetica');}
2351 if ($width<=0){
2352 // error, pretend it printed ok, otherwise risking a loop
2353 return '';
2355 $w=0;
2356 $break=0;
2357 $breakWidth=0;
2358 $len=strlen($text);
2359 $cf = $this->currentFont;
2360 $tw = $width/$size*1000;
2361 for ($i=0;$i<$len;$i++){
2362 $f=1;
2363 $directive = $this->PRVTcheckTextDirective($text,$i,$f);
2364 if ($directive){
2365 if ($f){
2366 $this->setCurrentFont();
2367 $cf = $this->currentFont;
2369 $i=$i+$directive-1;
2370 } else {
2371 $cOrd = ord($text[$i]);
2372 if (isset($this->fonts[$cf]['differences'][$cOrd])){
2373 // then this character is being replaced by another
2374 $cOrd2 = $this->fonts[$cf]['differences'][$cOrd];
2375 } else {
2376 $cOrd2 = $cOrd;
2379 if (isset($this->fonts[$cf]['C'][$cOrd2]['WX'])){
2380 $w+=$this->fonts[$cf]['C'][$cOrd2]['WX'];
2382 if ($w>$tw){
2383 // then we need to truncate this line
2384 if ($break>0){
2385 // then we have somewhere that we can split :)
2386 if ($text[$break]==' '){
2387 $tmp = substr($text,0,$break);
2388 } else {
2389 $tmp = substr($text,0,$break+1);
2391 $adjust=0;
2392 $this->PRVTadjustWrapText($tmp,$breakWidth,$width,$x,$adjust,$justification);
2394 // reset the text state
2395 $this->currentTextState = $store_currentTextState;
2396 $this->setCurrentFont();
2397 if (!$test){
2398 $this->addText($x,$y,$size,$tmp,$angle,$adjust);
2400 return substr($text,$break+1);
2401 } else {
2402 // just split before the current character
2403 $tmp = substr($text,0,$i);
2404 $adjust=0;
2405 $ctmp=ord($text[$i]);
2406 if (isset($this->fonts[$cf]['differences'][$ctmp])){
2407 $ctmp=$this->fonts[$cf]['differences'][$ctmp];
2409 $tmpw=($w-$this->fonts[$cf]['C'][$ctmp]['WX'])*$size/1000;
2410 $this->PRVTadjustWrapText($tmp,$tmpw,$width,$x,$adjust,$justification);
2411 // reset the text state
2412 $this->currentTextState = $store_currentTextState;
2413 $this->setCurrentFont();
2414 if (!$test){
2415 $this->addText($x,$y,$size,$tmp,$angle,$adjust);
2417 return substr($text,$i);
2420 if ($text[$i]=='-'){
2421 $break=$i;
2422 $breakWidth = $w*$size/1000;
2424 if ($text[$i]==' '){
2425 $break=$i;
2426 $ctmp=ord($text[$i]);
2427 if (isset($this->fonts[$cf]['differences'][$ctmp])){
2428 $ctmp=$this->fonts[$cf]['differences'][$ctmp];
2430 $breakWidth = ($w-$this->fonts[$cf]['C'][$ctmp]['WX'])*$size/1000;
2434 // then there was no need to break this line
2435 if ($justification=='full'){
2436 $justification='left';
2438 $adjust=0;
2439 $tmpw=$w*$size/1000;
2440 $this->PRVTadjustWrapText($text,$tmpw,$width,$x,$adjust,$justification);
2441 // reset the text state
2442 $this->currentTextState = $store_currentTextState;
2443 $this->setCurrentFont();
2444 if (!$test){
2445 $this->addText($x,$y,$size,$text,$angle,$adjust,$angle);
2447 return '';
2451 * this will be called at a new page to return the state to what it was on the
2452 * end of the previous page, before the stack was closed down
2453 * This is to get around not being able to have open 'q' across pages
2456 function saveState($pageEnd=0){
2457 if ($pageEnd){
2458 // this will be called at a new page to return the state to what it was on the
2459 // end of the previous page, before the stack was closed down
2460 // This is to get around not being able to have open 'q' across pages
2461 $opt = $this->stateStack[$pageEnd]; // ok to use this as stack starts numbering at 1
2462 $this->setColor($opt['col']['r'],$opt['col']['g'],$opt['col']['b'],1);
2463 $this->setStrokeColor($opt['str']['r'],$opt['str']['g'],$opt['str']['b'],1);
2464 $this->objects[$this->currentContents]['c'].="\n".$opt['lin'];
2465 // $this->currentLineStyle = $opt['lin'];
2466 } else {
2467 $this->nStateStack++;
2468 $this->stateStack[$this->nStateStack]=array(
2469 'col'=>$this->currentColour
2470 ,'str'=>$this->currentStrokeColour
2471 ,'lin'=>$this->currentLineStyle
2474 $this->objects[$this->currentContents]['c'].="\nq";
2478 * restore a previously saved state
2480 function restoreState($pageEnd=0){
2481 if (!$pageEnd){
2482 $n = $this->nStateStack;
2483 $this->currentColour = $this->stateStack[$n]['col'];
2484 $this->currentStrokeColour = $this->stateStack[$n]['str'];
2485 $this->objects[$this->currentContents]['c'].="\n".$this->stateStack[$n]['lin'];
2486 $this->currentLineStyle = $this->stateStack[$n]['lin'];
2487 unset($this->stateStack[$n]);
2488 $this->nStateStack--;
2490 $this->objects[$this->currentContents]['c'].="\nQ";
2494 * make a loose object, the output will go into this object, until it is closed, then will revert to
2495 * the current one.
2496 * this object will not appear until it is included within a page.
2497 * the function will return the object number
2499 function openObject(){
2500 $this->nStack++;
2501 $this->stack[$this->nStack]=array('c'=>$this->currentContents,'p'=>$this->currentPage);
2502 // add a new object of the content type, to hold the data flow
2503 $this->numObj++;
2504 $this->o_contents($this->numObj,'new');
2505 $this->currentContents=$this->numObj;
2506 $this->looseObjects[$this->numObj]=1;
2508 return $this->numObj;
2512 * open an existing object for editing
2514 function reopenObject($id){
2515 $this->nStack++;
2516 $this->stack[$this->nStack]=array('c'=>$this->currentContents,'p'=>$this->currentPage);
2517 $this->currentContents=$id;
2518 // also if this object is the primary contents for a page, then set the current page to its parent
2519 if (isset($this->objects[$id]['onPage'])){
2520 $this->currentPage = $this->objects[$id]['onPage'];
2525 * close an object
2527 function closeObject(){
2528 // close the object, as long as there was one open in the first place, which will be indicated by
2529 // an objectId on the stack.
2530 if ($this->nStack>0){
2531 $this->currentContents=$this->stack[$this->nStack]['c'];
2532 $this->currentPage=$this->stack[$this->nStack]['p'];
2533 $this->nStack--;
2534 // easier to probably not worry about removing the old entries, they will be overwritten
2535 // if there are new ones.
2540 * stop an object from appearing on pages from this point on
2542 function stopObject($id){
2543 // if an object has been appearing on pages up to now, then stop it, this page will
2544 // be the last one that could contian it.
2545 if (isset($this->addLooseObjects[$id])){
2546 $this->addLooseObjects[$id]='';
2551 * after an object has been created, it wil only show if it has been added, using this function.
2553 function addObject($id,$options='add'){
2554 // add the specified object to the page
2555 if (isset($this->looseObjects[$id]) && $this->currentContents!=$id){
2556 // then it is a valid object, and it is not being added to itself
2557 switch($options){
2558 case 'all':
2559 // then this object is to be added to this page (done in the next block) and
2560 // all future new pages.
2561 $this->addLooseObjects[$id]='all';
2562 case 'add':
2563 if (isset($this->objects[$this->currentContents]['onPage'])){
2564 // then the destination contents is the primary for the page
2565 // (though this object is actually added to that page)
2566 $this->o_page($this->objects[$this->currentContents]['onPage'],'content',$id);
2568 break;
2569 case 'even':
2570 $this->addLooseObjects[$id]='even';
2571 $pageObjectId=$this->objects[$this->currentContents]['onPage'];
2572 if ($this->objects[$pageObjectId]['info']['pageNum']%2==0){
2573 $this->addObject($id); // hacky huh :)
2575 break;
2576 case 'odd':
2577 $this->addLooseObjects[$id]='odd';
2578 $pageObjectId=$this->objects[$this->currentContents]['onPage'];
2579 if ($this->objects[$pageObjectId]['info']['pageNum']%2==1){
2580 $this->addObject($id); // hacky huh :)
2582 break;
2583 case 'next':
2584 $this->addLooseObjects[$id]='all';
2585 break;
2586 case 'nexteven':
2587 $this->addLooseObjects[$id]='even';
2588 break;
2589 case 'nextodd':
2590 $this->addLooseObjects[$id]='odd';
2591 break;
2597 * add content to the documents info object
2599 function addInfo($label,$value=0){
2600 // this will only work if the label is one of the valid ones.
2601 // modify this so that arrays can be passed as well.
2602 // if $label is an array then assume that it is key=>value pairs
2603 // else assume that they are both scalar, anything else will probably error
2604 if (is_array($label)){
2605 foreach ($label as $l=>$v){
2606 $this->o_info($this->infoObject,$l,$v);
2608 } else {
2609 $this->o_info($this->infoObject,$label,$value);
2614 * set the viewer preferences of the document, it is up to the browser to obey these.
2616 function setPreferences($label,$value=0){
2617 // this will only work if the label is one of the valid ones.
2618 if (is_array($label)){
2619 foreach ($label as $l=>$v){
2620 $this->o_catalog($this->catalogId,'viewerPreferences',array($l=>$v));
2622 } else {
2623 $this->o_catalog($this->catalogId,'viewerPreferences',array($label=>$value));
2628 * extract an integer from a position in a byte stream
2630 * @access private
2632 function PRVT_getBytes(&$data,$pos,$num){
2633 // return the integer represented by $num bytes from $pos within $data
2634 $ret=0;
2635 for ($i=0;$i<$num;$i++){
2636 $ret=$ret*256;
2637 $ret+=ord($data[$pos+$i]);
2639 return $ret;
2643 * add a PNG image into the document, from a file
2644 * this should work with remote files
2646 function addPngFromFile($file,$x,$y,$w=0,$h=0){
2647 // read in a png file, interpret it, then add to the system
2648 $error=0;
2649 $tmp = get_magic_quotes_runtime();
2650 set_magic_quotes_runtime(0);
2651 $fp = @fopen($file,'rb');
2652 if ($fp){
2653 $data='';
2654 while(!feof($fp)){
2655 $data .= fread($fp,1024);
2657 fclose($fp);
2658 } else {
2659 $error = 1;
2660 $errormsg = 'trouble opening file: '.$file;
2662 set_magic_quotes_runtime($tmp);
2664 if (!$error){
2665 $header = chr(137).chr(80).chr(78).chr(71).chr(13).chr(10).chr(26).chr(10);
2666 if (substr($data,0,8)!=$header){
2667 $error=1;
2668 $errormsg = 'this file does not have a valid header';
2672 if (!$error){
2673 // set pointer
2674 $p = 8;
2675 $len = strlen($data);
2676 // cycle through the file, identifying chunks
2677 $haveHeader=0;
2678 $info=array();
2679 $idata='';
2680 $pdata='';
2681 while ($p<$len){
2682 $chunkLen = $this->PRVT_getBytes($data,$p,4);
2683 $chunkType = substr($data,$p+4,4);
2684 // echo $chunkType.' - '.$chunkLen.'<br>';
2686 switch($chunkType){
2687 case 'IHDR':
2688 // this is where all the file information comes from
2689 $info['width']=$this->PRVT_getBytes($data,$p+8,4);
2690 $info['height']=$this->PRVT_getBytes($data,$p+12,4);
2691 $info['bitDepth']=ord($data[$p+16]);
2692 $info['colorType']=ord($data[$p+17]);
2693 $info['compressionMethod']=ord($data[$p+18]);
2694 $info['filterMethod']=ord($data[$p+19]);
2695 $info['interlaceMethod']=ord($data[$p+20]);
2696 //print_r($info);
2697 $haveHeader=1;
2698 if ($info['compressionMethod']!=0){
2699 $error=1;
2700 $errormsg = 'unsupported compression method';
2702 if ($info['filterMethod']!=0){
2703 $error=1;
2704 $errormsg = 'unsupported filter method';
2706 break;
2707 case 'PLTE':
2708 $pdata.=substr($data,$p+8,$chunkLen);
2709 break;
2710 case 'IDAT':
2711 $idata.=substr($data,$p+8,$chunkLen);
2712 break;
2713 case 'tRNS':
2714 //this chunk can only occur once and it must occur after the PLTE chunk and before IDAT chunk
2715 //print "tRNS found, color type = ".$info['colorType']."<BR>";
2716 $transparency = array();
2717 if ($info['colorType'] == 3) { // indexed color, rbg
2718 /* corresponding to entries in the plte chunk
2719 Alpha for palette index 0: 1 byte
2720 Alpha for palette index 1: 1 byte
2721 ...etc...
2723 // there will be one entry for each palette entry. up until the last non-opaque entry.
2724 // set up an array, stretching over all palette entries which will be o (opaque) or 1 (transparent)
2725 $transparency['type']='indexed';
2726 $numPalette = strlen($pdata)/3;
2727 $trans=0;
2728 for ($i=$chunkLen;$i>=0;$i--){
2729 if (ord($data[$p+8+$i])==0){
2730 $trans=$i;
2733 $transparency['data'] = $trans;
2735 } elseif($info['colorType'] == 0) { // grayscale
2736 /* corresponding to entries in the plte chunk
2737 Gray: 2 bytes, range 0 .. (2^bitdepth)-1
2739 // $transparency['grayscale']=$this->PRVT_getBytes($data,$p+8,2); // g = grayscale
2740 $transparency['type']='indexed';
2741 $transparency['data'] = ord($data[$p+8+1]);
2743 } elseif($info['colorType'] == 2) { // truecolor
2744 /* corresponding to entries in the plte chunk
2745 Red: 2 bytes, range 0 .. (2^bitdepth)-1
2746 Green: 2 bytes, range 0 .. (2^bitdepth)-1
2747 Blue: 2 bytes, range 0 .. (2^bitdepth)-1
2749 $transparency['r']=$this->PRVT_getBytes($data,$p+8,2); // r from truecolor
2750 $transparency['g']=$this->PRVT_getBytes($data,$p+10,2); // g from truecolor
2751 $transparency['b']=$this->PRVT_getBytes($data,$p+12,2); // b from truecolor
2753 } else {
2754 //unsupported transparency type
2756 // KS End new code
2757 break;
2758 default:
2759 break;
2762 $p += $chunkLen+12;
2765 if(!$haveHeader){
2766 $error = 1;
2767 $errormsg = 'information header is missing';
2769 if (isset($info['interlaceMethod']) && $info['interlaceMethod']){
2770 $error = 1;
2771 $errormsg = 'There appears to be no support for interlaced images in pdf.';
2775 if (!$error && $info['bitDepth'] > 8){
2776 $error = 1;
2777 $errormsg = 'only bit depth of 8 or less is supported';
2780 if (!$error){
2781 if ($info['colorType']!=2 && $info['colorType']!=0 && $info['colorType']!=3){
2782 $error = 1;
2783 $errormsg = 'transparancey alpha channel not supported, transparency only supported for palette images.';
2784 } else {
2785 switch ($info['colorType']){
2786 case 3:
2787 $color = 'DeviceRGB';
2788 $ncolor=1;
2789 break;
2790 case 2:
2791 $color = 'DeviceRGB';
2792 $ncolor=3;
2793 break;
2794 case 0:
2795 $color = 'DeviceGray';
2796 $ncolor=1;
2797 break;
2801 if ($error){
2802 $this->addMessage('PNG error - ('.$file.') '.$errormsg);
2803 return;
2805 if ($w==0){
2806 $w=$h/$info['height']*$info['width'];
2808 if ($h==0){
2809 $h=$w*$info['height']/$info['width'];
2811 //print_r($info);
2812 // so this image is ok... add it in.
2813 $this->numImages++;
2814 $im=$this->numImages;
2815 $label='I'.$im;
2816 $this->numObj++;
2817 // $this->o_image($this->numObj,'new',array('label'=>$label,'data'=>$idata,'iw'=>$w,'ih'=>$h,'type'=>'png','ic'=>$info['width']));
2818 $options = array('label'=>$label,'data'=>$idata,'bitsPerComponent'=>$info['bitDepth'],'pdata'=>$pdata
2819 ,'iw'=>$info['width'],'ih'=>$info['height'],'type'=>'png','color'=>$color,'ncolor'=>$ncolor);
2820 if (isset($transparency)){
2821 $options['transparency']=$transparency;
2823 $this->o_image($this->numObj,'new',$options);
2825 $this->objects[$this->currentContents]['c'].="\nq";
2826 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$w)." 0 0 ".sprintf('%.3f',$h)." ".sprintf('%.3f',$x)." ".sprintf('%.3f',$y)." cm";
2827 $this->objects[$this->currentContents]['c'].="\n/".$label.' Do';
2828 $this->objects[$this->currentContents]['c'].="\nQ";
2832 * add a JPEG image into the document, from a file
2834 function addJpegFromFile($img,$x,$y,$w=0,$h=0){
2835 // attempt to add a jpeg image straight from a file, using no GD commands
2836 // note that this function is unable to operate on a remote file.
2838 if (!file_exists($img)){
2839 return;
2842 $tmp=getimagesize($img);
2843 $imageWidth=$tmp[0];
2844 $imageHeight=$tmp[1];
2846 if (isset($tmp['channels'])){
2847 $channels = $tmp['channels'];
2848 } else {
2849 $channels = 3;
2852 if ($w<=0 && $h<=0){
2853 $w=$imageWidth;
2855 if ($w==0){
2856 $w=$h/$imageHeight*$imageWidth;
2858 if ($h==0){
2859 $h=$w*$imageHeight/$imageWidth;
2862 $fp=fopen($img,'rb');
2864 $tmp = get_magic_quotes_runtime();
2865 set_magic_quotes_runtime(0);
2866 $data = fread($fp,filesize($img));
2867 set_magic_quotes_runtime($tmp);
2869 fclose($fp);
2871 $this->addJpegImage_common($data,$x,$y,$w,$h,$imageWidth,$imageHeight,$channels);
2875 * add an image into the document, from a GD object
2876 * this function is not all that reliable, and I would probably encourage people to use
2877 * the file based functions
2879 function addImage(&$img,$x,$y,$w=0,$h=0,$quality=75){
2880 // add a new image into the current location, as an external object
2881 // add the image at $x,$y, and with width and height as defined by $w & $h
2883 // note that this will only work with full colour images and makes them jpg images for display
2884 // later versions could present lossless image formats if there is interest.
2886 // there seems to be some problem here in that images that have quality set above 75 do not appear
2887 // not too sure why this is, but in the meantime I have restricted this to 75.
2888 if ($quality>75){
2889 $quality=75;
2892 // if the width or height are set to zero, then set the other one based on keeping the image
2893 // height/width ratio the same, if they are both zero, then give up :)
2894 $imageWidth=imagesx($img);
2895 $imageHeight=imagesy($img);
2897 if ($w<=0 && $h<=0){
2898 return;
2900 if ($w==0){
2901 $w=$h/$imageHeight*$imageWidth;
2903 if ($h==0){
2904 $h=$w*$imageHeight/$imageWidth;
2907 // gotta get the data out of the img..
2909 // so I write to a temp file, and then read it back.. soo ugly, my apologies.
2910 $tmpDir='/tmp';
2911 $tmpName=tempnam($tmpDir,'img');
2912 imagejpeg($img,$tmpName,$quality);
2913 $fp=fopen($tmpName,'rb');
2915 $tmp = get_magic_quotes_runtime();
2916 set_magic_quotes_runtime(0);
2917 $fp = @fopen($tmpName,'rb');
2918 if ($fp){
2919 $data='';
2920 while(!feof($fp)){
2921 $data .= fread($fp,1024);
2923 fclose($fp);
2924 } else {
2925 $error = 1;
2926 $errormsg = 'trouble opening file';
2928 // $data = fread($fp,filesize($tmpName));
2929 set_magic_quotes_runtime($tmp);
2930 // fclose($fp);
2931 unlink($tmpName);
2932 $this->addJpegImage_common($data,$x,$y,$w,$h,$imageWidth,$imageHeight);
2936 * common code used by the two JPEG adding functions
2938 * @access private
2940 function addJpegImage_common(&$data,$x,$y,$w=0,$h=0,$imageWidth,$imageHeight,$channels=3){
2941 // note that this function is not to be called externally
2942 // it is just the common code between the GD and the file options
2943 $this->numImages++;
2944 $im=$this->numImages;
2945 $label='I'.$im;
2946 $this->numObj++;
2947 $this->o_image($this->numObj,'new',array('label'=>$label,'data'=>$data,'iw'=>$imageWidth,'ih'=>$imageHeight,'channels'=>$channels));
2949 $this->objects[$this->currentContents]['c'].="\nq";
2950 $this->objects[$this->currentContents]['c'].="\n".sprintf('%.3f',$w)." 0 0 ".sprintf('%.3f',$h)." ".sprintf('%.3f',$x)." ".sprintf('%.3f',$y)." cm";
2951 $this->objects[$this->currentContents]['c'].="\n/".$label.' Do';
2952 $this->objects[$this->currentContents]['c'].="\nQ";
2956 * specify where the document should open when it first starts
2958 function openHere($style,$a=0,$b=0,$c=0){
2959 // this function will open the document at a specified page, in a specified style
2960 // the values for style, and the required paramters are:
2961 // 'XYZ' left, top, zoom
2962 // 'Fit'
2963 // 'FitH' top
2964 // 'FitV' left
2965 // 'FitR' left,bottom,right
2966 // 'FitB'
2967 // 'FitBH' top
2968 // 'FitBV' left
2969 $this->numObj++;
2970 $this->o_destination($this->numObj,'new',array('page'=>$this->currentPage,'type'=>$style,'p1'=>$a,'p2'=>$b,'p3'=>$c));
2971 $id = $this->catalogId;
2972 $this->o_catalog($id,'openHere',$this->numObj);
2976 * create a labelled destination within the document
2978 function addDestination($label,$style,$a=0,$b=0,$c=0){
2979 // associates the given label with the destination, it is done this way so that a destination can be specified after
2980 // it has been linked to
2981 // styles are the same as the 'openHere' function
2982 $this->numObj++;
2983 $this->o_destination($this->numObj,'new',array('page'=>$this->currentPage,'type'=>$style,'p1'=>$a,'p2'=>$b,'p3'=>$c));
2984 $id = $this->numObj;
2985 // store the label->idf relationship, note that this means that labels can be used only once
2986 $this->destinations["$label"]=$id;
2990 * define font families, this is used to initialize the font families for the default fonts
2991 * and for the user to add new ones for their fonts. The default bahavious can be overridden should
2992 * that be desired.
2994 function setFontFamily($family,$options=''){
2995 if (!is_array($options)){
2996 if ($family=='init'){
2997 // set the known family groups
2998 // these font families will be used to enable bold and italic markers to be included
2999 // within text streams. html forms will be used... <b></b> <i></i>
3000 $this->fontFamilies['Helvetica.afm']=array(
3001 'b'=>'Helvetica-Bold.afm'
3002 ,'i'=>'Helvetica-Oblique.afm'
3003 ,'bi'=>'Helvetica-BoldOblique.afm'
3004 ,'ib'=>'Helvetica-BoldOblique.afm'
3006 $this->fontFamilies['Courier.afm']=array(
3007 'b'=>'Courier-Bold.afm'
3008 ,'i'=>'Courier-Oblique.afm'
3009 ,'bi'=>'Courier-BoldOblique.afm'
3010 ,'ib'=>'Courier-BoldOblique.afm'
3012 $this->fontFamilies['Times-Roman.afm']=array(
3013 'b'=>'Times-Bold.afm'
3014 ,'i'=>'Times-Italic.afm'
3015 ,'bi'=>'Times-BoldItalic.afm'
3016 ,'ib'=>'Times-BoldItalic.afm'
3019 } else {
3020 // the user is trying to set a font family
3021 // note that this can also be used to set the base ones to something else
3022 if (strlen($family)){
3023 $this->fontFamilies[$family] = $options;
3029 * used to add messages for use in debugging
3031 function addMessage($message){
3032 $this->messages.=$message."\n";
3036 * a few functions which should allow the document to be treated transactionally.
3038 function transaction($action){
3039 switch ($action){
3040 case 'start':
3041 // store all the data away into the checkpoint variable
3042 $data = get_object_vars($this);
3043 $this->checkpoint = $data;
3044 unset($data);
3045 break;
3046 case 'commit':
3047 if (is_array($this->checkpoint) && isset($this->checkpoint['checkpoint'])){
3048 $tmp = $this->checkpoint['checkpoint'];
3049 $this->checkpoint = $tmp;
3050 unset($tmp);
3051 } else {
3052 $this->checkpoint='';
3054 break;
3055 case 'rewind':
3056 // do not destroy the current checkpoint, but move us back to the state then, so that we can try again
3057 if (is_array($this->checkpoint)){
3058 // can only abort if were inside a checkpoint
3059 $tmp = $this->checkpoint;
3060 foreach ($tmp as $k=>$v){
3061 if ($k != 'checkpoint'){
3062 $this->$k=$v;
3065 unset($tmp);
3067 break;
3068 case 'abort':
3069 if (is_array($this->checkpoint)){
3070 // can only abort if were inside a checkpoint
3071 $tmp = $this->checkpoint;
3072 foreach ($tmp as $k=>$v){
3073 $this->$k=$v;
3075 unset($tmp);
3077 break;
3082 } // end of class