2020-07-09 16:05:17 +02:00
|
|
|
<?php
|
|
|
|
|
2020-07-22 14:30:55 +02:00
|
|
|
/**
|
|
|
|
* Invoice Ninja (https://invoiceninja.com)
|
|
|
|
*
|
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
|
|
*
|
|
|
|
* @copyright Copyright (c) 2020. Invoice Ninja LLC (https://invoiceninja.com)
|
|
|
|
*
|
|
|
|
* @license https://opensource.org/licenses/AAL
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Services\PdfMaker;
|
2020-07-09 16:05:17 +02:00
|
|
|
|
|
|
|
use DOMDocument;
|
2020-08-26 09:02:38 +02:00
|
|
|
use DOMDomError;
|
2020-07-09 16:05:17 +02:00
|
|
|
use DOMXPath;
|
|
|
|
|
|
|
|
trait PdfMakerUtilities
|
|
|
|
{
|
|
|
|
private function initializeDomDocument()
|
|
|
|
{
|
|
|
|
$document = new DOMDocument();
|
|
|
|
|
|
|
|
$document->validateOnParse = true;
|
|
|
|
@$document->loadHTML($this->design->html());
|
2020-07-13 14:16:18 +02:00
|
|
|
|
2020-07-09 16:05:17 +02:00
|
|
|
$this->document = $document;
|
|
|
|
$this->xpath = new DOMXPath($document);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSection(string $selector, string $section = null)
|
|
|
|
{
|
|
|
|
$element = $this->document->getElementById($selector);
|
|
|
|
|
|
|
|
if ($section) {
|
|
|
|
return $element->getAttribute($section);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $element->nodeValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSectionNode(string $selector)
|
|
|
|
{
|
|
|
|
return $this->document->getElementById($selector);
|
|
|
|
}
|
|
|
|
|
2020-07-13 17:49:28 +02:00
|
|
|
public function updateElementProperties(array $elements)
|
|
|
|
{
|
|
|
|
foreach ($elements as $element) {
|
2020-09-04 10:18:41 +02:00
|
|
|
|
|
|
|
// if (!isset($element['tag']) || !isset($element['id']) || is_null($this->document->getElementById($element['id']))) {
|
|
|
|
// continue;
|
|
|
|
// }
|
|
|
|
|
2020-08-25 10:51:49 +02:00
|
|
|
if (isset($element['tag'])) {
|
|
|
|
$node = $this->document->getElementsByTagName($element['tag'])->item(0);
|
2020-09-04 10:18:41 +02:00
|
|
|
} elseif(!is_null($this->document->getElementById($element['id']))) {
|
2020-08-25 10:51:49 +02:00
|
|
|
$node = $this->document->getElementById($element['id']);
|
2020-09-04 10:18:41 +02:00
|
|
|
} else {
|
|
|
|
continue;
|
2020-08-25 10:51:49 +02:00
|
|
|
}
|
2020-07-13 17:49:28 +02:00
|
|
|
|
|
|
|
if (isset($element['properties'])) {
|
|
|
|
foreach ($element['properties'] as $property => $value) {
|
2020-07-14 13:50:00 +02:00
|
|
|
$this->updateElementProperty($node, $property, $value);
|
2020-07-13 17:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($element['elements'])) {
|
2020-07-14 14:35:27 +02:00
|
|
|
$sorted = $this->processChildrenOrder($element['elements']);
|
|
|
|
|
|
|
|
$this->createElementContent($node, $sorted);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function processChildrenOrder(array $children)
|
|
|
|
{
|
|
|
|
$processed = [];
|
|
|
|
|
2020-08-10 16:43:25 +02:00
|
|
|
foreach ($children as $child) {
|
2020-07-14 14:35:27 +02:00
|
|
|
if (!isset($child['order'])) {
|
|
|
|
$child['order'] = 0;
|
2020-07-13 17:49:28 +02:00
|
|
|
}
|
2020-07-14 14:35:27 +02:00
|
|
|
|
|
|
|
$processed[] = $child;
|
2020-07-13 17:49:28 +02:00
|
|
|
}
|
2020-07-14 15:05:29 +02:00
|
|
|
|
2020-07-14 14:35:27 +02:00
|
|
|
usort($processed, function ($a, $b) {
|
|
|
|
return $a['order'] <=> $b['order'];
|
|
|
|
});
|
2020-07-14 15:05:29 +02:00
|
|
|
|
2020-07-14 14:35:27 +02:00
|
|
|
return $processed;
|
2020-07-13 17:49:28 +02:00
|
|
|
}
|
|
|
|
|
2020-07-14 13:50:00 +02:00
|
|
|
public function updateElementProperty($element, string $attribute, string $value)
|
2020-07-09 16:05:17 +02:00
|
|
|
{
|
2020-08-10 16:43:25 +02:00
|
|
|
// We have exception for "hidden" property.
|
|
|
|
// hidden="true" or hidden="false" will both hide the element,
|
|
|
|
// that's why we have to create an exception here for this rule.
|
|
|
|
|
|
|
|
if ($attribute == 'hidden' && ($value == false || $value == "false")) {
|
|
|
|
return $element;
|
|
|
|
}
|
|
|
|
|
2020-07-09 16:05:17 +02:00
|
|
|
$element->setAttribute($attribute, $value);
|
|
|
|
|
|
|
|
if ($element->getAttribute($attribute) === $value) {
|
|
|
|
return $element;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $element;
|
|
|
|
}
|
|
|
|
|
2020-07-13 17:49:28 +02:00
|
|
|
public function createElementContent($element, $children)
|
|
|
|
{
|
|
|
|
foreach ($children as $child) {
|
2020-09-04 10:18:41 +02:00
|
|
|
$_child = $this->document->createElement($child['element'], isset($child['content']) ? $child['content'] : '');
|
2020-07-13 17:49:28 +02:00
|
|
|
$element->appendChild($_child);
|
2020-07-14 13:50:00 +02:00
|
|
|
|
|
|
|
if (isset($child['properties'])) {
|
|
|
|
foreach ($child['properties'] as $property => $value) {
|
|
|
|
$this->updateElementProperty($_child, $property, $value);
|
|
|
|
}
|
|
|
|
}
|
2020-07-14 14:35:27 +02:00
|
|
|
|
2020-07-13 17:49:28 +02:00
|
|
|
if (isset($child['elements'])) {
|
2020-07-14 15:05:29 +02:00
|
|
|
$sorted = $this->processChildrenOrder($child['elements']);
|
|
|
|
|
|
|
|
$this->createElementContent($_child, $sorted);
|
2020-07-13 17:49:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-13 14:16:18 +02:00
|
|
|
public function updateVariables(array $variables)
|
|
|
|
{
|
2020-07-29 13:37:05 +02:00
|
|
|
$html = strtr($this->getCompiledHTML(), $variables['labels']);
|
2020-08-10 16:43:25 +02:00
|
|
|
|
2020-07-30 16:43:57 +02:00
|
|
|
$html = strtr($html, $variables['values']);
|
2020-07-13 14:16:18 +02:00
|
|
|
|
2020-08-11 17:53:11 +02:00
|
|
|
@$this->document->loadHTML($html);
|
2020-07-13 14:16:18 +02:00
|
|
|
|
|
|
|
$this->document->saveHTML();
|
|
|
|
}
|
|
|
|
|
2020-07-09 16:05:17 +02:00
|
|
|
public function updateVariable(string $element, string $variable, string $value)
|
|
|
|
{
|
|
|
|
$element = $this->document->getElementById($element);
|
|
|
|
|
|
|
|
$original = $element->nodeValue;
|
|
|
|
|
|
|
|
$element->nodeValue = '';
|
|
|
|
|
|
|
|
$replaced = strtr($original, [$variable => $value]);
|
|
|
|
|
|
|
|
$element->appendChild(
|
|
|
|
$this->document->createTextNode($replaced)
|
|
|
|
);
|
|
|
|
|
|
|
|
return $element;
|
|
|
|
}
|
2020-08-25 10:51:49 +02:00
|
|
|
|
|
|
|
public function processOptions()
|
|
|
|
{
|
2020-08-27 08:47:51 +02:00
|
|
|
if (!isset($this->options['all_pages_header']) && !isset($this->options['all_pages_footer'])) {
|
2020-08-27 08:39:14 +02:00
|
|
|
return;
|
2020-08-25 10:51:49 +02:00
|
|
|
}
|
2020-08-27 08:39:14 +02:00
|
|
|
|
|
|
|
$this->insertPrintCSS();
|
|
|
|
$this->wrapIntoTable();
|
2020-08-25 10:51:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function insertPrintCSS()
|
|
|
|
{
|
2020-08-26 09:02:38 +02:00
|
|
|
$css = <<<EOT
|
|
|
|
table.page-container {
|
|
|
|
page-break-after: always;
|
|
|
|
}
|
|
|
|
|
|
|
|
thead.page-header {
|
|
|
|
display: table-header-group;
|
|
|
|
}
|
|
|
|
|
|
|
|
tfoot.page-footer {
|
|
|
|
display: table-footer-group;
|
|
|
|
}
|
|
|
|
EOT;
|
2020-08-25 10:51:49 +02:00
|
|
|
|
|
|
|
$css_node = $this->document->createTextNode($css);
|
|
|
|
|
|
|
|
$style = $this->document->getElementsByTagName('style')->item(0);
|
|
|
|
|
|
|
|
if ($style) {
|
|
|
|
return $style->appendChild($css_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
$head = $this->document->getElementsByTagName('head')->item(0);
|
|
|
|
|
|
|
|
if ($head) {
|
|
|
|
$style_node = $this->document->createElement('style', $css);
|
|
|
|
|
|
|
|
return $head->appendChild($style_node);
|
|
|
|
}
|
|
|
|
}
|
2020-08-26 09:02:38 +02:00
|
|
|
|
|
|
|
public function wrapIntoTable()
|
|
|
|
{
|
|
|
|
$markup = <<<EOT
|
|
|
|
<table class="page-container" id="page-container">
|
|
|
|
<thead class="page-report">
|
|
|
|
<tr>
|
|
|
|
<th class="page-report-cell" id="repeat-header">
|
|
|
|
<!-- Repeating header goes here.. -->
|
|
|
|
</th>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tfoot class="report-footer">
|
|
|
|
<tr>
|
|
|
|
<td class="report-footer-cell" id="repeat-footer">
|
|
|
|
<!-- Repeating footer goes here -->
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tfoot>
|
|
|
|
<tbody class="report-content">
|
|
|
|
<tr>
|
|
|
|
<td class="report-content-cell" id="repeat-content">
|
|
|
|
<!-- Rest of the content goes here -->
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
EOT;
|
|
|
|
|
|
|
|
$document = new DOMDocument();
|
|
|
|
$document->loadHTML($markup);
|
|
|
|
|
|
|
|
$table = $document->getElementById('page-container');
|
|
|
|
|
2020-08-26 12:52:20 +02:00
|
|
|
$body = $this->document->getElementsByTagName('body')
|
|
|
|
->item(0);
|
2020-08-26 09:02:38 +02:00
|
|
|
|
2020-08-26 12:52:20 +02:00
|
|
|
$body->appendChild(
|
|
|
|
$this->document->importNode($table, true)
|
|
|
|
);
|
|
|
|
|
|
|
|
for ($i = 0; $i < $body->childNodes->length; $i++) {
|
|
|
|
$element = $body->childNodes->item($i);
|
|
|
|
|
|
|
|
if ($element->nodeType !== 1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
$element->getAttribute('id') == 'header' ||
|
|
|
|
$element->getAttribute('id') == 'footer' ||
|
|
|
|
$element->getAttribute('id') === 'page-container'
|
|
|
|
) {
|
2020-08-26 09:02:38 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-08-26 12:52:20 +02:00
|
|
|
$clone = $element->cloneNode(true);
|
|
|
|
$element->parentNode->removeChild($element);
|
2020-08-26 09:02:38 +02:00
|
|
|
|
2020-08-26 12:52:20 +02:00
|
|
|
$this->document->getElementById('repeat-content')->appendChild($clone);
|
2020-08-26 09:02:38 +02:00
|
|
|
}
|
|
|
|
|
2020-08-27 08:39:14 +02:00
|
|
|
if (
|
2020-09-04 10:18:41 +02:00
|
|
|
$header = $this->document->getElementById('header') &&
|
2020-08-27 08:47:51 +02:00
|
|
|
isset($this->data['options']['all_pages_header']) &&
|
|
|
|
$this->data['options']['all_pages_header']
|
2020-08-27 08:39:14 +02:00
|
|
|
) {
|
|
|
|
|
2020-08-26 09:02:38 +02:00
|
|
|
$header = $this->document->getElementById('header');
|
2020-08-26 12:52:20 +02:00
|
|
|
$clone = $header->cloneNode(true);
|
2020-08-26 09:02:38 +02:00
|
|
|
|
2020-08-26 12:52:20 +02:00
|
|
|
$header->parentNode->removeChild($header);
|
|
|
|
$this->document->getElementById('repeat-header')->appendChild($clone);
|
2020-08-26 09:02:38 +02:00
|
|
|
}
|
|
|
|
|
2020-08-27 08:39:14 +02:00
|
|
|
if (
|
|
|
|
$footer = $this->document->getElementById('footer') &&
|
2020-08-27 08:47:51 +02:00
|
|
|
isset($this->data['options']['all_pages_footer']) &&
|
|
|
|
$this->data['options']['all_pages_footer']
|
2020-08-27 08:39:14 +02:00
|
|
|
) {
|
2020-08-26 09:02:38 +02:00
|
|
|
$footer = $this->document->getElementById('footer');
|
2020-08-26 12:52:20 +02:00
|
|
|
$clone = $footer->cloneNode(true);
|
2020-08-26 09:02:38 +02:00
|
|
|
|
2020-08-26 12:52:20 +02:00
|
|
|
$footer->parentNode->removeChild($footer);
|
|
|
|
$this->document->getElementById('repeat-footer')->appendChild($clone);
|
2020-08-26 09:02:38 +02:00
|
|
|
}
|
|
|
|
}
|
2020-07-13 14:16:18 +02:00
|
|
|
}
|