3 +--------------------------------------------------------------------+
4 | Copyright CiviCRM LLC. All rights reserved. |
6 | This work is published under the GNU AGPLv3 license with some |
7 | permitted exceptions and without any warranty. For full license |
8 | and copyright information, see https://civicrm.org/licensing |
9 +--------------------------------------------------------------------+
19 * @copyright CiviCRM LLC https://civicrm.org/licensing
22 class CRM_Utils_PDF_Utils
{
24 public static function enqueuePDF($pdf) {
26 $fname = time().'_lp.pdf';
27 file_put_contents('/tmp/'.$fname, $pdf);
28 header('Location: /civicrm/lp-setup?file='.$fname);
33 public static function latex2pdf(&$text, $fileName = 'civicrm.pdf', $output = FALSE, $pdfFormat = NULL) {
34 /* FIXME: get $paper_size, $orientation, $margins */
36 if (is_array($text)) {
40 $pages = array($text);
43 $head='\documentclass[12pt]{letter}
48 \usepackage[T1]{fontenc}
50 \usepackage{fontspec,xunicode}
51 %% VERY IMPORTANT. Configures supported languages and fonts to use for each one.
52 \usepackage[Latin, Hebrew, Arabics, CJK, Diacritics]{ucharclasses}
53 \newfontfamily{\normalfont}[Ligatures=TeX]{FreeSerif}
54 \newfontfamily{\cjkfont}{WenQuanYi Zen Hei}
55 \setDefaultTransitions{\normalfont}{}
56 \setTransitionsForLatin{\normalfont}{}
57 \setTransitionsForArabics{\normalfont}{}
58 \setTransitionsForCJK{\cjkfont}{}
59 \setTransitionsForDiacritics{\normalfont}{}
60 \setTransitionTo{Hebrew}{\normalfont}
61 \setmainfont{FreeSerif}
63 \newcommand{\fsfclosing}[1]{\par\nobreak\vspace{\parskip}
66 \ifx\@empty\fromaddress\else
67 \hspace*{\longindentation}\fi
68 \parbox{\indentedwidth}{\raggedright
69 \ignorespaces #1\\\\[1\medskipamount]
70 \hspace*{-0.25in}\includegraphics[scale=1.0]{/var/www/drupal-7.27/sites/all/modules/civicrm/sigjohns.pdf}
75 \else \fromsig \fi\strut}
77 \medskipamount=\parskip
81 \address{\vspace{0.05in}}
82 \signature{John Sullivan \\\\ Executive Director}
87 right = 1.0in]{geometry}
94 foreach ($pages as $page) {
99 $descriptorspec = array(
100 0 => array("pipe", "r"),
101 1 => array("pipe", "w")
106 $process = proc_open("/usr/local/bin/pdflatex_wrapper.sh", $descriptorspec, $pipes);
109 if (is_resource($process)) {
110 fwrite($pipes[0], $latex);
113 $pdf = stream_get_contents($pipes[1]);
116 CRM_Core_Error
::debug_log_message("ERROR creating PDF. Check /tmp/pdflatex_*");
123 header('Content-Type: application/pdf');
124 header('Content-Disposition: attachment; filename="' . $fileName . '"');
126 // quidam: comment previous line and uncomment next one during printing
127 //CRM_Utils_PDF_Utils::enqueuePDF($pdf);
134 * List of HTML snippets.
135 * @param string $fileName
136 * The logical filename to display.
137 * Ex: "HelloWorld.pdf".
138 * @param bool $output
139 * FALSE to display PDF. TRUE to return as string.
140 * @param null $pdfFormat
141 * Unclear. Possibly PdfFormat or formValues.
143 * @return string|void
145 public static function html2pdf($text, $fileName = 'civicrm.pdf', $output = FALSE, $pdfFormat = NULL) {
146 if (is_array($text)) {
152 // Get PDF Page Format
153 $format = CRM_Core_BAO_PdfFormat
::getDefaultValues();
154 if (is_array($pdfFormat)) {
155 // PDF Page Format parameters passed in
156 $format = array_merge($format, $pdfFormat);
158 elseif (!empty($pdfFormat)) {
159 // PDF Page Format ID passed in
160 $format = CRM_Core_BAO_PdfFormat
::getById($pdfFormat);
162 $paperSize = CRM_Core_BAO_PaperSize
::getByName($format['paper_size']);
163 $paper_width = self
::convertMetric($paperSize['width'], $paperSize['metric'], 'pt');
164 $paper_height = self
::convertMetric($paperSize['height'], $paperSize['metric'], 'pt');
165 // dompdf requires dimensions in points
166 $paper_size = [0, 0, $paper_width, $paper_height];
167 $orientation = CRM_Core_BAO_PdfFormat
::getValue('orientation', $format);
168 $metric = CRM_Core_BAO_PdfFormat
::getValue('metric', $format);
169 $t = CRM_Core_BAO_PdfFormat
::getValue('margin_top', $format);
170 $r = CRM_Core_BAO_PdfFormat
::getValue('margin_right', $format);
171 $b = CRM_Core_BAO_PdfFormat
::getValue('margin_bottom', $format);
172 $l = CRM_Core_BAO_PdfFormat
::getValue('margin_left', $format);
174 $margins = [$metric, $t, $r, $b, $l];
176 // Add a special region for the HTML header of PDF files:
177 $pdfHeaderRegion = CRM_Core_Region
::instance('export-document-header', FALSE);
178 $htmlHeader = ($pdfHeaderRegion) ?
$pdfHeaderRegion->render('', FALSE) : '';
183 <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\"/>
184 <style>@page { margin: {$t}{$metric} {$r}{$metric} {$b}{$metric} {$l}{$metric}; }</style>
185 <style type=\"text/css\">@import url(" . CRM_Core_Config
::singleton()->userFrameworkResourceURL
. "css/print.css);</style>
189 <div id=\"crm-container\">\n";
191 // Strip <html>, <header>, and <body> tags from each page
193 $htmlElementstoStrip = [
194 '<head[^>]*?>.*?</head>',
195 '<script[^>]*?>.*?</script>',
202 foreach ($pages as & $page) {
203 foreach ($htmlElementstoStrip as $pattern) {
204 $page = mb_eregi_replace($pattern, '', $page);
207 // Glue the pages together
208 $html .= implode("\n<div style=\"page-break-after: always\"></div>\n", $pages);
213 if (CRM_Core_Config
::singleton()->wkhtmltopdfPath
) {
214 return self
::_html2pdf_wkhtmltopdf($paper_size, $orientation, $margins, $html, $output, $fileName);
217 return self
::_html2pdf_dompdf($paper_size, $orientation, $html, $output, $fileName);
222 * Convert html to tcpdf.
226 * @param $orientation
231 * @param $stationery_path
233 public static function _html2pdf_tcpdf($paper_size, $orientation, $margins, $html, $output, $fileName, $stationery_path) {
234 CRM_Core_Error
::deprecatedFunctionWarning('CRM_Utils_PDF::_html2pdf_dompdf');
235 return self
::_html2pdf_dompdf($paper_size, $orientation, $margins, $html, $output, $fileName);
236 // Documentation on the TCPDF library can be found at: http://www.tcpdf.org
237 // This function also uses the FPDI library documented at: http://www.setasign.com/products/fpdi/about/
238 // Syntax borrowed from https://github.com/jake-mw/CDNTaxReceipts/blob/master/cdntaxreceipts.functions.inc
239 require_once 'tcpdf/tcpdf.php';
240 // This library is only in the 'packages' area as of version 4.5
241 require_once 'FPDI/fpdi.php';
243 $paper_size_arr = [$paper_size[2], $paper_size[3]];
245 $pdf = new TCPDF($orientation, 'pt', $paper_size_arr);
248 if (is_readable($stationery_path)) {
249 $pdf->SetStationery($stationery_path);
253 $pdf->SetKeywords('CiviCRM.org');
254 $pdf->setPageUnit($margins[0]);
255 $pdf->SetMargins($margins[4], $margins[1], $margins[2], TRUE);
257 $pdf->setJPEGQuality('100');
258 $pdf->SetAutoPageBreak(TRUE, $margins[3]);
268 // output the HTML content
269 $pdf->writeHTML($html, $ln, $fill, $reset_parm, $cell, $align);
271 // reset pointer to the last page
274 // close and output the PDF
276 $pdf_file = 'CiviLetter' . '.pdf';
277 $pdf->Output($pdf_file, 'D');
278 CRM_Utils_System
::civiExit();
283 * @param $orientation
286 * @param string $fileName
290 public static function _html2pdf_dompdf($paper_size, $orientation, $html, $output, $fileName) {
291 // CRM-12165 - Remote file support required for image handling.
292 $options = new Options();
293 $options->set('isRemoteEnabled', TRUE);
295 $dompdf = new DOMPDF($options);
296 $dompdf->set_paper($paper_size, $orientation);
297 $dompdf->load_html($html);
301 return $dompdf->output();
303 // CRM-19183 remove .pdf extension from filename
304 $fileName = basename($fileName, ".pdf");
305 if (CIVICRM_UF
=== 'UnitTests' && headers_sent()) {
306 // Streaming content will 'die' in unit tests unless ob_start()
308 throw new CRM_Core_Exception_PrematureExitException('_html2pdf_dompdf called', [
310 'fileName' => $fileName,
313 $dompdf->stream($fileName);
318 * @param $orientation
322 * @param string $fileName
324 public static function _html2pdf_wkhtmltopdf($paper_size, $orientation, $margins, $html, $output, $fileName) {
325 $config = CRM_Core_Config
::singleton();
326 // if the path doesn't exist fall back on the current backup which is DOMPDF.
327 if (!file_exists($config->wkhtmltopdfPath
)) {
328 return self
::_html2pdf_dompdf($paper_size, $orientation, $html, $output, $fileName);
330 require_once 'snappy/src/autoload.php';
331 $snappy = new Knp\Snappy\
Pdf($config->wkhtmltopdfPath
);
332 $snappy->setOption("page-width", $paper_size[2] . "pt");
333 $snappy->setOption("page-height", $paper_size[3] . "pt");
334 $snappy->setOption("orientation", $orientation);
335 $snappy->setOption("margin-top", $margins[1] . $margins[0]);
336 $snappy->setOption("margin-right", $margins[2] . $margins[0]);
337 $snappy->setOption("margin-bottom", $margins[3] . $margins[0]);
338 $snappy->setOption("margin-left", $margins[4] . $margins[0]);
339 $html = preg_replace('/{ }/', ' ', $html);
340 $pdf = $snappy->getOutputFromHtml($html);
345 CRM_Utils_System
::setHttpHeader('Content-Type', 'application/pdf');
346 CRM_Utils_System
::setHttpHeader('Content-Disposition', 'attachment; filename="' . $fileName . '"');
348 // quidam: comment previous line and uncomment next one during printing
349 //CRM_Utils_PDF_Utils::enqueuePDF($pdf);
355 * convert value from one metric to another.
360 * @param null $precision
364 public static function convertMetric($value, $from, $to, $precision = NULL) {
365 switch ($from . $to) {
414 if (!is_null($precision)) {
415 $value = round($value, $precision);