mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-11-04 02:07:33 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			192 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
/**
 | 
						|
 * Invoice Ninja (https://invoiceninja.com).
 | 
						|
 *
 | 
						|
 * @link https://github.com/invoiceninja/invoiceninja source repository
 | 
						|
 *
 | 
						|
 * @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
 | 
						|
 *
 | 
						|
 * @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'));
 | 
						|
 | 
						|
        $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);
 | 
						|
            } 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'])) {
 | 
						|
                $this->createElementContent($node, $element['elements']);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function updateElementProperty($element, string $attribute, ?string $value)
 | 
						|
    {
 | 
						|
        // 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;
 | 
						|
        }
 | 
						|
 | 
						|
        $element->setAttribute($attribute, $value);
 | 
						|
 | 
						|
        if ($element->getAttribute($attribute) === $value) {
 | 
						|
            return $element;
 | 
						|
        }
 | 
						|
 | 
						|
        return $element;
 | 
						|
    }
 | 
						|
 | 
						|
    public function createElementContent($element, $children)
 | 
						|
    {
 | 
						|
        foreach ($children as $child) {
 | 
						|
            $contains_html = false;
 | 
						|
 | 
						|
            if ($child['element'] !== 'script') {
 | 
						|
                if (array_key_exists('process_markdown', $this->data) && array_key_exists('content', $child) && $this->data['process_markdown']) {
 | 
						|
                    $child['content'] = str_replace('<br>', "\r", ($child['content'] ?? ''));
 | 
						|
                    $child['content'] = $this->commonmark->convert($child['content'] ?? '');
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            if (isset($child['content'])) {
 | 
						|
                if (isset($child['is_empty']) && $child['is_empty'] === true) {
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
 | 
						|
                $contains_html = preg_match('#(?<=<)\w+(?=[^<]*?>)#', $child['content'], $m) != 0;
 | 
						|
            }
 | 
						|
 | 
						|
            if ($contains_html) {
 | 
						|
                // If the element contains the HTML, we gonna display it as is. Backend is going to
 | 
						|
                // encode it for us, preventing any errors on the 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-state' property.
 | 
						|
 | 
						|
                $_child = $this->document->createElement($child['element'], '');
 | 
						|
                $_child->setAttribute('data-state', 'encoded-html');
 | 
						|
                $_child->nodeValue = htmlspecialchars($child['content']);
 | 
						|
            } else {
 | 
						|
                // .. in case string doesn't contain any HTML, we'll just return
 | 
						|
                // raw $content.
 | 
						|
 | 
						|
                $_child = $this->document->createElement($child['element'], isset($child['content']) ? htmlspecialchars($child['content']) : '');
 | 
						|
            }
 | 
						|
 | 
						|
            $element->appendChild($_child);
 | 
						|
 | 
						|
            if (isset($child['properties'])) {
 | 
						|
                foreach ($child['properties'] as $property => $value) {
 | 
						|
                    $this->updateElementProperty($_child, $property, $value);
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            if (isset($child['elements'])) {
 | 
						|
                $this->createElementContent($_child, $child['elements']);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function updateVariables(array $variables)
 | 
						|
    {
 | 
						|
        $html = strtr($this->getCompiledHTML(), $variables['labels']);
 | 
						|
 | 
						|
        $html = strtr($html, $variables['values']);
 | 
						|
 | 
						|
        @$this->document->loadHTML(mb_convert_encoding($html, 'HTML-ENTITIES', 'UTF-8'));
 | 
						|
 | 
						|
        $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 getEmptyElements(array &$elements, array $variables)
 | 
						|
    {
 | 
						|
        foreach ($elements as &$element) {
 | 
						|
            if (isset($element['elements'])) {
 | 
						|
                $this->getEmptyChildrens($element['elements'], $variables);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    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 === ' ' || $value === ' ') {
 | 
						|
                    $child['is_empty'] = true;
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            if (isset($child['elements'])) {
 | 
						|
                $this->getEmptyChildrens($child['elements'], $variables);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |