From f20e1e900f1eeacb33295573ef70efec0b3cd51b Mon Sep 17 00:00:00 2001 From: Progi1984 Date: Thu, 17 Mar 2016 09:53:45 +0100 Subject: [PATCH] #192 : Image Adapter (Feature) --- samples/Sample_01_Simple.php | 52 +++--- samples/Sample_03_Image.php | 27 ++- samples/Sample_Header.php | 16 +- samples/resources/SamplePassword.pptx | Bin 0 -> 37376 bytes src/PhpPresentation/AbstractShape.php | 2 +- src/PhpPresentation/Reader/ODPresentation.php | 13 +- src/PhpPresentation/Reader/PowerPoint2007.php | 4 +- src/PhpPresentation/Reader/PowerPoint97.php | 3 +- ...bstractDrawing.php => AbstractGraphic.php} | 2 +- src/PhpPresentation/Shape/Chart.php | 5 +- src/PhpPresentation/Shape/Drawing.php | 40 +---- .../Shape/Drawing/AbstractDrawingAdapter.php | 28 +++ src/PhpPresentation/Shape/Drawing/Base64.php | 101 +++++++++++ src/PhpPresentation/Shape/Drawing/File.php | 85 ++++++++++ src/PhpPresentation/Shape/Drawing/Gd.php | 159 ++++++++++++++++++ src/PhpPresentation/Shape/Drawing/ZipFile.php | 97 +++++++++++ src/PhpPresentation/Shape/Group.php | 4 +- src/PhpPresentation/Shape/Table.php | 3 +- .../{MemoryDrawing.php => _MemoryDrawing.php} | 10 +- src/PhpPresentation/Slide.php | 4 +- src/PhpPresentation/Writer/AbstractWriter.php | 41 +++++ src/PhpPresentation/Writer/ODPresentation.php | 36 ---- .../Writer/ODPresentation/Content.php | 15 +- .../Writer/ODPresentation/MetaInfManifest.php | 31 ++-- .../Writer/ODPresentation/Pictures.php | 45 +---- src/PhpPresentation/Writer/PowerPoint2007.php | 39 +---- .../AbstractDecoratorWriter.php | 10 +- .../Writer/PowerPoint2007/ContentTypes.php | 57 +------ .../Writer/PowerPoint2007/PptMedia.php | 30 +--- .../Writer/PowerPoint2007/PptSlides.php | 52 +++--- 30 files changed, 654 insertions(+), 357 deletions(-) create mode 100644 samples/resources/SamplePassword.pptx rename src/PhpPresentation/Shape/{AbstractDrawing.php => AbstractGraphic.php} (98%) create mode 100644 src/PhpPresentation/Shape/Drawing/AbstractDrawingAdapter.php create mode 100644 src/PhpPresentation/Shape/Drawing/Base64.php create mode 100644 src/PhpPresentation/Shape/Drawing/File.php create mode 100644 src/PhpPresentation/Shape/Drawing/Gd.php create mode 100644 src/PhpPresentation/Shape/Drawing/ZipFile.php rename src/PhpPresentation/Shape/{MemoryDrawing.php => _MemoryDrawing.php} (92%) diff --git a/samples/Sample_01_Simple.php b/samples/Sample_01_Simple.php index c99bb867b..e7ebd0b9a 100644 --- a/samples/Sample_01_Simple.php +++ b/samples/Sample_01_Simple.php @@ -11,48 +11,48 @@ $objPHPPresentation = new PhpPresentation(); // Set properties -echo date('H:i:s') . ' Set properties'.EOL; -$objPHPPresentation->getProperties()->setCreator('PHPOffice') - ->setLastModifiedBy('PHPPresentation Team') - ->setTitle('Sample 01 Title') - ->setSubject('Sample 01 Subject') - ->setDescription('Sample 01 Description') - ->setKeywords('office 2007 openxml libreoffice odt php') - ->setCategory('Sample Category'); +echo date('H:i:s') . ' Set properties' . EOL; +$objPHPPresentation->getDocumentProperties()->setCreator('PHPOffice') + ->setLastModifiedBy('PHPPresentation Team') + ->setTitle('Sample 01 Title') + ->setSubject('Sample 01 Subject') + ->setDescription('Sample 01 Description') + ->setKeywords('office 2007 openxml libreoffice odt php') + ->setCategory('Sample Category'); // Create slide -echo date('H:i:s') . ' Create slide'.EOL; +echo date('H:i:s') . ' Create slide' . EOL; $currentSlide = $objPHPPresentation->getActiveSlide(); // Create a shape (drawing) -echo date('H:i:s') . ' Create a shape (drawing)'.EOL; +echo date('H:i:s') . ' Create a shape (drawing)' . EOL; $shape = $currentSlide->createDrawingShape(); $shape->setName('PHPPresentation logo') - ->setDescription('PHPPresentation logo') - ->setPath('./resources/phppowerpoint_logo.gif') - ->setHeight(36) - ->setOffsetX(10) - ->setOffsetY(10); + ->setDescription('PHPPresentation logo') + ->setPath('./resources/phppowerpoint_logo.gif') + ->setHeight(36) + ->setOffsetX(10) + ->setOffsetY(10); $shape->getShadow()->setVisible(true) - ->setDirection(45) - ->setDistance(10); + ->setDirection(45) + ->setDistance(10); $shape->getHyperlink()->setUrl('https://github.com/PHPOffice/PHPPresentation/')->setTooltip('PHPPresentation'); // Create a shape (text) -echo date('H:i:s') . ' Create a shape (rich text)'.EOL; +echo date('H:i:s') . ' Create a shape (rich text)' . EOL; $shape = $currentSlide->createRichTextShape() - ->setHeight(300) - ->setWidth(600) - ->setOffsetX(170) - ->setOffsetY(180); -$shape->getActiveParagraph()->getAlignment()->setHorizontal( Alignment::HORIZONTAL_CENTER ); + ->setHeight(300) + ->setWidth(600) + ->setOffsetX(170) + ->setOffsetY(180); +$shape->getActiveParagraph()->getAlignment()->setHorizontal(Alignment::HORIZONTAL_CENTER); $textRun = $shape->createTextRun('Thank you for using PHPPresentation!'); $textRun->getFont()->setBold(true) - ->setSize(60) - ->setColor( new Color( 'FFE06B20' ) ); + ->setSize(60) + ->setColor(new Color('FFE06B20')); // Save file echo write($objPHPPresentation, basename(__FILE__, '.php'), $writers); if (!CLI) { - include_once 'Sample_Footer.php'; + include_once 'Sample_Footer.php'; } diff --git a/samples/Sample_03_Image.php b/samples/Sample_03_Image.php index 5bdca9863..d06902376 100644 --- a/samples/Sample_03_Image.php +++ b/samples/Sample_03_Image.php @@ -4,7 +4,6 @@ use PhpOffice\PhpPresentation\PhpPresentation; use PhpOffice\PhpPresentation\Shape\Drawing; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; // Create new PHPPresentation object echo date('H:i:s') . ' Create new PHPPresentation object'.EOL; @@ -22,19 +21,19 @@ // Add a generated drawing to the slide echo date('H:i:s') . ' Add a drawing to the slide'.EOL; -$shape = new MemoryDrawing(); +$shape = new Drawing\Gd(); $shape->setName('Sample image') ->setDescription('Sample image') ->setImageResource($gdImage) - ->setRenderingFunction(MemoryDrawing::RENDERING_JPEG) - ->setMimeType(MemoryDrawing::MIMETYPE_DEFAULT) + ->setRenderingFunction(Drawing\Gd::RENDERING_JPEG) + ->setMimeType(Drawing\Gd::MIMETYPE_DEFAULT) ->setHeight(36) ->setOffsetX(10) ->setOffsetY(10); $currentSlide->addShape($shape); // Add a file drawing (GIF) to the slide -$shape = new Drawing(); +$shape = new Drawing\File(); $shape->setName('PHPPresentation logo') ->setDescription('PHPPresentation logo') ->setPath('./resources/phppowerpoint_logo.gif') @@ -43,11 +42,23 @@ ->setOffsetY(100); $currentSlide->addShape($shape); -// Add a file drawing (GIF) to the slide -$shape = new Drawing(); +// Add a file drawing (Zip) to the slide +$shape = new Drawing\ZipFile(); +$shape->setName('PHPPresentation logo') + ->setDescription('PHPPresentation logo') + ->setPath('zip://./resources/Sample_12.pptx#ppt/media/phppowerpoint_logo1.gif') + ->setResizeProportional(false) + ->setHeight(36) + ->setWidth(36) + ->setOffsetX(10) + ->setOffsetY(150); +$currentSlide->addShape($shape); + +// Add a file drawing (JPEG) to the slide +$shape = new Drawing\Base64(); $shape->setName('PHPPresentation logo') ->setDescription('PHPPresentation logo') - ->setPath('', false) + ->setData('', false) ->setResizeProportional(false) ->setHeight(36) ->setWidth(36) diff --git a/samples/Sample_Header.php b/samples/Sample_Header.php index 3d42577f0..92eba86f4 100644 --- a/samples/Sample_Header.php +++ b/samples/Sample_Header.php @@ -43,7 +43,7 @@ $pageTitle .= 'PHPPresentation'; $pageHeading = IS_INDEX ? '' : "

