1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-16 08:02:40 +01:00
invoiceninja/app/Services/PdfMaker/PdfMakerUtilities.php

335 lines
10 KiB
PHP
Raw Normal View History

<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
*/
namespace App\Services\PdfMaker;
use DOMDocument;
use DOMXPath;
trait PdfMakerUtilities
{
private function initializeDomDocument()
{
$document = new DOMDocument();
$document->validateOnParse = true;
@$document->loadHTML(mb_convert_encoding($this->design->html(), 'HTML-ENTITIES', 'UTF-8'));
2020-07-13 14:16:18 +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);
}
public function updateElementProperties(array $elements)
{
foreach ($elements as $element) {
if (isset($element['tag'])) {
$node = $this->document->getElementsByTagName($element['tag'])->item(0);
2020-09-08 14:26:13 +02:00
} elseif (!is_null($this->document->getElementById($element['id']))) {
$node = $this->document->getElementById($element['id']);
} else {
continue;
}
if (isset($element['properties'])) {
foreach ($element['properties'] as $property => $value) {
$this->updateElementProperty($node, $property, $value);
}
}
if (isset($element['elements'])) {
2021-05-17 13:28:33 +02:00
$this->createElementContent($node, $element['elements']);
}
}
}
2021-01-07 07:24:02 +01:00
public function updateElementProperty($element, string $attribute, ?string $value)
{
2020-08-10 16:43:25 +02:00
// We have exception for "hidden" property.
// hidden="true" or hidden="false" will both hide the element,
2020-08-10 16:43:25 +02:00
// that's why we have to create an exception here for this rule.
if ($attribute == 'hidden' && ($value == false || $value == 'false')) {
2020-08-10 16:43:25 +02:00
return $element;
}
$element->setAttribute($attribute, $value);
if ($element->getAttribute($attribute) === $value) {
return $element;
}
return $element;
}
public function createElementContent($element, $children)
{
foreach ($children as $child) {
2020-10-09 15:05:57 +02:00
$contains_html = false;
2020-12-23 13:49:25 +01:00
if (isset($child['content'])) {
2021-05-31 14:08:24 +02:00
// Commented cause it keeps adding <br> at the end, if markdown parsing is turned on.
// Should update with 'parse_markdown_on_pdfs' setting.
2021-06-04 13:26:32 +02:00
$child['content'] = nl2br($child['content']);
2020-12-23 13:49:25 +01:00
}
2020-10-09 15:05:57 +02:00
// "/\/[a-z]*>/i" -> checks for HTML-like tags:
// <my-tag></my-tag> => true
// <my-tag /> => true
// <my-tag> => false
if (isset($child['content'])) {
if (isset($child['is_empty']) && $child['is_empty'] === true) {
continue;
}
$contains_html = preg_match("/\/[a-z]*>/i", $child['content'], $m) != 0;
2020-10-09 15:05:57 +02:00
}
if ($contains_html) {
2021-04-09 13:06:14 +02:00
// If the element contains the HTML, we gonna display it as is. DOMDocument, is going to
// encode it for us, preventing any errors on the backend due processing stage.
// Later, we decode this using Javascript so it looks like it's normal HTML being injected.
// To get all elements that need frontend decoding, we use 'data-ref' property.
2020-10-09 15:05:57 +02:00
$_child = $this->document->createElement($child['element'], '');
2021-04-09 13:06:14 +02:00
$_child->setAttribute('data-state', 'encoded-html');
2021-04-09 13:11:28 +02:00
$_child->nodeValue = htmlspecialchars($child['content']);
2020-10-09 15:05:57 +02:00
} else {
// .. in case string doesn't contain any HTML, we'll just return
// raw $content.
2020-11-16 00:49:44 +01:00
$_child = $this->document->createElement($child['element'], isset($child['content']) ? htmlspecialchars($child['content']) : '');
2020-10-09 15:05:57 +02:00
}
$element->appendChild($_child);
if (isset($child['properties'])) {
foreach ($child['properties'] as $property => $value) {
$this->updateElementProperty($_child, $property, $value);
}
}
if (isset($child['elements'])) {
2021-05-17 13:28:33 +02:00
$this->createElementContent($_child, $child['elements']);
}
}
}
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
@$this->document->loadHTML(mb_convert_encoding($html, 'HTML-ENTITIES', 'UTF-8'));
2020-07-13 14:16:18 +02:00
$this->document->saveHTML();
}
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;
}
public function processOptions()
{
2020-09-08 14:26:13 +02:00
if (!isset($this->options['all_pages_header']) || $this->options['all_pages_header'] == false) {
return;
}
2020-10-09 15:05:57 +02:00
2020-09-08 14:26:13 +02:00
if (!isset($this->options['all_pages_footer']) || $this->options['all_pages_footer'] == false) {
return;
}
$this->insertPrintCSS();
$this->wrapIntoTable();
}
public function insertPrintCSS()
{
$css = <<<'EOT'
2020-08-26 09:02:38 +02:00
table.page-container {
page-break-after: always;
min-width: 100%;
2020-08-26 09:02:38 +02:00
}
2020-10-09 15:05:57 +02:00
2020-08-26 09:02:38 +02:00
thead.page-header {
display: table-header-group;
}
tfoot.page-footer {
display: table-footer-group;
}
EOT;
$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-10-09 15:05:57 +02:00
return $this;
}
2020-08-26 09:02:38 +02:00
public function wrapIntoTable()
{
$markup = <<<'EOT'
2020-08-26 09:02:38 +02:00
<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');
$body = $this->document->getElementsByTagName('body')
->item(0);
2020-08-26 09:02:38 +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;
}
$clone = $element->cloneNode(true);
$element->parentNode->removeChild($element);
2020-08-26 09:02:38 +02:00
$this->document->getElementById('repeat-content')->appendChild($clone);
2020-08-26 09:02:38 +02:00
}
2020-11-25 15:19:52 +01:00
// info($this->data['options']);
2020-09-08 14:26:13 +02:00
if (
$header = $this->document->getElementById('header') &&
isset($this->data['options']['all_pages_header']) &&
$this->data['options']['all_pages_header']
) {
2020-08-26 09:02:38 +02:00
$header = $this->document->getElementById('header');
$clone = $header->cloneNode(true);
2020-08-26 09:02:38 +02:00
$header->parentNode->removeChild($header);
$this->document->getElementById('repeat-header')->appendChild($clone);
2020-08-26 09:02:38 +02:00
}
if (
$footer = $this->document->getElementById('footer') &&
isset($this->data['options']['all_pages_footer']) &&
$this->data['options']['all_pages_footer']
) {
2020-08-26 09:02:38 +02:00
$footer = $this->document->getElementById('footer');
$clone = $footer->cloneNode(true);
2020-08-26 09:02:38 +02:00
$footer->parentNode->removeChild($footer);
$this->document->getElementById('repeat-footer')->appendChild($clone);
2020-08-26 09:02:38 +02:00
}
}
2020-11-25 15:19:52 +01:00
public function getEmptyElements(array &$elements, array $variables)
{
foreach ($elements as &$element) {
if (isset($element['elements'])) {
$this->getEmptyChildrens($element['elements'], $variables);
}
}
}
2020-11-25 15:19:52 +01:00
public function getEmptyChildrens(array &$children, array $variables)
{
foreach ($children as $key => &$child) {
if (isset($child['content']) && isset($child['show_empty']) && $child['show_empty'] === false) {
$value = strtr($child['content'], $variables['values']);
if ($value === '' || $value === '&nbsp;') {
$child['is_empty'] = true;
}
}
2020-11-25 15:19:52 +01:00
if (isset($child['elements'])) {
$this->getEmptyChildrens($child['elements'], $variables);
}
}
}
2020-07-13 14:16:18 +02:00
}