Passed
Push — master ( 5132bf...6af461 )
by Esteban De La Fuente
04:21
created

RendererWorker::getFilesystemLoader()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 7
Code Lines 3

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 4
CRAP Score 2

Importance

Changes 0
Metric Value
cc 2
eloc 3
c 0
b 0
f 0
nc 2
nop 0
dl 0
loc 7
ccs 4
cts 4
cp 1
crap 2
rs 10
1
<?php
2
3
declare(strict_types=1);
4
5
/**
6
 * Derafu: Biblioteca PHP (Núcleo).
7
 * Copyright (C) Derafu <https://www.derafu.org>
8
 *
9
 * Este programa es software libre: usted puede redistribuirlo y/o modificarlo
10
 * bajo los términos de la Licencia Pública General Affero de GNU publicada por
11
 * la Fundación para el Software Libre, ya sea la versión 3 de la Licencia, o
12
 * (a su elección) cualquier versión posterior de la misma.
13
 *
14
 * Este programa se distribuye con la esperanza de que sea útil, pero SIN
15
 * GARANTÍA ALGUNA; ni siquiera la garantía implícita MERCANTIL o de APTITUD
16
 * PARA UN PROPÓSITO DETERMINADO. Consulte los detalles de la Licencia Pública
17
 * General Affero de GNU para obtener una información más detallada.
18
 *
19
 * Debería haber recibido una copia de la Licencia Pública General Affero de GNU
20
 * junto a este programa.
21
 *
22
 * En caso contrario, consulte <http://www.gnu.org/licenses/agpl.html>.
23
 */
24
25
namespace Derafu\Lib\Core\Package\Prime\Component\Template\Worker;
26
27
use Derafu\Lib\Core\Foundation\Abstract\AbstractWorker;
28
use Derafu\Lib\Core\Package\Prime\Component\Template\Contract\RendererWorkerInterface;
29
use Mpdf\Mpdf;
30
use Mpdf\Output\Destination;
31
use Twig\Environment;
32
use Twig\Loader\FilesystemLoader;
33
34
/**
35
 * Worker de renderización de plantillas.
36
 *
37
 * Permite renderizar plantillas Twig a HTML y PDF.
38
 *
39
 * TODO: Ordenar código, usar configuraciones/opciones y separar en estrategias.
40
 */
41
class RendererWorker extends AbstractWorker implements RendererWorkerInterface
42
{
43
    /**
44
     * Formato por defecto que se debe utilizar.
45
     *
46
     * @var string
47
     */
48
    private string $defaultFormat = 'pdf';
49
50
    /**
51
     * Rutas donde están las plantillas.
52
     *
53
     * @var array
54
     */
55
    private array $paths = [];
56
57
    /**
58
     * Cargador de plantillas mediante el sistema de archivos para Twig.
59
     */
60
    private FilesystemLoader $filesystemLoader;
61
62
    /**
63
     * Renderizador de plantillas HTML con Twig.
64
     *
65
     * @var Environment
66
     */
67
    private Environment $twig;
68
69
    /**
70
     * Constructor del worker.
71
     *
72
     * @param string|array $paths Rutas dónde se buscarán las plantillas.
73
     */
74 1
    public function __construct(string|array $paths = [])
75
    {
76 1
        $this->paths = is_string($paths) ? [$paths] : $paths;
0 ignored issues
show
introduced by
The condition is_string($paths) is always false.
Loading history...
77
    }
78
79
    /**
80
     * {@inheritdoc}
81
     */
82 1
    public function render(string $template, array $data = []): string
83
    {
84
        // Formato en el que se renderizará la plantilla.
85 1
        $format = $data['options']['format'] ?? $this->defaultFormat;
86
87
        // Renderizar HTML de la plantilla Twig.
88 1
        $html = $this->renderHtml($template, $data);
89
90
        // Si el formato solicitado es HTML se retorna directamente.
91 1
        if ($format === 'html') {
92
            return $html;
93
        }
94
95
        // Renderizar el PDF a partir del HTML.
96 1
        $configPdf = $data['options']['config']['pdf'] ?? [];
97 1
        $pdf = $this->renderPdf($html, $configPdf);
98
99
        // Entregar el contenido del PDF renderizado.
100 1
        return $pdf;
101
    }
102
103
    /**
104
     * Entrega la instancia de Twig.
105
     *
106
     * Este método evita crearla en el constructor y se crea solo cuando
107
     * realmente se utiliza. Útil porque los workers son lazy services.
108
     *
109
     * @return Environment
110
     */
111 1
    private function getTwig(): Environment
112
    {
113 1
        if (!isset($this->twig)) {
114 1
            $this->twig = new Environment($this->getFilesystemLoader());
115
        }
116
117 1
        return $this->twig;
118
    }
119
120
    /**
121
     * Entrega la instancia del cargador de plantillas desde el sistema de
122
     * archivos.
123
     *
124
     * @return FilesystemLoader
125
     */
126 1
    private function getFilesystemLoader(): FilesystemLoader
127
    {
128 1
        if (!isset($this->filesystemLoader)) {
129 1
            $this->filesystemLoader = new FilesystemLoader($this->paths);
130
        }
131
132 1
        return $this->filesystemLoader;
133
    }
134
135
    /**
136
     * Renderiza una plantilla Twig en HTML.
137
     *
138
     * @param string $template Plantilla Twig a renderizar.
139
     * @param array $data Datos que se pasarán a la plantilla Twig.
140
     * @return string Código HTML con el renderizado de la plantilla Twig.
141
     */
142 1
    private function renderHtml(string $template, array $data): string
143
    {
144
        // Resolver plantilla.
145 1
        $template = $this->resolveTemplate($template);
146
147
        // Renderizar la plantilla.
148 1
        return $this->getTwig()->render($template, $data);
149
    }
150
151
    /**
152
     * Renderiza un HTML en un documento PDF.
153
     *
154
     * El renderizado se realiza a partir de un HTML previamente renderizado que
155
     * será pasado a PDF.
156
     */
157 1
    private function renderPdf(string $html, array $config): string
158
    {
159
        // Generar el PDF con mPDF.
160 1
        $mpdf = new Mpdf();
161 1
        $mpdf->WriteHTML($html);
162
163
        // Obtener el contenido del PDF.
164 1
        $pdf = $mpdf->Output('', Destination::STRING_RETURN);
165
166
        // Entregar el contenido del PDF.
167 1
        return $pdf;
168
    }
169
170
    /**
171
     * Resuelve la plantilla que se está solicitando.
172
     *
173
     * Se encarga de:
174
     *
175
     *   - Agregar la extensión a la plantilla.
176
     *   - Agregar el directorio si se pasó una ruta absoluta de la plantilla.
177
     *
178
     * @param string $template
179
     * @return string
180
     */
181 1
    private function resolveTemplate(string $template): string
182
    {
183
        // Agregar extensión.
184 1
        if (!str_ends_with($template, '.html.twig')) {
185 1
            $template .= '.html.twig';
186
        }
187
188
        // Agregar el directorio si se pasó una ruta absoluta.
189 1
        if ($template[0] === '/') {
190 1
            $dir = dirname($template);
191 1
            $this->getFilesystemLoader()->addPath($dir);
192 1
            $template = basename($template);
193
        }
194
195
        // Entregar nombre de la plantilla.
196 1
        return $template;
197
    }
198
}
199