Files
espocrm/application/Espo/Core/Htmlizer/Htmlizer.php
Yuri Kuznetsov 1f4ce20c8b template renderer
2021-09-19 18:47:15 +03:00

753 lines
23 KiB
PHP

<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2021 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
* Website: https://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
*
* The interactive user interfaces in modified source and object code versions
* of this program must display Appropriate Legal Notices, as required under
* Section 5 of the GNU General Public License version 3.
*
* In accordance with Section 7(b) of the GNU General Public License version 3,
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
************************************************************************/
namespace Espo\Core\Htmlizer;
use Espo\Core\{
Utils\File\Manager as FileManager,
Utils\DateTime,
Utils\NumberUtil,
Utils\Config,
Utils\Language,
Utils\Metadata,
Utils\Log,
ServiceFactory,
Acl,
InjectableFactory,
};
use Espo\ORM\{
Entity,
EntityManager,
Collection,
};
use LightnCandy\LightnCandy as LightnCandy;
use StdClass;
use const JSON_PRESERVE_ZERO_FRACTION;
/**
* Generates an HTML for an entity. Used by Print-to-PDF, system email notifications.
*/
class Htmlizer
{
protected $fileManager;
protected $dateTime;
protected $config;
protected $acl;
protected $entityManager;
protected $metadata;
protected $language;
protected $serviceFactory;
private $log;
protected $injectableFactory;
public function __construct(
FileManager $fileManager,
DateTime $dateTime,
NumberUtil $number,
?Acl $acl = null,
?EntityManager $entityManager = null,
?Metadata $metadata = null,
?Language $language = null,
?Config $config = null,
?ServiceFactory $serviceFactory = null,
?Log $log = null,
?InjectableFactory $injectableFactory = null
) {
$this->fileManager = $fileManager;
$this->dateTime = $dateTime;
$this->number = $number;
$this->acl = $acl;
$this->entityManager = $entityManager;
$this->metadata = $metadata;
$this->language = $language;
$this->config = $config;
$this->serviceFactory = $serviceFactory;
$this->log = $log;
$this->injectableFactory = $injectableFactory;
}
/**
* Generate an HTML for entity by a given template.
*
* @param $cacheId @deprecated To be skipped..
* @param $additionalData Data will be passed to the template.
* @param $skipLinks Do not process related records.
*/
public function render(
?Entity $entity,
string $template,
?string $cacheId = null,
?array $additionalData = null,
bool $skipLinks = false
): string {
$template = str_replace('<tcpdf ', '', $template);
$helpers = $this->getHelpers();
$code = LightnCandy::compile($template, [
'flags' => LightnCandy::FLAG_HANDLEBARSJS | LightnCandy::FLAG_ERROR_EXCEPTION,
'helpers' => $this->getHelpers(),
]);
$renderer = LightnCandy::prepare($code);
if ($additionalData === null) {
$additionalData = $additionalData ?? [];
}
$data = $entity ?
$this->getDataFromEntity($entity, $skipLinks, 0, $template, $additionalData) :
$additionalData;
if (!array_key_exists('today', $data)) {
$data['today'] = $this->dateTime->getTodayString();
$data['today_RAW'] = date('Y-m-d');
}
if (!array_key_exists('now', $data)) {
$data['now'] = $this->dateTime->getNowString();
$data['now_RAW'] = date('Y-m-d H:i:s');
}
$data['__injectableFactory'] = $this->injectableFactory;
$data['__config'] = $this->config;
$data['__dateTime'] = $this->dateTime;
$data['__metadata'] = $this->metadata;
$data['__entityManager'] = $this->entityManager;
$data['__language'] = $this->language;
$data['__serviceFactory'] = $this->serviceFactory;
$data['__log'] = $this->log;
$data['__entityType'] = $entity->getEntityType();
$html = $renderer($data);
$html = str_replace('?entryPoint=attachment&amp;', '?entryPoint=attachment&', $html);
if ($this->entityManager) {
$html = preg_replace_callback(
'/\?entryPoint=attachment\&id=([A-Za-z0-9]*)/',
function ($matches) {
$id = $matches[1];
$attachment = $this->entityManager->getEntity('Attachment', $id);
if ($attachment) {
$filePath = $this->entityManager
->getRepository('Attachment')
->getFilePath($attachment);
return $filePath;
}
},
$html
);
}
return $html;
}
protected function format($value)
{
if (is_float($value)) {
return $this->number->format($value, 2);
}
if (is_int($value)) {
return $this->number->format($value);
}
if (is_string($value)) {
return nl2br($value);
}
return $value;
}
protected function getDataFromEntity(
Entity $entity,
bool $skipLinks = false,
int $level = 0,
?string $template = null,
?array $additionalData = null
): array {
$entityType = $entity->getEntityType();
$data = get_object_vars($entity->getValueMap());
if ($additionalData) {
foreach ($additionalData as $k => $value) {
$data[$k] = $value;
}
}
$attributeList = $entity->getAttributeList();
$forbiddenAttributeList = [];
$skipAttributeList = [];
$forbiddenLinkList = [];
if ($this->acl) {
$forbiddenAttributeList = array_merge(
$this->acl->getScopeForbiddenAttributeList($entityType, 'read'),
$this->acl->getScopeRestrictedAttributeList(
$entityType,
['forbidden', 'internal', 'onlyAdmin']
)
);
$forbiddenLinkList = $this->acl->getScopeRestrictedLinkList(
$entity->getEntityType(),
['forbidden', 'internal', 'onlyAdmin']
);
}
$relationList = $entity->getRelationList();
if (!$skipLinks && $level === 0 && $this->entityManager && $entity->id) {
foreach ($relationList as $relation) {
$collection = null;
$orderData = $this->getRelationOrder($entity->getEntityType(), $relation);
if ($entity->hasLinkMultipleField($relation)) {
$collection = $this->entityManager
->getRDBRepository($entity->getEntityType())
->getRelation($entity, $relation)
->order($orderData)
->find();
}
else if (
$template &&
in_array(
$entity->getRelationType($relation),
['hasMany', 'manyMany', 'hasChildren']
) &&
mb_stripos($template, '{{#each '.$relation.'}}') !== false
) {
$limit = 100;
if ($this->config) {
$limit = $this->config->get('htmlizerLinkLimit') ?? $limit;
}
$collection = $this->entityManager
->getRDBRepository($entity->getEntityType())
->getRelation($entity, $relation)
->limit(0, $limit)
->order($orderData)
->find();
}
if ($collection) {
$data[$relation] = $collection;
}
}
}
foreach ($data as $key => $value) {
if ($value instanceof Collection) {
$skipAttributeList[] = $key;
$collection = $value;
$list = [];
foreach ($collection as $item) {
$list[] = $this->getDataFromEntity($item, $skipLinks, $level + 1);
}
$data[$key] = $list;
}
}
foreach ($attributeList as $attribute) {
if (in_array($attribute, $forbiddenAttributeList)) {
unset($data[$attribute]);
continue;
}
if (in_array($attribute, $skipAttributeList)) {
continue;
}
$type = $entity->getAttributeType($attribute);
$fieldType = $entity->getAttributeParam($attribute, 'fieldType');
if ($type == Entity::DATETIME) {
if (!empty($data[$attribute])) {
$data[$attribute . '_RAW'] = $data[$attribute];
$data[$attribute] = $this->dateTime->convertSystemDateTime($data[$attribute]);
}
}
else if ($type == Entity::DATE) {
if (!empty($data[$attribute])) {
$data[$attribute . '_RAW'] = $data[$attribute];
$data[$attribute] = $this->dateTime->convertSystemDate($data[$attribute]);
}
}
else if ($type == Entity::JSON_ARRAY) {
if (!empty($data[$attribute])) {
$list = $data[$attribute];
$newList = [];
foreach ($list as $item) {
$v = $item;
if ($item instanceof StdClass) {
$v = json_decode(json_encode($v, JSON_PRESERVE_ZERO_FRACTION), true);
}
if (is_array($v)) {
foreach ($v as $k => $w) {
$keyRaw = $k . '_RAW';
$v[$keyRaw] = $v[$k];
$v[$k] = $this->format($v[$k]);
}
}
$newList[] = $v;
}
$data[$attribute] = $newList;
}
}
else if ($type == Entity::JSON_OBJECT) {
if (!empty($data[$attribute])) {
$value = $data[$attribute];
if ($value instanceof StdClass) {
$data[$attribute] = json_decode(json_encode($value, JSON_PRESERVE_ZERO_FRACTION), true);
}
foreach ($data[$attribute] as $k => $w) {
$keyRaw = $k . '_RAW';
$data[$attribute][$keyRaw] = $data[$attribute][$k];
$data[$attribute][$k] = $this->format($data[$attribute][$k]);
}
}
}
else if ($type === Entity::PASSWORD) {
unset($data[$attribute]);
}
if ($fieldType === 'currency' && $this->metadata) {
if ($entity->getAttributeParam($attribute, 'attributeRole') === 'currency') {
$currencyValue = $data[$attribute] ?? null;
if ($currencyValue) {
$data[$attribute . 'Symbol'] =
$this->metadata->get(['app', 'currency', 'symbolMap', $currencyValue]);
}
}
}
if (array_key_exists($attribute, $data)) {
$keyRaw = $attribute . '_RAW';
if (!isset($data[$keyRaw])) {
$data[$keyRaw] = $data[$attribute];
}
$fieldType = $this->getFieldType($entity->getEntityType(), $attribute);
if ($fieldType === 'enum') {
if ($this->language) {
$data[$attribute] = $this->language->translateOption(
$data[$attribute], $attribute, $entity->getEntityType()
);
}
}
$data[$attribute] = $this->format($data[$attribute]);
}
}
if (!$skipLinks) {
foreach ($entity->getRelationList() as $relation) {
if (in_array($relation, $forbiddenLinkList)) {
continue;
}
$relationType = $entity->getRelationType($relation);
if (
$relationType === 'belongsTo' ||
$relationType === 'belongsToParent'
) {
$relatedEntity = $this->entityManager
->getRepository($entity->getEntityType())
->getRelation($entity, $relation)
->findOne();
if (!$relatedEntity) {
continue;
}
if ($this->acl) {
if (!$this->acl->check($relatedEntity, 'read')) {
continue;
}
}
$data[$relation] = $this->getDataFromEntity($relatedEntity, true, $level + 1);
}
}
}
return $data;
}
protected function getHelpers()
{
$helpers = [
'file' => function () {
$args = func_get_args();
$id = $args[0] ?? null;
if (!$id) {
return '';
}
return new LightnCandy\SafeString("?entryPoint=attachment&id=" . $id);
},
'pagebreak' => function () {
return new LightnCandy\SafeString('<br pagebreak="true">');
},
'imageTag' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
$field = $context['hash']['field'] ?? null;
if ($field) {
$id = $context['_this'][$field . 'Id'] ?? null;
}
else if (count($args) > 1) {
$id = $args[0];
}
if (!$id || !is_string($id)) {
return null;
}
$width = $context['hash']['width'] ?? null;
$height = $context['hash']['height'] ?? null;
$attributesPart = "";
if ($width) {
$attributesPart .= " width=\"" .strval($width) . "\"";
}
if ($height) {
$attributesPart .= " height=\"" .strval($height) . "\"";
}
$html = "<img src=\"?entryPoint=attachment&id={$id}\"{$attributesPart}>";
return new LightnCandy\SafeString($html);
},
'var' => function () {
$args = func_get_args();
$c = $args[1] ?? [];
$key = $args[0] ?? null;
if (is_null($key)) {
return null;
}
return $c[$key] ?? null;
},
'numberFormat' => function () {
$args = func_get_args();
if (count($args) !== 2) {
return null;
}
$context = $args[count($args) - 1];
$number = $args[0] ?? null;
if (is_null($number)) {
return '';
}
$decimals = $context['hash']['decimals'] ?? 0;
$decimalPoint = $context['hash']['decimalPoint'] ?? '.';
$thousandsSeparator = $context['hash']['thousandsSeparator'] ?? ',';
return number_format($number, $decimals, $decimalPoint, $thousandsSeparator);
},
'dateFormat' => function () {
$args = func_get_args();
if (count($args) !== 2) {
return null;
}
$context = $args[count($args) - 1];
$dateValue = $args[0];
$format = $context['hash']['format'] ?? null;
$timezone = $context['hash']['timezone'] ?? null;
$language = $context['hash']['language'] ?? null;
$dateTime = $context['data']['root']['__dateTime'];
if (strlen($dateValue) > 11) {
return $dateTime->convertSystemDateTime($dateValue, $timezone, $format, $language);
}
return $dateTime->convertSystemDate($dateValue, $format, $language);
},
'barcodeImage' => function () {
$args = func_get_args();
if (count($args) !== 2) {
return null;
}
$context = $args[count($args) - 1];
$value = $args[0];
$params = $context['hash'];
$params['value'] = $value;
$paramsString = urlencode(json_encode($params));
return new LightnCandy\SafeString("<barcodeimage data=\"{$paramsString}\"/>");
},
'ifEqual' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
if ($args[0] === $args[1]) {
return $context['fn']();
}
return $context['inverse'] ? $context['inverse']() : '';
},
'ifNotEqual' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
if ($args[0] !== $args[1]) {
return $context['fn']();
}
return $context['inverse'] ? $context['inverse']() : '';
},
'ifInArray' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
$array = $args[1] ?? [];
if (in_array($args[0], $array)) {
return $context['fn']();
}
return $context['inverse'] ? $context['inverse']() : '';
},
'ifMultipleOf' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
if ($args[0] % $args[1] === 0) {
return $context['fn']();
}
return $context['inverse'] ? $context['inverse']() : '';
},
'tableTag' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
$border = $context['hash']['border'] ?? '0.5pt';
$cellpadding = $context['hash']['cellpadding'] ?? '2';
$width = $context['hash']['width'] ?? null;
$attributesPart = "";
if ($width) {
$attributesPart .= " width=\"{$width}\"";
}
return
new LightnCandy\SafeString(
"<table border=\"{$border}\" cellpadding=\"{$cellpadding}\" {$attributesPart}>"
) .
$context['fn']() .
new LightnCandy\SafeString("</table>");
},
'tdTag' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
$align = strtolower($context['hash']['align'] ?? 'left');
if (!in_array($align, ['left', 'right', 'center'])) {
$align = 'left';
}
$width = $context['hash']['width'] ?? null;
$attributesPart = "align=\"{$align}\"";
if ($width) {
$attributesPart .= " width=\"{$width}\"";
}
return
new LightnCandy\SafeString("<td {$attributesPart}>") .
$context['fn']() .
new LightnCandy\SafeString("</td>");
},
'trTag' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
return
new LightnCandy\SafeString("<tr>") .
$context['fn']() .
new LightnCandy\SafeString("</tr>");
},
'checkboxTag' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
if (count($args) < 2) {
return null;
}
$color = $context['hash']['color'] ?? '#000';
$option = $context['hash']['option'] ?? null;
if (is_null($option)) {
return null;
}
$option = strval($option);
$list = $args[0] ?? [];
if (!is_array($list)) {
return null;
}
$css = "font-family: zapfdingbats; color: {$color}";
if (in_array($option, $list)) {
$html =
'<input type="checkbox" checked="checked" name="1" ' .
'readonly="true" value="1" style="'.$css.'">';
} else {
$html = '<input type="checkbox" name="1" readonly="true" value="1" style="color: '.$css.'">';
}
return new LightnCandy\SafeString($html);
},
];
if ($this->metadata) {
$additionalHelpers = $this->metadata->get(['app', 'templateHelpers']) ?? [];
$helpers = array_merge($helpers, $additionalHelpers);
}
return $helpers;
}
protected function getFieldType(string $entityType, string $field)
{
if (!$this->metadata) {
return null;
}
return $this->metadata->get(['entityDefs', $entityType, 'fields', $field, 'type']);
}
private function getRelationOrder(string $entityType, string $relation): array
{
$relationDefs = $this->entityManager
->getDefs()
->getEntity($entityType)
->getRelation($relation);
if (!$relationDefs->hasForeignEntityType()) {
return [];
}
$foreignEntityType = $relationDefs->getForeignEntityType();
$collectionParams = $this->entityManager
->getDefs()
->getEntity($foreignEntityType)
->getParam('collection') ?? [];
$order = $collectionParams['order'] ?? null;
$orderBy = $collectionParams['orderBy'] ?? null;
if ($order === null && $orderBy === null) {
return [];
}
return [[$orderBy, $order]];
}
}