{$pageHeading}

"; -$oShapeDrawing = new Drawing(); +$oShapeDrawing = new Drawing\File(); $oShapeDrawing->setName('PHPPresentation logo') ->setDescription('PHPPresentation logo') ->setPath('./resources/phppowerpoint_logo.gif') @@ -241,14 +241,18 @@ protected function displayPhpPresentation(PhpPresentation $oPHPPpt) protected function displayShape(AbstractShape $shape) { - if($shape instanceof MemoryDrawing) { - $this->append('
  • Shape "MemoryDrawing"
  • '); - } elseif($shape instanceof Drawing) { - $this->append('
  • Shape "Drawing"
  • '); + if($shape instanceof Drawing\Gd) { + $this->append('
  • Shape "Drawing\Gd"
  • '); + } elseif($shape instanceof Drawing\File) { + $this->append('
  • Shape "Drawing\File"
  • '); + } elseif($shape instanceof Drawing\Base64) { + $this->append('
  • Shape "Drawing\Base64"
  • '); + } elseif($shape instanceof Drawing\Zip) { + $this->append('
  • Shape "Drawing\Zip"
  • '); } elseif($shape instanceof RichText) { $this->append('
  • Shape "RichText"
  • '); } else { - var_export($shape); + var_dump($shape); } } diff --git a/samples/resources/SamplePassword.pptx b/samples/resources/SamplePassword.pptx new file mode 100644 index 0000000000000000000000000000000000000000..5065dc466e7fbeab0e59a01fa318d0c44e7e50c9 GIT binary patch literal 37376 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:'') . $this->offsetX . $this->offsetY . $this->width . $this->height . $this->rotation . $this->getFill()->getHashCode() . (is_null($this->shadow) ? '' : $this->shadow->getHashCode()) . (is_null($this->hyperlink) ? '' : $this->hyperlink->getHashCode()) . __CLASS__); + return md5((is_object($this->container)?$this->container->getHashCode():'') . $this->offsetX . $this->offsetY . $this->width . $this->height . $this->rotation . (is_null($this->getFill()) ? '' : $this->getFill()->getHashCode()) . (is_null($this->shadow) ? '' : $this->shadow->getHashCode()) . (is_null($this->hyperlink) ? '' : $this->hyperlink->getHashCode()) . __CLASS__); } /** diff --git a/src/PhpPresentation/Reader/ODPresentation.php b/src/PhpPresentation/Reader/ODPresentation.php index 56bed465b..514582914 100644 --- a/src/PhpPresentation/Reader/ODPresentation.php +++ b/src/PhpPresentation/Reader/ODPresentation.php @@ -21,17 +21,14 @@ use PhpOffice\Common\XMLReader; use PhpOffice\Common\Drawing as CommonDrawing; use PhpOffice\PhpPresentation\PhpPresentation; -use PhpOffice\PhpPresentation\Shape\Hyperlink; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; +use PhpOffice\PhpPresentation\Shape\Drawing\Gd; +use PhpOffice\PhpPresentation\Shape\RichText; +use PhpOffice\PhpPresentation\Shape\RichText\Paragraph; use PhpOffice\PhpPresentation\Style\Bullet; use PhpOffice\PhpPresentation\Style\Color; use PhpOffice\PhpPresentation\Style\Font; use PhpOffice\PhpPresentation\Style\Shadow; use PhpOffice\PhpPresentation\Style\Alignment; -use PhpOffice\PhpPresentation\Style\PhpOffice\PhpPresentation\Style; -use PhpOffice\PhpPresentation\Shape\RichText; -use PhpOffice\PhpPresentation\Shape\RichText\Paragraph; -use PhpOffice\Common\PhpOffice\Common; /** * Serialized format reader @@ -365,9 +362,9 @@ protected function loadSlide(\DOMElement $nodeSlide) protected function loadShapeDrawing(\DOMElement $oNodeFrame) { // Core - $oShape = new MemoryDrawing(); + $oShape = new Gd(); $oShape->getShadow()->setVisible(false); - + $oNodeImage = $this->oXMLReader->getElement('draw:image', $oNodeFrame); if ($oNodeImage) { if ($oNodeImage->hasAttribute('xlink:href')) { diff --git a/src/PhpPresentation/Reader/PowerPoint2007.php b/src/PhpPresentation/Reader/PowerPoint2007.php index ffead7ec5..7ab5d31d8 100644 --- a/src/PhpPresentation/Reader/PowerPoint2007.php +++ b/src/PhpPresentation/Reader/PowerPoint2007.php @@ -17,11 +17,11 @@ namespace PhpOffice\PhpPresentation\Reader; +use PhpOffice\PhpPresentation\Shape\Drawing\Gd; use ZipArchive; use PhpOffice\Common\XMLReader; use PhpOffice\Common\Drawing as CommonDrawing; use PhpOffice\PhpPresentation\PhpPresentation; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; use PhpOffice\PhpPresentation\Style\Bullet; use PhpOffice\PhpPresentation\Style\Color; use PhpOffice\PhpPresentation\Writer\PowerPoint2007\LayoutPack\TemplateBased; @@ -328,7 +328,7 @@ protected function loadSlide($sPart, $baseFile) protected function loadShapeDrawing(XMLReader $document, \DOMElement $node, $baseFile) { // Core - $oShape = new MemoryDrawing(); + $oShape = new Gd(); $oShape->getShadow()->setVisible(false); // Variables $fileRels = 'ppt/slides/_rels/'.$baseFile.'.rels'; diff --git a/src/PhpPresentation/Reader/PowerPoint97.php b/src/PhpPresentation/Reader/PowerPoint97.php index c2215e622..4a5d03d87 100644 --- a/src/PhpPresentation/Reader/PowerPoint97.php +++ b/src/PhpPresentation/Reader/PowerPoint97.php @@ -20,7 +20,6 @@ use PhpOffice\Common\Microsoft\OLERead; use PhpOffice\PhpPresentation\Shape\Drawing; use PhpOffice\PhpPresentation\PhpPresentation; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; use PhpOffice\PhpPresentation\Style\Alignment; use PhpOffice\PhpPresentation\Style\Color; use PhpOffice\PhpPresentation\Shape\RichText; @@ -1422,7 +1421,7 @@ private function readRecordOfficeArtSpContainer($stream, $pos) $drawingPib = $shpPrimaryOptions['pib']; if (isset($this->arrayPictures[$drawingPib - 1])) { $gdImage = imagecreatefromstring($this->arrayPictures[$drawingPib - 1]); - $arrayReturn['shape'] = new MemoryDrawing(); + $arrayReturn['shape'] = new Drawing\Gd(); $arrayReturn['shape']->setImageResource($gdImage); } } elseif (isset($shpPrimaryOptions['line']) && $shpPrimaryOptions['line']) { diff --git a/src/PhpPresentation/Shape/AbstractDrawing.php b/src/PhpPresentation/Shape/AbstractGraphic.php similarity index 98% rename from src/PhpPresentation/Shape/AbstractDrawing.php rename to src/PhpPresentation/Shape/AbstractGraphic.php index 1a11119e9..2c0b8fba6 100644 --- a/src/PhpPresentation/Shape/AbstractDrawing.php +++ b/src/PhpPresentation/Shape/AbstractGraphic.php @@ -23,7 +23,7 @@ /** * Abstract drawing */ -abstract class AbstractDrawing extends AbstractShape implements ComparableInterface +abstract class AbstractGraphic extends AbstractShape implements ComparableInterface { /** * Image counter diff --git a/src/PhpPresentation/Shape/Chart.php b/src/PhpPresentation/Shape/Chart.php index 73a7ed690..1849b2de5 100644 --- a/src/PhpPresentation/Shape/Chart.php +++ b/src/PhpPresentation/Shape/Chart.php @@ -17,6 +17,7 @@ namespace PhpOffice\PhpPresentation\Shape; +use PhpOffice\PhpPresentation\AbstractShape; use PhpOffice\PhpPresentation\ComparableInterface; use PhpOffice\PhpPresentation\Shape\Chart\Legend; use PhpOffice\PhpPresentation\Shape\Chart\PlotArea; @@ -26,7 +27,7 @@ /** * Chart element */ -class Chart extends AbstractDrawing implements ComparableInterface +class Chart extends AbstractGraphic implements ComparableInterface { /** * Title @@ -64,7 +65,7 @@ class Chart extends AbstractDrawing implements ComparableInterface private $includeSpreadsheet = false; /** - * Create a new \PhpOffice\PhpPresentation\Slide\MemoryDrawing + * Create a new Chart */ public function __construct() { diff --git a/src/PhpPresentation/Shape/Drawing.php b/src/PhpPresentation/Shape/Drawing.php index 91651bf23..f7ec791ab 100644 --- a/src/PhpPresentation/Shape/Drawing.php +++ b/src/PhpPresentation/Shape/Drawing.php @@ -22,7 +22,7 @@ /** * Drawing element */ -class Drawing extends AbstractDrawing implements ComparableInterface +class Drawing extends AbstractGraphic implements ComparableInterface { /** * Path @@ -75,44 +75,6 @@ public function getExtension() return $exploded[count($exploded) - 1]; } - /** - * Get Path - * - * @return string - */ - public function getPath() - { - return $this->path; - } - - /** - * Set Path - * - * @param string $pValue File path - * @param boolean $pVerifyFile Verify file - * @throws \Exception - * @return \PhpOffice\PhpPresentation\Shape\Drawing - */ - public function setPath($pValue = '', $pVerifyFile = true) - { - if ($pVerifyFile) { - if (file_exists($pValue)) { - $this->path = $pValue; - - if ($this->width == 0 && $this->height == 0) { - // Get width/height - list($this->width, $this->height) = getimagesize($pValue); - } - } else { - throw new \Exception("File $pValue not found!"); - } - } else { - $this->path = $pValue; - } - - return $this; - } - /** * Get hash code * diff --git a/src/PhpPresentation/Shape/Drawing/AbstractDrawingAdapter.php b/src/PhpPresentation/Shape/Drawing/AbstractDrawingAdapter.php new file mode 100644 index 000000000..ee47e1213 --- /dev/null +++ b/src/PhpPresentation/Shape/Drawing/AbstractDrawingAdapter.php @@ -0,0 +1,28 @@ + 'jpg', + ); + + /** + * Base64 constructor. + */ + public function __construct() + { + parent::__construct(); + $this->uniqueName = md5(rand(0, 9999) . time() . rand(0, 9999)); + } + + /** + * @return mixed + */ + public function getData() + { + return $this->data; + } + + /** + * @param mixed $data + * @return Base64 + */ + public function setData($data) + { + $this->data = $data; + return $this; + } + + /** + * @return string + */ + public function getContents() + { + list(, $imageContents) = explode(';', $this->getData()); + list(, $imageContents) = explode(',', $imageContents); + return base64_decode($imageContents); + } + + /** + * @return string + */ + public function getExtension() + { + list($data, ) = explode(';', $this->getData()); + list(, $mime) = explode(':', $data); + + if (!array_key_exists($mime, $this->arrayMimeExtension)) { + throw new \Exception('Type Mime not found : "'.$mime.'"'); + } + return $this->arrayMimeExtension[$mime]; + } + + /** + * @return string + */ + public function getIndexedFilename() + { + return $this->uniqueName . $this->getImageIndex() . '.' . $this->getExtension(); + } + + /** + * @return string + */ + public function getMimeType() + { + list(, $sImage) = explode(';', $this->getData()); + list(, $sImage) = explode(',', $sImage); + if (!function_exists('getimagesizefromstring')) { + $uri = 'data://application/octet-stream;base64,' . base64_encode($sImage); + $image = getimagesize($uri); + } else { + $image = getimagesizefromstring($sImage); + } + image_type_to_mime_type($image[2]); + } +} \ No newline at end of file diff --git a/src/PhpPresentation/Shape/Drawing/File.php b/src/PhpPresentation/Shape/Drawing/File.php new file mode 100644 index 000000000..8c76b2d62 --- /dev/null +++ b/src/PhpPresentation/Shape/Drawing/File.php @@ -0,0 +1,85 @@ +oFileInfo->getPathname(); + } + + /** + * Set Path + * + * @param string $pValue File path + * @param boolean $pVerifyFile Verify file + * @throws \Exception + * @return \PhpOffice\PhpPresentation\Shape\Drawing + */ + public function setPath($pValue = '', $pVerifyFile = true) + { + if ($pVerifyFile) { + if (!file_exists($pValue)) { + throw new \Exception("File $pValue not found!"); + } + } + $this->oFileInfo = new SplFileInfo($pValue); + + if ($this->width == 0 && $this->height == 0) { + list($this->width, $this->height) = getimagesize($this->getPath()); + } + + return $this; + } + + /** + * @return string + */ + public function getContents() + { + return file_get_contents($this->getPath()); + } + + + /** + * @return string + */ + public function getExtension() + { + return $this->oFileInfo->getExtension(); + } + + /** + * @return string + */ + public function getMimeType() + { + $image = getimagesize($this->getPath()); + return image_type_to_mime_type($image[2]); + } + + /** + * @return string + */ + public function getIndexedFilename() + { + $output = str_replace('.' . $this->getExtension(), '', $this->oFileInfo->getFilename()); + $output .= $this->getImageIndex(); + $output .= $this->getExtension(); + $output = str_replace(' ', '_', $output); + return $output; + } +} \ No newline at end of file diff --git a/src/PhpPresentation/Shape/Drawing/Gd.php b/src/PhpPresentation/Shape/Drawing/Gd.php new file mode 100644 index 000000000..cb4cdcbad --- /dev/null +++ b/src/PhpPresentation/Shape/Drawing/Gd.php @@ -0,0 +1,159 @@ +uniqueName = md5(rand(0, 9999) . time() . rand(0, 9999)); + } + + /** + * Get image resource + * + * @return resource + */ + public function getImageResource() + { + return $this->imageResource; + } + + /** + * Set image resource + * + * @param $value resource + * @return $this + */ + public function setImageResource($value = null) + { + $this->imageResource = $value; + + if (!is_null($this->imageResource)) { + // Get width/height + $this->width = imagesx($this->imageResource); + $this->height = imagesy($this->imageResource); + } + + return $this; + } + + /** + * Get rendering function + * + * @return string + */ + public function getRenderingFunction() + { + return $this->renderingFunction; + } + + /** + * Set rendering function + * + * @param string $value + * @return $this + */ + public function setRenderingFunction($value = self::RENDERING_DEFAULT) + { + $this->renderingFunction = $value; + return $this; + } + + /** + * Get mime type + * + * @return string + */ + public function getMimeType() + { + return $this->mimeType; + } + + /** + * Set mime type + * + * @param string $value + * @return $this + */ + public function setMimeType($value = self::MIMETYPE_DEFAULT) + { + $this->mimeType = $value; + return $this; + } + + /** + * @return string + */ + public function getContents() + { + ob_start(); + call_user_func($this->getRenderingFunction(), $this->getImageResource()); + $imageContents = ob_get_contents(); + ob_end_clean(); + return $imageContents; + } + + /** + * @return string + */ + public function getExtension() + { + $extension = strtolower($this->getMimeType()); + $extension = explode('/', $extension); + $extension = $extension[1]; + return $extension; + } + + /** + * @return string + */ + public function getIndexedFilename() + { + return $this->uniqueName . $this->getImageIndex() . '.' . $this->getExtension(); + } +} \ No newline at end of file diff --git a/src/PhpPresentation/Shape/Drawing/ZipFile.php b/src/PhpPresentation/Shape/Drawing/ZipFile.php new file mode 100644 index 000000000..94b5c3e6a --- /dev/null +++ b/src/PhpPresentation/Shape/Drawing/ZipFile.php @@ -0,0 +1,97 @@ +path; + } + + /** + * Set Path + * + * @param string $pValue File path + * @return \PhpOffice\PhpPresentation\Shape\Drawing + */ + public function setPath($pValue = '') + { + $this->path = $pValue; + return $this; + } + + /** + * @return string + */ + public function getContents() + { + $imagePath = substr($this->getPath(), 6); + $imagePathSplitted = explode('#', $imagePath); + + $imageZip = new \ZipArchive(); + $imageZip->open($imagePathSplitted[0]); + $imageContents = $imageZip->getFromName($imagePathSplitted[1]); + $imageZip->close(); + unset($imageZip); + return $imageContents; + } + + + /** + * @return string + */ + public function getExtension() + { + $imagePath = substr($this->getPath(), 6); + $imagePathSplitted = explode('#', $imagePath); + return pathinfo($imagePathSplitted[1], PATHINFO_EXTENSION); + } + + /** + * @return string + */ + public function getMimeType() + { + $pZIPFile = str_replace('zip://', '', $this->getPath()); + $pZIPFile = substr($pZIPFile, 0, strpos($pZIPFile, '#')); + if (!CommonFile::fileExists($pZIPFile)) { + throw new \Exception("File $pZIPFile does not exist"); + } + $pImgFile = substr($this->getPath(), strpos($this->getPath(), '#') + 1); + $oArchive = new \ZipArchive(); + $oArchive->open($pZIPFile); + if (!function_exists('getimagesizefromstring')) { + $uri = 'data://application/octet-stream;base64,' . base64_encode($oArchive->getFromName($pImgFile)); + $image = getimagesize($uri); + } else { + $image = getimagesizefromstring($oArchive->getFromName($pImgFile)); + } + return image_type_to_mime_type($image[2]); + } + + /** + * @return string + */ + public function getIndexedFilename() + { + $output = substr($this->getPath(), strpos($this->getPath(), '#') + 1); + $output = str_replace('.' . $this->getExtension(), '', $output); + $output .= $this->getImageIndex(); + $output .= $this->getExtension(); + $output = str_replace(' ', '_', $output); + return $output; + } +} \ No newline at end of file diff --git a/src/PhpPresentation/Shape/Group.php b/src/PhpPresentation/Shape/Group.php index 02184d1b1..f12242cb6 100644 --- a/src/PhpPresentation/Shape/Group.php +++ b/src/PhpPresentation/Shape/Group.php @@ -237,11 +237,11 @@ public function createChartShape() /** * Create drawing shape * - * @return \PhpOffice\PhpPresentation\Shape\Drawing + * @return \PhpOffice\PhpPresentation\Shape\Drawing\File */ public function createDrawingShape() { - $shape = new Drawing(); + $shape = new Drawing\File(); $this->addShape($shape); return $shape; diff --git a/src/PhpPresentation/Shape/Table.php b/src/PhpPresentation/Shape/Table.php index 0e056e6a7..524fd2c5a 100644 --- a/src/PhpPresentation/Shape/Table.php +++ b/src/PhpPresentation/Shape/Table.php @@ -17,13 +17,14 @@ namespace PhpOffice\PhpPresentation\Shape; +use PhpOffice\PhpPresentation\AbstractShape; use PhpOffice\PhpPresentation\ComparableInterface; use PhpOffice\PhpPresentation\Shape\Table\Row; /** * Table shape */ -class Table extends AbstractDrawing implements ComparableInterface +class Table extends AbstractGraphic implements ComparableInterface { /** * Rows diff --git a/src/PhpPresentation/Shape/MemoryDrawing.php b/src/PhpPresentation/Shape/_MemoryDrawing.php similarity index 92% rename from src/PhpPresentation/Shape/MemoryDrawing.php rename to src/PhpPresentation/Shape/_MemoryDrawing.php index b1ca2546a..c559155d7 100644 --- a/src/PhpPresentation/Shape/MemoryDrawing.php +++ b/src/PhpPresentation/Shape/_MemoryDrawing.php @@ -22,7 +22,7 @@ /** * Memory drawing shape */ -class MemoryDrawing extends AbstractDrawing implements ComparableInterface +class MemoryDrawing extends AbstractGraphic implements ComparableInterface { /* Rendering functions */ const RENDERING_DEFAULT = 'imagepng'; @@ -65,7 +65,7 @@ class MemoryDrawing extends AbstractDrawing implements ComparableInterface private $uniqueName; /** - * Create a new \PhpOffice\PhpPresentation\Slide\MemoryDrawing + * Create a new MemoryDrawing */ public function __construct() { @@ -93,7 +93,7 @@ public function getImageResource() * Set image resource * * @param $value resource - * @return \PhpOffice\PhpPresentation\Shape\MemoryDrawing + * @return MemoryDrawing */ public function setImageResource($value = null) { @@ -122,7 +122,7 @@ public function getRenderingFunction() * Set rendering function * * @param string $value - * @return \PhpOffice\PhpPresentation\Shape\MemoryDrawing + * @return MemoryDrawing */ public function setRenderingFunction($value = self::RENDERING_DEFAULT) { @@ -145,7 +145,7 @@ public function getMimeType() * Set mime type * * @param string $value - * @return \PhpOffice\PhpPresentation\Shape\MemoryDrawing + * @return MemoryDrawing */ public function setMimeType($value = self::MIMETYPE_DEFAULT) { diff --git a/src/PhpPresentation/Slide.php b/src/PhpPresentation/Slide.php index 76ea01944..a603e08b8 100644 --- a/src/PhpPresentation/Slide.php +++ b/src/PhpPresentation/Slide.php @@ -217,11 +217,11 @@ public function createChartShape() /** * Create drawing shape * - * @return \PhpOffice\PhpPresentation\Shape\Drawing + * @return \PhpOffice\PhpPresentation\Shape\Drawing\File */ public function createDrawingShape() { - $shape = new Drawing(); + $shape = new Drawing\File(); $this->addShape($shape); return $shape; diff --git a/src/PhpPresentation/Writer/AbstractWriter.php b/src/PhpPresentation/Writer/AbstractWriter.php index 2052dd747..02bb4bede 100644 --- a/src/PhpPresentation/Writer/AbstractWriter.php +++ b/src/PhpPresentation/Writer/AbstractWriter.php @@ -4,6 +4,8 @@ use PhpOffice\Common\Adapter\Zip\ZipInterface; use PhpOffice\PhpPresentation\PhpPresentation; +use PhpOffice\PhpPresentation\Shape\Drawing\AbstractDrawingAdapter; +use PhpOffice\PhpPresentation\Shape\Group; abstract class AbstractWriter { @@ -81,4 +83,43 @@ public function getZipAdapter() { return $this->oZipAdapter; } + + /** + * Get an array of all drawings + * + * @return \PhpOffice\PhpPresentation\Shape\AbstractDrawing[] All drawings in PhpPresentation + * @throws \Exception + */ + protected function allDrawings() + { + // Get an array of all drawings + $aDrawings = array(); + + // Loop trough PhpPresentation + foreach ($this->getPhpPresentation()->getAllSlides() as $oSlide) { + $arrayReturn = $this->iterateCollection($oSlide->getShapeCollection()->getIterator()); + $aDrawings = array_merge($aDrawings, $arrayReturn); + } + return $aDrawings; + } + + private function iterateCollection(\ArrayIterator $oIterator) + { + $arrayReturn = array(); + if ($oIterator->count() <= 0) { + return $arrayReturn; + } + + while ($oIterator->valid()) { + $oShape = $oIterator->current(); + if ($oShape instanceof AbstractDrawingAdapter) { + $arrayReturn[] = $oShape; + } elseif ($oShape instanceof Group) { + $arrayGroup = $this->iterateCollection($oShape->getShapeCollection()->getIterator()); + $arrayReturn = array_merge($arrayReturn, $arrayGroup); + } + $oIterator->next(); + } + return $arrayReturn; + } } diff --git a/src/PhpPresentation/Writer/ODPresentation.php b/src/PhpPresentation/Writer/ODPresentation.php index f9089fa88..9c14f3188 100644 --- a/src/PhpPresentation/Writer/ODPresentation.php +++ b/src/PhpPresentation/Writer/ODPresentation.php @@ -176,40 +176,4 @@ public function getDiskCachingDirectory() { return $this->diskCachingDirectory; } - - /** - * Get an array of all drawings - * - * @return \PhpOffice\PhpPresentation\Shape\AbstractDrawing[] All drawings in PhpPresentation - * @throws \Exception - */ - protected function allDrawings() - { - // Get an array of all drawings - $aDrawings = array(); - - // Loop trough PhpPresentation - $slideCount = $this->getPhpPresentation()->getSlideCount(); - for ($i = 0; $i < $slideCount; ++$i) { - // Loop trough images and add to array - $iterator = $this->getPhpPresentation()->getSlide($i)->getShapeCollection()->getIterator(); - while ($iterator->valid()) { - if ($iterator->current() instanceof AbstractDrawing && !($iterator->current() instanceof Table)) { - $aDrawings[] = $iterator->current(); - } elseif ($iterator->current() instanceof Group) { - $iterator2 = $iterator->current()->getShapeCollection()->getIterator(); - while ($iterator2->valid()) { - if ($iterator2->current() instanceof AbstractDrawing && !($iterator2->current() instanceof Table)) { - $aDrawings[] = $iterator2->current(); - } - $iterator2->next(); - } - } - - $iterator->next(); - } - } - - return $aDrawings; - } } diff --git a/src/PhpPresentation/Writer/ODPresentation/Content.php b/src/PhpPresentation/Writer/ODPresentation/Content.php index 13bc9b6a1..f5c665b6f 100644 --- a/src/PhpPresentation/Writer/ODPresentation/Content.php +++ b/src/PhpPresentation/Writer/ODPresentation/Content.php @@ -13,7 +13,6 @@ use PhpOffice\PhpPresentation\Shape\Drawing as ShapeDrawing; use PhpOffice\PhpPresentation\Shape\Group; use PhpOffice\PhpPresentation\Shape\Line; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; use PhpOffice\PhpPresentation\Shape\RichText\BreakElement; use PhpOffice\PhpPresentation\Shape\RichText\Run; use PhpOffice\PhpPresentation\Shape\RichText\TextElement; @@ -282,8 +281,8 @@ public function writeContent() $this->writeShapeLine($objWriter, $shape); } elseif ($shape instanceof Chart) { $this->writeShapeChart($objWriter, $shape); - } elseif ($shape instanceof AbstractDrawing) { - $this->writeShapePic($objWriter, $shape); + } elseif ($shape instanceof ShapeDrawing\AbstractDrawingAdapter) { + $this->writeShapeDrawing($objWriter, $shape); } elseif ($shape instanceof Group) { $this->writeShapeGroup($objWriter, $shape); } elseif ($shape instanceof Comment) { @@ -322,7 +321,7 @@ public function writeContent() * @param \PhpOffice\Common\XMLWriter $objWriter * @param \PhpOffice\PhpPresentation\Shape\AbstractDrawing $shape */ - public function writeShapePic(XMLWriter $objWriter, AbstractDrawing $shape) + public function writeShapeDrawing(XMLWriter $objWriter, ShapeDrawing\AbstractDrawingAdapter $shape) { // draw:frame $objWriter->startElement('draw:frame'); @@ -334,9 +333,7 @@ public function writeShapePic(XMLWriter $objWriter, AbstractDrawing $shape) $objWriter->writeAttribute('draw:style-name', 'gr' . $this->shapeId); // draw:image $objWriter->startElement('draw:image'); - if ($shape instanceof ShapeDrawing) { - $objWriter->writeAttribute('xlink:href', 'Pictures/' . md5($shape->getPath()) . '.' . $shape->getExtension()); - } elseif ($shape instanceof MemoryDrawing) { + if ($shape instanceof ShapeDrawing\AbstractDrawingAdapter) { $objWriter->writeAttribute('xlink:href', 'Pictures/' . $shape->getIndexedFilename()); } $objWriter->writeAttribute('xlink:type', 'simple'); @@ -712,8 +709,8 @@ public function writeShapeGroup(XMLWriter $objWriter, Group $group) $this->writeShapeLine($objWriter, $shape); } elseif ($shape instanceof Chart) { $this->writeShapeChart($objWriter, $shape); - } elseif ($shape instanceof AbstractDrawing) { - $this->writeShapePic($objWriter, $shape); + } elseif ($shape instanceof ShapeDrawing\AbstractDrawingAdapter) { + $this->writeShapeDrawing($objWriter, $shape); } elseif ($shape instanceof Group) { $this->writeShapeGroup($objWriter, $shape); } diff --git a/src/PhpPresentation/Writer/ODPresentation/MetaInfManifest.php b/src/PhpPresentation/Writer/ODPresentation/MetaInfManifest.php index e89bc961e..4d052bdb7 100644 --- a/src/PhpPresentation/Writer/ODPresentation/MetaInfManifest.php +++ b/src/PhpPresentation/Writer/ODPresentation/MetaInfManifest.php @@ -6,7 +6,6 @@ use PhpOffice\Common\File; use PhpOffice\Common\XMLWriter; use PhpOffice\PhpPresentation\Shape\Drawing as ShapeDrawing; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; use PhpOffice\PhpPresentation\Slide\Background\Image; use PhpOffice\PhpPresentation\Writer\ODPresentation; @@ -65,27 +64,17 @@ public function render() $arrMedia = array(); for ($i = 0; $i < $this->getDrawingHashTable()->count(); ++$i) { $shape = $this->getDrawingHashTable()->getByIndex($i); - if ($shape instanceof ShapeDrawing) { - if (!in_array(md5($shape->getPath()), $arrMedia)) { - $arrMedia[] = md5($shape->getPath()); - $mimeType = $this->getImageMimeType($shape->getPath()); - - $objWriter->startElement('manifest:file-entry'); - $objWriter->writeAttribute('manifest:media-type', $mimeType); - $objWriter->writeAttribute('manifest:full-path', 'Pictures/' . md5($shape->getPath()) . '.' . $shape->getExtension()); - $objWriter->endElement(); - } - } elseif ($shape instanceof MemoryDrawing) { - if (!in_array(str_replace(' ', '_', $shape->getIndexedFilename()), $arrMedia)) { - $arrMedia[] = str_replace(' ', '_', $shape->getIndexedFilename()); - $mimeType = $shape->getMimeType(); - - $objWriter->startElement('manifest:file-entry'); - $objWriter->writeAttribute('manifest:media-type', $mimeType); - $objWriter->writeAttribute('manifest:full-path', 'Pictures/' . str_replace(' ', '_', $shape->getIndexedFilename())); - $objWriter->endElement(); - } + if (! ($shape instanceof ShapeDrawing\AbstractDrawingAdapter)) { + continue; + } + if (in_array($shape->getIndexedFilename(), $arrMedia)) { + continue; } + $arrMedia[] = $shape->getIndexedFilename(); + $objWriter->startElement('manifest:file-entry'); + $objWriter->writeAttribute('manifest:media-type', $shape->getMimeType()); + $objWriter->writeAttribute('manifest:full-path', 'Pictures/' . $shape->getIndexedFilename()); + $objWriter->endElement(); } foreach ($this->getPresentation()->getAllSlides() as $numSlide => $oSlide) { diff --git a/src/PhpPresentation/Writer/ODPresentation/Pictures.php b/src/PhpPresentation/Writer/ODPresentation/Pictures.php index cd6beffe3..373f5270e 100644 --- a/src/PhpPresentation/Writer/ODPresentation/Pictures.php +++ b/src/PhpPresentation/Writer/ODPresentation/Pictures.php @@ -4,7 +4,6 @@ use PhpOffice\Common\Adapter\Zip\ZipInterface; use PhpOffice\PhpPresentation\Shape\Drawing; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; use PhpOffice\PhpPresentation\Slide\Background\Image; class Pictures extends AbstractDecoratorWriter @@ -17,43 +16,15 @@ public function render() { $arrMedia = array(); for ($i = 0; $i < $this->getDrawingHashTable()->count(); ++$i) { - $shape = $this->getDrawingHashTable()->getByIndex($i); - if ($shape instanceof Drawing) { - if (!in_array(md5($shape->getPath()), $arrMedia)) { - $arrMedia[] = md5($shape->getPath()); - - $imagePath = $shape->getPath(); - - if (strpos($imagePath, 'zip://') !== false) { - $imagePath = substr($imagePath, 6); - $imagePathSplitted = explode('#', $imagePath); - - $imageZip = new \ZipArchive(); - $imageZip->open($imagePathSplitted[0]); - $imageContents = $imageZip->getFromName($imagePathSplitted[1]); - $imageZip->close(); - unset($imageZip); - } elseif (strpos($imagePath, 'data:image/') === 0) { - list(, $imageContents) = explode(';', $imagePath); - list(, $imageContents) = explode(',', $imageContents); - $imageContents = base64_decode($imageContents); - } else { - $imageContents = file_get_contents($imagePath); - } - - $this->getZip()->addFromString('Pictures/' . md5($shape->getPath()).'.'.$shape->getExtension(), $imageContents); - } - } elseif ($shape instanceof MemoryDrawing) { - if (!in_array(str_replace(' ', '_', $shape->getIndexedFilename()), $arrMedia)) { - $arrMedia[] = str_replace(' ', '_', $shape->getIndexedFilename()); - ob_start(); - call_user_func($shape->getRenderingFunction(), $shape->getImageResource()); - $imageContents = ob_get_contents(); - ob_end_clean(); - - $this->getZip()->addFromString('Pictures/' . str_replace(' ', '_', $shape->getIndexedFilename()), $imageContents); - } + $shape = $this->getDrawingHashTable()->getByIndex($i);gi + if (!($shape instanceof Drawing\AbstractDrawingAdapter)) { + continue; + } + if (in_array($shape->getIndexedFilename(), $arrMedia)) { + continue; } + $arrMedia[] = $shape->getIndexedFilename(); + $this->getZip()->addFromString('Pictures/' . $shape->getIndexedFilename(), $shape->getContents()); } foreach ($this->getPresentation()->getAllSlides() as $keySlide => $oSlide) { diff --git a/src/PhpPresentation/Writer/PowerPoint2007.php b/src/PhpPresentation/Writer/PowerPoint2007.php index 121508edf..495683f1d 100644 --- a/src/PhpPresentation/Writer/PowerPoint2007.php +++ b/src/PhpPresentation/Writer/PowerPoint2007.php @@ -23,6 +23,7 @@ use PhpOffice\PhpPresentation\PhpPresentation; use PhpOffice\PhpPresentation\Shape\AbstractDrawing; use PhpOffice\PhpPresentation\Shape\Chart as ChartShape; +use PhpOffice\PhpPresentation\Shape\Drawing\AbstractDrawingAdapter; use PhpOffice\PhpPresentation\Shape\Group; use PhpOffice\PhpPresentation\Shape\Table; use PhpOffice\PhpPresentation\Writer\PowerPoint2007\LayoutPack\AbstractLayoutPack; @@ -202,42 +203,4 @@ public function setLayoutPack(AbstractLayoutPack $pValue = null) return $this; } - - /** - * Get an array of all drawings - * - * @return \PhpOffice\PhpPresentation\Shape\AbstractDrawing[] All drawings in PhpPresentation - * @throws \Exception - */ - protected function allDrawings() - { - // Get an array of all drawings - $aDrawings = array(); - - // Loop trough PhpPresentation - foreach ($this->getPhpPresentation()->getAllSlides() as $oSlide) { - $oCollection = $oSlide->getShapeCollection(); - if ($oCollection->count() <= 0) { - continue; - } - $oIterator = $oCollection->getIterator(); - while ($oIterator->valid()) { - if ($oIterator->current() instanceof AbstractDrawing && !($oIterator->current() instanceof Table)) { - $aDrawings[] = $oIterator->current(); - } elseif ($oIterator->current() instanceof Group) { - $oSubIterator = $oIterator->current()->getShapeCollection()->getIterator(); - while ($oSubIterator->valid()) { - if ($oSubIterator->current() instanceof AbstractDrawing && !($oSubIterator->current() instanceof Table)) { - $aDrawings[] = $oSubIterator->current(); - } - $oSubIterator->next(); - } - } - - $oIterator->next(); - } - } - - return $aDrawings; - } } diff --git a/src/PhpPresentation/Writer/PowerPoint2007/AbstractDecoratorWriter.php b/src/PhpPresentation/Writer/PowerPoint2007/AbstractDecoratorWriter.php index d8a126442..147d5bdf5 100644 --- a/src/PhpPresentation/Writer/PowerPoint2007/AbstractDecoratorWriter.php +++ b/src/PhpPresentation/Writer/PowerPoint2007/AbstractDecoratorWriter.php @@ -51,8 +51,16 @@ protected function writeRelationship(XMLWriter $objWriter, $pId = 1, $pType = '' * @param string $pElementName Element name * @throws \Exception */ - protected function writeBorder(XMLWriter $objWriter, Border $pBorder, $pElementName = 'L') + protected function writeBorder(XMLWriter $objWriter, $pBorder, $pElementName = 'L') { + if (!($pBorder instanceof Border)) { + return; + } + + if ($pBorder->getLineStyle() == Border::LINE_NONE && $pElementName == '') { + return; + } + // Line style $lineStyle = $pBorder->getLineStyle(); if ($lineStyle == Border::LINE_NONE) { diff --git a/src/PhpPresentation/Writer/PowerPoint2007/ContentTypes.php b/src/PhpPresentation/Writer/PowerPoint2007/ContentTypes.php index e1d0901fb..fa560b785 100644 --- a/src/PhpPresentation/Writer/PowerPoint2007/ContentTypes.php +++ b/src/PhpPresentation/Writer/PowerPoint2007/ContentTypes.php @@ -19,7 +19,6 @@ use PhpOffice\PhpPresentation\Shape\Chart as ShapeChart; use PhpOffice\PhpPresentation\Shape\Drawing as ShapeDrawing; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; use PhpOffice\Common\File; use PhpOffice\Common\XMLWriter; use PhpOffice\PhpPresentation\Writer\PowerPoint2007; @@ -140,16 +139,8 @@ public function render() // Chart content type $this->writeOverrideContentType($objWriter, '/ppt/charts/chart' . $shapeIndex->getImageIndex() . '.xml', 'application/vnd.openxmlformats-officedocument.drawingml.chart+xml'); } else { - if ($shapeIndex instanceof ShapeDrawing) { - $extension = strtolower($shapeIndex->getExtension()); - $mimeType = $this->getImageMimeType($shapeIndex->getPath()); - } elseif ($shapeIndex instanceof MemoryDrawing) { - $extension = strtolower($shapeIndex->getMimeType()); - $extension = explode('/', $extension); - $extension = $extension[1]; - - $mimeType = $shapeIndex->getMimeType(); - } + $extension = strtolower($shapeIndex->getExtension()); + $mimeType = $shapeIndex->getMimeType(); if (!isset($aMediaContentTypes[$extension])) { $aMediaContentTypes[$extension] = $mimeType; @@ -166,50 +157,6 @@ public function render() return $this->oZip; } - /** - * Get image mime type - * - * @param string $pFile Filename - * @return string Mime Type - * @throws \Exception - */ - private function getImageMimeType($pFile = '') - { - if (strpos($pFile, 'zip://') === 0) { - $pZIPFile = str_replace('zip://', '', $pFile); - $pZIPFile = substr($pZIPFile, 0, strpos($pZIPFile, '#')); - if (!File::fileExists($pZIPFile)) { - throw new \Exception("File $pFile does not exist"); - } - $pImgFile = substr($pFile, strpos($pFile, '#') + 1); - $oArchive = new \ZipArchive(); - $oArchive->open($pZIPFile); - if (!function_exists('getimagesizefromstring')) { - $uri = 'data://application/octet-stream;base64,' . base64_encode($oArchive->getFromName($pImgFile)); - $image = getimagesize($uri); - } else { - $image = getimagesizefromstring($oArchive->getFromName($pImgFile)); - } - } elseif (strpos($pFile, 'data:image/') === 0) { - $sImage = $pFile; - list(, $sImage) = explode(';', $sImage); - list(, $sImage) = explode(',', $sImage); - if (!function_exists('getimagesizefromstring')) { - $uri = 'data://application/octet-stream;base64,' . base64_encode($sImage); - $image = getimagesize($uri); - } else { - $image = getimagesizefromstring($sImage); - } - } else { - if (!File::fileExists($pFile)) { - throw new \Exception("File $pFile does not exist"); - } - $image = getimagesize($pFile); - } - - return image_type_to_mime_type($image[2]); - } - /** * Write Default content type * diff --git a/src/PhpPresentation/Writer/PowerPoint2007/PptMedia.php b/src/PhpPresentation/Writer/PowerPoint2007/PptMedia.php index 7671a7e2c..017c19113 100644 --- a/src/PhpPresentation/Writer/PowerPoint2007/PptMedia.php +++ b/src/PhpPresentation/Writer/PowerPoint2007/PptMedia.php @@ -3,7 +3,6 @@ namespace PhpOffice\PhpPresentation\Writer\PowerPoint2007; use PhpOffice\PhpPresentation\Shape\Drawing; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing; class PptMedia extends AbstractDecoratorWriter { @@ -14,33 +13,10 @@ public function render() { for ($i = 0; $i < $this->getDrawingHashTable()->count(); ++$i) { $shape = $this->getDrawingHashTable()->getByIndex($i); - if ($shape instanceof Drawing) { - $imagePath = $shape->getPath(); - if (strpos($imagePath, 'zip://') !== false) { - $imagePath = substr($imagePath, 6); - $imagePathSplitted = explode('#', $imagePath); - - $imageZip = new \ZipArchive(); - $imageZip->open($imagePathSplitted[0]); - $imageContents = $imageZip->getFromName($imagePathSplitted[1]); - $imageZip->close(); - unset($imageZip); - } elseif (strpos($imagePath, 'data:image/') === 0) { - list(, $imageContents) = explode(';', $imagePath); - list(, $imageContents) = explode(',', $imageContents); - $imageContents = base64_decode($imageContents); - } else { - $imageContents = file_get_contents($imagePath); - } - $this->getZip()->addFromString('ppt/media/' . str_replace(' ', '_', $shape->getIndexedFilename()), $imageContents); - } elseif ($shape instanceof MemoryDrawing) { - ob_start(); - call_user_func($shape->getRenderingFunction(), $shape->getImageResource()); - $imageContents = ob_get_contents(); - ob_end_clean(); - - $this->getZip()->addFromString('ppt/media/' . str_replace(' ', '_', $shape->getIndexedFilename()), $imageContents); + if (!$shape instanceof Drawing\AbstractDrawingAdapter) { + continue; } + $this->getZip()->addFromString('ppt/media/' . $shape->getIndexedFilename(), $shape->getContents()); } return $this->getZip(); diff --git a/src/PhpPresentation/Writer/PowerPoint2007/PptSlides.php b/src/PhpPresentation/Writer/PowerPoint2007/PptSlides.php index cccaac5a2..95c198454 100644 --- a/src/PhpPresentation/Writer/PowerPoint2007/PptSlides.php +++ b/src/PhpPresentation/Writer/PowerPoint2007/PptSlides.php @@ -11,7 +11,6 @@ use PhpOffice\PhpPresentation\Shape\Drawing as ShapeDrawing; use PhpOffice\PhpPresentation\Shape\Group; use PhpOffice\PhpPresentation\Shape\Line; -use PhpOffice\PhpPresentation\Shape\MemoryDrawing as MemoryDrawing; use PhpOffice\PhpPresentation\Shape\RichText; use PhpOffice\PhpPresentation\Shape\RichText\BreakElement; use PhpOffice\PhpPresentation\Shape\RichText\Run; @@ -91,12 +90,10 @@ protected function writeSlideRelationships(Slide $pSlide) // Loop trough images and write relationships $iterator = $pSlide->getShapeCollection()->getIterator(); while ($iterator->valid()) { - if ($iterator->current() instanceof ShapeDrawing || $iterator->current() instanceof MemoryDrawing) { + if ($iterator->current() instanceof ShapeDrawing\AbstractDrawingAdapter) { // Write relationship for image drawing - $this->writeRelationship($objWriter, $relId, 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/image', '../media/' . str_replace(' ', '_', $iterator->current()->getIndexedFilename())); - + $this->writeRelationship($objWriter, $relId, 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/image', '../media/' . $iterator->current()->getIndexedFilename()); $iterator->current()->relationId = 'rId' . $relId; - ++$relId; } elseif ($iterator->current() instanceof ShapeChart) { // Write relationship for chart drawing @@ -108,17 +105,15 @@ protected function writeSlideRelationships(Slide $pSlide) } elseif ($iterator->current() instanceof Group) { $iterator2 = $iterator->current()->getShapeCollection()->getIterator(); while ($iterator2->valid()) { - if ($iterator2->current() instanceof ShapeDrawing || $iterator2->current() instanceof MemoryDrawing) { + if ($iterator->current() instanceof ShapeDrawing\AbstractDrawingAdapter) { // Write relationship for image drawing - $this->writeRelationship($objWriter, $relId, 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/image', '../media/' . str_replace(' ', '_', $iterator2->current()->getIndexedFilename())); - + $this->writeRelationship($objWriter, $relId, 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/image', '../media/' . $iterator2->current()->getIndexedFilename()); $iterator2->current()->relationId = 'rId' . $relId; ++$relId; } elseif ($iterator2->current() instanceof ShapeChart) { // Write relationship for chart drawing $this->writeRelationship($objWriter, $relId, 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/chart', '../charts/' . $iterator2->current()->getIndexedFilename()); - $iterator2->current()->relationId = 'rId' . $relId; ++$relId; @@ -489,8 +484,8 @@ public function writeSlide(Slide $pSlide) $this->writeShapeLine($objWriter, $shape, $shapeId); } elseif ($shape instanceof ShapeChart) { $this->writeShapeChart($objWriter, $shape, $shapeId); - } elseif ($shape instanceof AbstractDrawing) { - $this->writeShapePic($objWriter, $shape, $shapeId); + } elseif ($shape instanceof ShapeDrawing\AbstractDrawingAdapter) { + $this->writeShapeDrawing($objWriter, $shape, $shapeId); } elseif ($shape instanceof Group) { $this->writeShapeGroup($objWriter, $shape, $shapeId); } @@ -597,8 +592,8 @@ private function writeShapeGroup(XMLWriter $objWriter, Group $group, &$shapeId) $this->writeShapeLine($objWriter, $shape, $shapeId); } elseif ($shape instanceof ShapeChart) { $this->writeShapeChart($objWriter, $shape, $shapeId); - } elseif ($shape instanceof AbstractDrawing) { - $this->writeShapePic($objWriter, $shape, $shapeId); + } elseif ($shape instanceof ShapeDrawing\AbstractDrawingAdapter) { + $this->writeShapeDrawing($objWriter, $shape, $shapeId); } elseif ($shape instanceof Group) { $this->writeShapeGroup($objWriter, $shape, $shapeId); } @@ -680,11 +675,11 @@ private function writeShapeChart(XMLWriter $objWriter, ShapeChart $shape, $shape * Write pic * * @param \PhpOffice\Common\XMLWriter $objWriter XML Writer - * @param \PhpOffice\PhpPresentation\Shape\AbstractDrawing $shape + * @param \PhpOffice\PhpPresentation\Shape\Drawing\AbstractDrawingAdapter $shape * @param int $shapeId * @throws \Exception */ - private function writeShapePic(XMLWriter $objWriter, AbstractDrawing $shape, $shapeId) + private function writeShapeDrawing(XMLWriter $objWriter, ShapeDrawing\AbstractDrawingAdapter $shape, $shapeId) { // p:pic $objWriter->startElement('p:pic'); @@ -764,13 +759,9 @@ private function writeShapePic(XMLWriter $objWriter, AbstractDrawing $shape, $sh $objWriter->endElement(); - if ($shape->getBorder()->getLineStyle() != Border::LINE_NONE) { - $this->writeBorder($objWriter, $shape->getBorder(), ''); - } + $this->writeBorder($objWriter, $shape->getBorder(), ''); - if ($shape->getShadow()->isVisible()) { - $this->writeShadow($objWriter, $shape->getShadow()); - } + $this->writeShadow($objWriter, $shape->getShadow()); $objWriter->endElement(); @@ -844,12 +835,9 @@ private function writeShapeText(XMLWriter $objWriter, RichText $shape, $shapeId) if ($shape->getFill()) { $this->writeFill($objWriter, $shape->getFill()); } - if ($shape->getBorder()->getLineStyle() != Border::LINE_NONE) { - $this->writeBorder($objWriter, $shape->getBorder(), ''); - } - if ($shape->getShadow()->isVisible()) { - $this->writeShadow($objWriter, $shape->getShadow()); - } + $this->writeBorder($objWriter, $shape->getBorder(), ''); + $this->writeShadow($objWriter, $shape->getShadow()); + // > p:sp\p:spPr $objWriter->endElement(); @@ -1374,8 +1362,16 @@ private function writeShapeLine(XMLWriter $objWriter, Line $shape, $shapeId) * @param XMLWriter $objWriter * @param Shadow $oShadow */ - protected function writeShadow(XMLWriter $objWriter, Shadow $oShadow) + protected function writeShadow(XMLWriter $objWriter, $oShadow) { + if (!($oShadow instanceof Shadow)) { + return; + } + + if (!$oShadow->isVisible()) { + return; + } + // a:effectLst $objWriter->startElement('a:effectLst');