2021-11-25 23:12:32 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace BookStack\Entities\Tools;
|
|
|
|
|
2024-04-24 22:13:44 +08:00
|
|
|
use Knp\Snappy\Pdf as SnappyPdf;
|
2024-04-22 23:40:42 +08:00
|
|
|
use Dompdf\Dompdf;
|
2021-11-25 23:12:32 +08:00
|
|
|
|
|
|
|
class PdfGenerator
|
|
|
|
{
|
2022-01-25 01:24:00 +08:00
|
|
|
const ENGINE_DOMPDF = 'dompdf';
|
|
|
|
const ENGINE_WKHTML = 'wkhtml';
|
2024-04-22 23:40:42 +08:00
|
|
|
const ENGINE_COMMAND = 'command';
|
2022-01-25 01:24:00 +08:00
|
|
|
|
2021-11-25 23:12:32 +08:00
|
|
|
/**
|
|
|
|
* Generate PDF content from the given HTML content.
|
|
|
|
*/
|
|
|
|
public function fromHtml(string $html): string
|
|
|
|
{
|
2024-04-22 23:40:42 +08:00
|
|
|
$engine = $this->getActiveEngine();
|
|
|
|
|
|
|
|
if ($engine === self::ENGINE_WKHTML) {
|
2024-04-24 22:13:44 +08:00
|
|
|
return $this->renderUsingWkhtml($html);
|
2024-04-22 23:40:42 +08:00
|
|
|
} else if ($engine === self::ENGINE_COMMAND) {
|
|
|
|
// TODO - Support PDF command
|
|
|
|
return '';
|
2021-11-25 23:12:32 +08:00
|
|
|
}
|
|
|
|
|
2024-04-22 23:40:42 +08:00
|
|
|
return $this->renderUsingDomPdf($html);
|
2021-11-25 23:12:32 +08:00
|
|
|
}
|
2022-01-25 01:24:00 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the currently active PDF engine.
|
|
|
|
* Returns the value of an `ENGINE_` const on this class.
|
|
|
|
*/
|
|
|
|
public function getActiveEngine(): string
|
|
|
|
{
|
2024-04-24 22:13:44 +08:00
|
|
|
if ($this->getWkhtmlBinaryPath() && config('app.allow_untrusted_server_fetching') === true) {
|
2024-04-22 23:40:42 +08:00
|
|
|
return self::ENGINE_WKHTML;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self::ENGINE_DOMPDF;
|
|
|
|
}
|
|
|
|
|
2024-04-24 22:13:44 +08:00
|
|
|
protected function getWkhtmlBinaryPath(): string
|
|
|
|
{
|
|
|
|
$wkhtmlBinaryPath = config('exports.snappy.pdf_binary');
|
|
|
|
if (file_exists(base_path('wkhtmltopdf'))) {
|
|
|
|
$wkhtmlBinaryPath = base_path('wkhtmltopdf');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $wkhtmlBinaryPath ?: '';
|
|
|
|
}
|
|
|
|
|
2024-04-22 23:40:42 +08:00
|
|
|
protected function renderUsingDomPdf(string $html): string
|
|
|
|
{
|
|
|
|
$options = config('exports.dompdf');
|
|
|
|
$domPdf = new Dompdf($options);
|
|
|
|
$domPdf->setBasePath(base_path('public'));
|
2022-01-25 04:55:03 +08:00
|
|
|
|
2024-04-22 23:40:42 +08:00
|
|
|
$domPdf->loadHTML($this->convertEntities($html));
|
|
|
|
$domPdf->render();
|
|
|
|
|
|
|
|
return (string) $domPdf->output();
|
|
|
|
}
|
|
|
|
|
2024-04-24 22:13:44 +08:00
|
|
|
protected function renderUsingWkhtml(string $html): string
|
|
|
|
{
|
|
|
|
$snappy = new SnappyPdf($this->getWkhtmlBinaryPath());
|
|
|
|
$options = config('exports.snappy.options');
|
|
|
|
return $snappy->getOutputFromHtml($html, $options);
|
|
|
|
}
|
|
|
|
|
2024-04-22 23:40:42 +08:00
|
|
|
/**
|
|
|
|
* Taken from https://github.com/barryvdh/laravel-dompdf/blob/v2.1.1/src/PDF.php
|
|
|
|
* Copyright (c) 2021 barryvdh, MIT License
|
|
|
|
* https://github.com/barryvdh/laravel-dompdf/blob/v2.1.1/LICENSE
|
|
|
|
*/
|
|
|
|
protected function convertEntities(string $subject): string
|
|
|
|
{
|
|
|
|
$entities = [
|
|
|
|
'€' => '€',
|
|
|
|
'£' => '£',
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ($entities as $search => $replace) {
|
|
|
|
$subject = str_replace($search, $replace, $subject);
|
|
|
|
}
|
|
|
|
return $subject;
|
2022-01-25 01:24:00 +08:00
|
|
|
}
|
2021-11-29 05:01:35 +08:00
|
|
|
}
|