File indexing completed on 2025-01-19 05:21:23

0001 <?php
0002 /**
0003  * Zend Framework
0004  *
0005  * LICENSE
0006  *
0007  * This source file is subject to the new BSD license that is bundled
0008  * with this package in the file LICENSE.txt.
0009  * It is also available through the world-wide-web at this URL:
0010  * http://framework.zend.com/license/new-bsd
0011  * If you did not receive a copy of the license and are unable to
0012  * obtain it through the world-wide-web, please send an email
0013  * to license@zend.com so we can send you a copy immediately.
0014  *
0015  * @category   Zend
0016  * @package    Zend_Pdf
0017  * @copyright  Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com)
0018  * @license    http://framework.zend.com/license/new-bsd     New BSD License
0019  * @version    $Id$
0020  */
0021 
0022 /** Internally used classes */
0023 // require_once 'Zend/Pdf/Element.php';
0024 // require_once 'Zend/Pdf/Element/Array.php';
0025 // require_once 'Zend/Pdf/Element/String/Binary.php';
0026 // require_once 'Zend/Pdf/Element/Boolean.php';
0027 // require_once 'Zend/Pdf/Element/Dictionary.php';
0028 // require_once 'Zend/Pdf/Element/Name.php';
0029 // require_once 'Zend/Pdf/Element/Null.php';
0030 // require_once 'Zend/Pdf/Element/Numeric.php';
0031 // require_once 'Zend/Pdf/Element/String.php';
0032 // require_once 'Zend/Pdf/Resource/Unified.php';
0033 
0034 // require_once 'Zend/Pdf/Canvas/Abstract.php';
0035 
0036 
0037 /**
0038  * PDF Page
0039  *
0040  * @package    Zend_Pdf
0041  * @copyright  Copyright (c) 2005-2015 Zend Technologies USA Inc. (http://www.zend.com)
0042  * @license    http://framework.zend.com/license/new-bsd     New BSD License
0043  */
0044 class Zend_Pdf_Page extends Zend_Pdf_Canvas_Abstract
0045 {
0046   /**** Class Constants ****/
0047 
0048 
0049   /* Page Sizes */
0050 
0051     /**
0052      * Size representing an A4 page in portrait (tall) orientation.
0053      */
0054     const SIZE_A4                = '595:842:';
0055 
0056     /**
0057      * Size representing an A4 page in landscape (wide) orientation.
0058      */
0059     const SIZE_A4_LANDSCAPE      = '842:595:';
0060 
0061     /**
0062      * Size representing a US Letter page in portrait (tall) orientation.
0063      */
0064     const SIZE_LETTER            = '612:792:';
0065 
0066     /**
0067      * Size representing a US Letter page in landscape (wide) orientation.
0068      */
0069     const SIZE_LETTER_LANDSCAPE  = '792:612:';
0070 
0071 
0072   /* Shape Drawing */
0073 
0074     /**
0075      * Stroke the path only. Do not fill.
0076      */
0077     const SHAPE_DRAW_STROKE      = 0;
0078 
0079     /**
0080      * Fill the path only. Do not stroke.
0081      */
0082     const SHAPE_DRAW_FILL        = 1;
0083 
0084     /**
0085      * Fill and stroke the path.
0086      */
0087     const SHAPE_DRAW_FILL_AND_STROKE = 2;
0088 
0089 
0090   /* Shape Filling Methods */
0091 
0092     /**
0093      * Fill the path using the non-zero winding rule.
0094      */
0095     const FILL_METHOD_NON_ZERO_WINDING = 0;
0096 
0097     /**
0098      * Fill the path using the even-odd rule.
0099      */
0100     const FILL_METHOD_EVEN_ODD        = 1;
0101 
0102 
0103   /* Line Dash Types */
0104 
0105     /**
0106      * Solid line dash.
0107      */
0108     const LINE_DASHING_SOLID = 0;
0109 
0110 
0111 
0112     /**
0113      * Page dictionary (refers to an inderect Zend_Pdf_Element_Dictionary object).
0114      *
0115      * @var Zend_Pdf_Element_Reference|Zend_Pdf_Element_Object
0116      */
0117     protected $_dictionary;
0118 
0119     /**
0120      * PDF objects factory.
0121      *
0122      * @var Zend_Pdf_ElementFactory_Interface
0123      */
0124     protected $_objFactory = null;
0125 
0126     /**
0127      * Flag which signals, that page is created separately from any PDF document or
0128      * attached to anyone.
0129      *
0130      * @var boolean
0131      */
0132     protected $_attached;
0133 
0134     /**
0135      * Safe Graphics State semafore
0136      *
0137      * If it's false, than we can't be sure Graphics State is restored withing
0138      * context of previous contents stream (ex. drawing coordinate system may be rotated).
0139      * We should encompass existing content with save/restore GS operators
0140      *
0141      * @var boolean
0142      */
0143     protected $_safeGS;
0144 
0145     /**
0146      * Object constructor.
0147      * Constructor signatures:
0148      *
0149      * 1. Load PDF page from a parsed PDF file.
0150      *    Object factory is created by PDF parser.
0151      * ---------------------------------------------------------
0152      * new Zend_Pdf_Page(Zend_Pdf_Element_Dictionary       $pageDict,
0153      *                   Zend_Pdf_ElementFactory_Interface $factory);
0154      * ---------------------------------------------------------
0155      *
0156      * 2. Make a copy of the PDF page.
0157      *    New page is created in the same context as source page. Object factory is shared.
0158      *    Thus it will be attached to the document, but need to be placed into Zend_Pdf::$pages array
0159      *    to be included into output.
0160      * ---------------------------------------------------------
0161      * new Zend_Pdf_Page(Zend_Pdf_Page $page);
0162      * ---------------------------------------------------------
0163      *
0164      * 3. Create new page with a specified pagesize.
0165      *    If $factory is null then it will be created and page must be attached to the document to be
0166      *    included into output.
0167      * ---------------------------------------------------------
0168      * new Zend_Pdf_Page(string $pagesize, Zend_Pdf_ElementFactory_Interface $factory = null);
0169      * ---------------------------------------------------------
0170      *
0171      * 4. Create new page with a specified pagesize (in default user space units).
0172      *    If $factory is null then it will be created and page must be attached to the document to be
0173      *    included into output.
0174      * ---------------------------------------------------------
0175      * new Zend_Pdf_Page(numeric $width, numeric $height, Zend_Pdf_ElementFactory_Interface $factory = null);
0176      * ---------------------------------------------------------
0177      *
0178      *
0179      * @param mixed $param1
0180      * @param mixed $param2
0181      * @param mixed $param3
0182      * @throws Zend_Pdf_Exception
0183      */
0184     public function __construct($param1, $param2 = null, $param3 = null)
0185     {
0186         if (($param1 instanceof Zend_Pdf_Element_Reference ||
0187              $param1 instanceof Zend_Pdf_Element_Object
0188             ) &&
0189             $param2 instanceof Zend_Pdf_ElementFactory_Interface &&
0190             $param3 === null
0191            ) {
0192             switch ($param1->getType()) {
0193                 case Zend_Pdf_Element::TYPE_DICTIONARY:
0194                     $this->_dictionary = $param1;
0195                     $this->_objFactory = $param2;
0196                     $this->_attached   = true;
0197                     $this->_safeGS     = false;
0198                     return;
0199                     break;
0200 
0201                 case Zend_Pdf_Element::TYPE_NULL:
0202                     $this->_objFactory = $param2;
0203                     $pageWidth = $pageHeight = 0;
0204                     break;
0205 
0206                 default:
0207                     // require_once 'Zend/Pdf/Exception.php';
0208                     throw new Zend_Pdf_Exception('Unrecognized object type.');
0209                     break;
0210 
0211             }
0212         } else if ($param1 instanceof Zend_Pdf_Page && $param2 === null && $param3 === null) {
0213             // Duplicate existing page.
0214             // Let already existing content and resources to be shared between pages
0215             // We don't give existing content modification functionality, so we don't need "deep copy"
0216             $this->_objFactory = $param1->_objFactory;
0217             $this->_attached   = &$param1->_attached;
0218             $this->_safeGS     = false;
0219 
0220             $this->_dictionary = $this->_objFactory->newObject(new Zend_Pdf_Element_Dictionary());
0221 
0222             foreach ($param1->_dictionary->getKeys() as $key) {
0223                 if ($key == 'Contents') {
0224                     // Clone Contents property
0225 
0226                     $this->_dictionary->Contents = new Zend_Pdf_Element_Array();
0227 
0228                     if ($param1->_dictionary->Contents->getType() != Zend_Pdf_Element::TYPE_ARRAY) {
0229                         // Prepare array of content streams and add existing stream
0230                         $this->_dictionary->Contents->items[] = $param1->_dictionary->Contents;
0231                     } else {
0232                         // Clone array of the content streams
0233                         foreach ($param1->_dictionary->Contents->items as $srcContentStream) {
0234                             $this->_dictionary->Contents->items[] = $srcContentStream;
0235                         }
0236                     }
0237                 } else {
0238                     $this->_dictionary->$key = $param1->_dictionary->$key;
0239                 }
0240             }
0241 
0242             return;
0243         } else if (is_string($param1) &&
0244                    ($param2 === null || $param2 instanceof Zend_Pdf_ElementFactory_Interface) &&
0245                    $param3 === null) {
0246             if ($param2 !== null) {
0247                 $this->_objFactory = $param2;
0248             } else {
0249                 // require_once 'Zend/Pdf/ElementFactory.php';
0250                 $this->_objFactory = Zend_Pdf_ElementFactory::createFactory(1);
0251             }
0252             $this->_attached   = false;
0253             $this->_safeGS     = true; /** New page created. That's users App responsibility to track GS changes */
0254 
0255             switch (strtolower($param1)) {
0256                 case 'a4':
0257                     $param1 = Zend_Pdf_Page::SIZE_A4;
0258                     break;
0259                 case 'a4-landscape':
0260                     $param1 = Zend_Pdf_Page::SIZE_A4_LANDSCAPE;
0261                     break;
0262                 case 'letter':
0263                     $param1 = Zend_Pdf_Page::SIZE_LETTER;
0264                     break;
0265                 case 'letter-landscape':
0266                     $param1 = Zend_Pdf_Page::SIZE_LETTER_LANDSCAPE;
0267                     break;
0268                 default:
0269                     // should be in "x:y" or "x:y:" form
0270             }
0271 
0272             $pageDim = explode(':', $param1);
0273             if(count($pageDim) == 2  ||  count($pageDim) == 3) {
0274                 $pageWidth  = $pageDim[0];
0275                 $pageHeight = $pageDim[1];
0276             } else {
0277                 /**
0278                  * @todo support of user defined pagesize notations, like:
0279                  *       "210x297mm", "595x842", "8.5x11in", "612x792"
0280                  */
0281                 // require_once 'Zend/Pdf/Exception.php';
0282                 throw new Zend_Pdf_Exception('Wrong pagesize notation.');
0283             }
0284             /**
0285              * @todo support of pagesize recalculation to "default user space units"
0286              */
0287 
0288         } else if (is_numeric($param1) && is_numeric($param2) &&
0289                    ($param3 === null || $param3 instanceof Zend_Pdf_ElementFactory_Interface)) {
0290             if ($param3 !== null) {
0291                 $this->_objFactory = $param3;
0292             } else {
0293                 // require_once 'Zend/Pdf/ElementFactory.php';
0294                 $this->_objFactory = Zend_Pdf_ElementFactory::createFactory(1);
0295             }
0296 
0297             $this->_attached = false;
0298             $this->_safeGS   = true; /** New page created. That's users App responsibility to track GS changes */
0299             $pageWidth  = $param1;
0300             $pageHeight = $param2;
0301 
0302         } else {
0303             // require_once 'Zend/Pdf/Exception.php';
0304             throw new Zend_Pdf_Exception('Unrecognized method signature, wrong number of arguments or wrong argument types.');
0305         }
0306 
0307         $this->_dictionary = $this->_objFactory->newObject(new Zend_Pdf_Element_Dictionary());
0308         $this->_dictionary->Type         = new Zend_Pdf_Element_Name('Page');
0309         // require_once 'Zend/Pdf.php';
0310         $this->_dictionary->LastModified = new Zend_Pdf_Element_String(Zend_Pdf::pdfDate());
0311         $this->_dictionary->Resources    = new Zend_Pdf_Element_Dictionary();
0312         $this->_dictionary->MediaBox     = new Zend_Pdf_Element_Array();
0313         $this->_dictionary->MediaBox->items[] = new Zend_Pdf_Element_Numeric(0);
0314         $this->_dictionary->MediaBox->items[] = new Zend_Pdf_Element_Numeric(0);
0315         $this->_dictionary->MediaBox->items[] = new Zend_Pdf_Element_Numeric($pageWidth);
0316         $this->_dictionary->MediaBox->items[] = new Zend_Pdf_Element_Numeric($pageHeight);
0317         $this->_dictionary->Contents     = new Zend_Pdf_Element_Array();
0318     }
0319 
0320 
0321     /**
0322      * Attach resource to the canvas
0323      *
0324      * Method returns a name of the resource which can be used
0325      * as a resource reference within drawing instructions stream
0326      * Allowed types: 'ExtGState', 'ColorSpace', 'Pattern', 'Shading',
0327      * 'XObject', 'Font', 'Properties'
0328      *
0329      * @param string $type
0330      * @param Zend_Pdf_Resource $resource
0331      * @return string
0332      */
0333     protected function _attachResource($type, Zend_Pdf_Resource $resource)
0334     {
0335         // Check that Resources dictionary contains appropriate resource set
0336         if ($this->_dictionary->Resources->$type === null) {
0337             $this->_dictionary->Resources->touch();
0338             $this->_dictionary->Resources->$type = new Zend_Pdf_Element_Dictionary();
0339         } else {
0340             $this->_dictionary->Resources->$type->touch();
0341         }
0342 
0343         // Check, that resource is already attached to resource set.
0344         $resObject = $resource->getResource();
0345         foreach ($this->_dictionary->Resources->$type->getKeys() as $ResID) {
0346             if ($this->_dictionary->Resources->$type->$ResID === $resObject) {
0347                 return $ResID;
0348             }
0349         }
0350 
0351         $idCounter = 1;
0352         do {
0353             $newResName = $type[0] . $idCounter++;
0354         } while ($this->_dictionary->Resources->$type->$newResName !== null);
0355 
0356         $this->_dictionary->Resources->$type->$newResName = $resObject;
0357         $this->_objFactory->attach($resource->getFactory());
0358 
0359         return $newResName;
0360     }
0361 
0362     /**
0363      * Add procedureSet to the Page description
0364      *
0365      * @param string $procSetName
0366      */
0367     protected function _addProcSet($procSetName)
0368     {
0369         // Check that Resources dictionary contains ProcSet entry
0370         if ($this->_dictionary->Resources->ProcSet === null) {
0371             $this->_dictionary->Resources->touch();
0372             $this->_dictionary->Resources->ProcSet = new Zend_Pdf_Element_Array();
0373         } else {
0374             $this->_dictionary->Resources->ProcSet->touch();
0375         }
0376 
0377         foreach ($this->_dictionary->Resources->ProcSet->items as $procSetEntry) {
0378             if ($procSetEntry->value == $procSetName) {
0379                 // Procset is already included into a ProcSet array
0380                 return;
0381             }
0382         }
0383 
0384         $this->_dictionary->Resources->ProcSet->items[] = new Zend_Pdf_Element_Name($procSetName);
0385     }
0386 
0387     /**
0388      * Returns dictionaries of used resources.
0389      *
0390      * Used for canvas implementations interoperability
0391      *
0392      * Structure of the returned array:
0393      * array(
0394      *   <resTypeName> => array(
0395      *                      <resName> => <Zend_Pdf_Resource object>,
0396      *                      <resName> => <Zend_Pdf_Resource object>,
0397      *                      <resName> => <Zend_Pdf_Resource object>,
0398      *                      ...
0399      *                    ),
0400      *   <resTypeName> => array(
0401      *                      <resName> => <Zend_Pdf_Resource object>,
0402      *                      <resName> => <Zend_Pdf_Resource object>,
0403      *                      <resName> => <Zend_Pdf_Resource object>,
0404      *                      ...
0405      *                    ),
0406      *   ...
0407      *   'ProcSet' => array()
0408      * )
0409      *
0410      * where ProcSet array is a list of used procedure sets names (strings).
0411      * Allowed procedure set names: 'PDF', 'Text', 'ImageB', 'ImageC', 'ImageI'
0412      *
0413      * @internal
0414      * @return array
0415      */
0416     public function getResources()
0417     {
0418         $resources = array();
0419         $resDictionary = $this->_dictionary->Resources;
0420 
0421         foreach ($resDictionary->getKeys() as $resType) {
0422             $resources[$resType] = array();
0423 
0424             if ($resType == 'ProcSet') {
0425                 foreach ($resDictionary->ProcSet->items as $procSetEntry) {
0426                     $resources[$resType][] = $procSetEntry->value;
0427                 }
0428             } else {
0429                 $resMap = $resDictionary->$resType;
0430 
0431                 foreach ($resMap->getKeys() as $resId) {
0432                     $resources[$resType][$resId] =new Zend_Pdf_Resource_Unified($resMap->$resId);
0433                 }
0434             }
0435         }
0436 
0437         return $resources;
0438     }
0439 
0440     /**
0441      * Get drawing instructions stream
0442      *
0443      * It has to be returned as a PDF stream object to make it reusable.
0444      *
0445      * @internal
0446      * @returns Zend_Pdf_Resource_ContentStream
0447      */
0448     public function getContents()
0449     {
0450         /** @todo implementation */
0451     }
0452 
0453     /**
0454      * Return the height of this page in points.
0455      *
0456      * @return float
0457      */
0458     public function getHeight()
0459     {
0460         return $this->_dictionary->MediaBox->items[3]->value -
0461                $this->_dictionary->MediaBox->items[1]->value;
0462     }
0463 
0464     /**
0465      * Return the width of this page in points.
0466      *
0467      * @return float
0468      */
0469     public function getWidth()
0470     {
0471         return $this->_dictionary->MediaBox->items[2]->value -
0472                $this->_dictionary->MediaBox->items[0]->value;
0473     }
0474 
0475     /**
0476      * Clone page, extract it and dependent objects from the current document,
0477      * so it can be used within other docs.
0478      */
0479     public function __clone()
0480     {
0481         $factory = Zend_Pdf_ElementFactory::createFactory(1);
0482         $processed = array();
0483 
0484         // Clone dictionary object.
0485         // Do it explicitly to prevent sharing page attributes between different
0486         // results of clonePage() operation (other resources are still shared)
0487         $dictionary = new Zend_Pdf_Element_Dictionary();
0488         foreach ($this->_dictionary->getKeys() as $key) {
0489             $dictionary->$key = $this->_dictionary->$key->makeClone($factory->getFactory(),
0490                                                                         $processed,
0491                                                                         Zend_Pdf_Element::CLONE_MODE_SKIP_PAGES);
0492         }
0493 
0494         $this->_dictionary = $factory->newObject($dictionary);
0495         $this->_objFactory     = $factory;
0496         $this->_attached       = false;
0497         $this->_style          = null;
0498         $this->_font           = null;
0499     }
0500 
0501     /**
0502      * Clone page, extract it and dependent objects from the current document,
0503      * so it can be used within other docs.
0504      *
0505      * @internal
0506      * @param Zend_Pdf_ElementFactory_Interface $factory
0507      * @param array $processed
0508      * @return Zend_Pdf_Page
0509      */
0510     public function clonePage($factory, &$processed)
0511     {
0512         // Clone dictionary object.
0513         // Do it explicitly to prevent sharing page attributes between different
0514         // results of clonePage() operation (other resources are still shared)
0515         $dictionary = new Zend_Pdf_Element_Dictionary();
0516         foreach ($this->_dictionary->getKeys() as $key) {
0517             $dictionary->$key = $this->_dictionary->$key->makeClone($factory->getFactory(),
0518                                                                         $processed,
0519                                                                         Zend_Pdf_Element::CLONE_MODE_SKIP_PAGES);
0520         }
0521 
0522         $clonedPage = new Zend_Pdf_Page($factory->newObject($dictionary), $factory);
0523         $clonedPage->_attached = false;
0524 
0525         return $clonedPage;
0526     }
0527 
0528     /**
0529      * Retrive PDF file reference to the page
0530      *
0531      * @internal
0532      * @return Zend_Pdf_Element_Dictionary
0533      */
0534     public function getPageDictionary()
0535     {
0536         return $this->_dictionary;
0537     }
0538 
0539     /**
0540      * Dump current drawing instructions into the content stream.
0541      *
0542      * @todo Don't forget to close all current graphics operations (like path drawing)
0543      *
0544      * @throws Zend_Pdf_Exception
0545      */
0546     public function flush()
0547     {
0548         if ($this->_saveCount != 0) {
0549             // require_once 'Zend/Pdf/Exception.php';
0550             throw new Zend_Pdf_Exception('Saved graphics state is not restored');
0551         }
0552 
0553         if ($this->_contents == '') {
0554             return;
0555         }
0556 
0557         if ($this->_dictionary->Contents->getType() != Zend_Pdf_Element::TYPE_ARRAY) {
0558             /**
0559              * It's a stream object.
0560              * Prepare Contents page attribute for update.
0561              */
0562             $this->_dictionary->touch();
0563 
0564             $currentPageContents = $this->_dictionary->Contents;
0565             $this->_dictionary->Contents = new Zend_Pdf_Element_Array();
0566             $this->_dictionary->Contents->items[] = $currentPageContents;
0567         } else {
0568             $this->_dictionary->Contents->touch();
0569         }
0570 
0571         if ((!$this->_safeGS)  &&  (count($this->_dictionary->Contents->items) != 0)) {
0572             /**
0573              * Page already has some content which is not treated as safe.
0574              *
0575              * Add save/restore GS operators
0576              */
0577             $this->_addProcSet('PDF');
0578 
0579             $newContentsArray = new Zend_Pdf_Element_Array();
0580             $newContentsArray->items[] = $this->_objFactory->newStreamObject(" q\n");
0581             foreach ($this->_dictionary->Contents->items as $contentStream) {
0582                 $newContentsArray->items[] = $contentStream;
0583             }
0584             $newContentsArray->items[] = $this->_objFactory->newStreamObject(" Q\n");
0585 
0586             $this->_dictionary->touch();
0587             $this->_dictionary->Contents = $newContentsArray;
0588 
0589             $this->_safeGS = true;
0590         }
0591 
0592         $this->_dictionary->Contents->items[] =
0593                 $this->_objFactory->newStreamObject($this->_contents);
0594 
0595         $this->_contents = '';
0596     }
0597 
0598     /**
0599      * Prepare page to be rendered into PDF.
0600      *
0601      * @todo Don't forget to close all current graphics operations (like path drawing)
0602      *
0603      * @param Zend_Pdf_ElementFactory_Interface $objFactory
0604      * @throws Zend_Pdf_Exception
0605      */
0606     public function render(Zend_Pdf_ElementFactory_Interface $objFactory)
0607     {
0608         $this->flush();
0609 
0610         if ($objFactory === $this->_objFactory) {
0611             // Page is already attached to the document.
0612             return;
0613         }
0614 
0615         if ($this->_attached) {
0616             // require_once 'Zend/Pdf/Exception.php';
0617             throw new Zend_Pdf_Exception('Page is attached to other documen. Use clone $page to get it context free.');
0618         } else {
0619             $objFactory->attach($this->_objFactory);
0620         }
0621     }
0622 
0623     /**
0624      * Extract resources attached to the page
0625      *
0626      * This method is not intended to be used in userland, but helps to optimize some document wide operations
0627      *
0628      * returns array of Zend_Pdf_Element_Dictionary objects
0629      *
0630      * @internal
0631      * @return array
0632      */
0633     public function extractResources()
0634     {
0635         return $this->_dictionary->Resources;
0636     }
0637 
0638     /**
0639      * Extract fonts attached to the page
0640      *
0641      * returns array of Zend_Pdf_Resource_Font_Extracted objects
0642      *
0643      * @return array
0644      * @throws Zend_Pdf_Exception
0645      */
0646     public function extractFonts()
0647     {
0648         if ($this->_dictionary->Resources->Font === null) {
0649             // Page doesn't have any font attached
0650             // Return empty array
0651             return array();
0652         }
0653 
0654         $fontResources = $this->_dictionary->Resources->Font;
0655 
0656         $fontResourcesUnique = array();
0657         foreach ($fontResources->getKeys() as $fontResourceName) {
0658             $fontDictionary = $fontResources->$fontResourceName;
0659 
0660             if (! ($fontDictionary instanceof Zend_Pdf_Element_Reference  ||
0661                    $fontDictionary instanceof Zend_Pdf_Element_Object) ) {
0662                 // require_once 'Zend/Pdf/Exception.php';
0663                 throw new Zend_Pdf_Exception('Font dictionary has to be an indirect object or object reference.');
0664             }
0665 
0666             $fontResourcesUnique[spl_object_hash($fontDictionary->getObject())] = $fontDictionary;
0667         }
0668 
0669         $fonts = array();
0670         // require_once 'Zend/Pdf/Exception.php';
0671         foreach ($fontResourcesUnique as $resourceId => $fontDictionary) {
0672             try {
0673                 // require_once 'Zend/Pdf/Resource/Font/Extracted.php';
0674                 // Try to extract font
0675                 $extractedFont = new Zend_Pdf_Resource_Font_Extracted($fontDictionary);
0676 
0677                 $fonts[$resourceId] = $extractedFont;
0678             } catch (Zend_Pdf_Exception $e) {
0679                 if ($e->getMessage() != 'Unsupported font type.') {
0680                     throw new Zend_Pdf_Exception($e->getMessage(), $e->getCode(), $e);
0681                 }
0682             }
0683         }
0684 
0685         return $fonts;
0686     }
0687 
0688     /**
0689      * Extract font attached to the page by specific font name
0690      *
0691      * $fontName should be specified in UTF-8 encoding
0692      *
0693      * @return Zend_Pdf_Resource_Font_Extracted|null
0694      * @throws Zend_Pdf_Exception
0695      */
0696     public function extractFont($fontName)
0697     {
0698         if ($this->_dictionary->Resources->Font === null) {
0699             // Page doesn't have any font attached
0700             return null;
0701         }
0702 
0703         $fontResources = $this->_dictionary->Resources->Font;
0704 
0705         $fontResourcesUnique = array();
0706 
0707         // require_once 'Zend/Pdf/Exception.php';
0708         foreach ($fontResources->getKeys() as $fontResourceName) {
0709             $fontDictionary = $fontResources->$fontResourceName;
0710 
0711             if (! ($fontDictionary instanceof Zend_Pdf_Element_Reference  ||
0712                    $fontDictionary instanceof Zend_Pdf_Element_Object) ) {
0713                 // require_once 'Zend/Pdf/Exception.php';
0714                 throw new Zend_Pdf_Exception('Font dictionary has to be an indirect object or object reference.');
0715             }
0716 
0717             $resourceId = spl_object_hash($fontDictionary->getObject());
0718             if (isset($fontResourcesUnique[$resourceId])) {
0719                 continue;
0720             } else {
0721                 // Mark resource as processed
0722                 $fontResourcesUnique[$resourceId] = 1;
0723             }
0724 
0725             if ($fontDictionary->BaseFont->value != $fontName) {
0726                 continue;
0727             }
0728 
0729             try {
0730                 // Try to extract font
0731                 // require_once 'Zend/Pdf/Resource/Font/Extracted.php';
0732                 return new Zend_Pdf_Resource_Font_Extracted($fontDictionary);
0733             } catch (Zend_Pdf_Exception $e) {
0734                 if ($e->getMessage() != 'Unsupported font type.') {
0735                     throw new Zend_Pdf_Exception($e->getMessage(), $e->getCode(), $e);
0736                 }
0737 
0738                 // Continue searhing font with specified name
0739             }
0740         }
0741 
0742         return null;
0743     }
0744 
0745     /**
0746      *
0747      * @param Zend_Pdf_Annotation $annotation
0748      * @return Zend_Pdf_Page
0749      */
0750     public function attachAnnotation(Zend_Pdf_Annotation $annotation)
0751     {
0752         $annotationDictionary = $annotation->getResource();
0753         if (!$annotationDictionary instanceof Zend_Pdf_Element_Object  &&
0754             !$annotationDictionary instanceof Zend_Pdf_Element_Reference) {
0755             $annotationDictionary = $this->_objFactory->newObject($annotationDictionary);
0756         }
0757 
0758         if ($this->_dictionary->Annots === null) {
0759             $this->_dictionary->touch();
0760             $this->_dictionary->Annots = new Zend_Pdf_Element_Array();
0761         } else {
0762             $this->_dictionary->Annots->touch();
0763         }
0764 
0765         $this->_dictionary->Annots->items[] = $annotationDictionary;
0766 
0767         $annotationDictionary->touch();
0768         $annotationDictionary->P = $this->_dictionary;
0769 
0770         return $this;
0771     }
0772 }
0773