diff --git a/CHANGELOG.md b/CHANGELOG.md index cb15cfcc89acae3f8471fcee65e69dd4812845b6..4842d09bf607c51228e07a5ee60314d4b37f04c3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,7 @@ * Added Is Exposed to Field Output Handlers. * Added Fallback Field Output Handler. +* Added Rewrite with Tokens Field Output Handler. # Version 1.118 diff --git a/Civi/DataProcessor/Factory.php b/Civi/DataProcessor/Factory.php index 9ce5a1267696497babebefe0c9e57e9eeb5a2ffb..15e10f667208c8c4a7781102212846594ee0446b 100644 --- a/Civi/DataProcessor/Factory.php +++ b/Civi/DataProcessor/Factory.php @@ -237,6 +237,7 @@ class Factory { $this->addOutputHandler('conditionally_map_strings', new Definition('Civi\DataProcessor\FieldOutputHandler\ConditionallyMapStrings'), E::ts('Compare the input with up to 5 strings and map the output accordingly.')); $this->addOutputHandler('contact_count_number_of_cases', new Definition('Civi\DataProcessor\FieldOutputHandler\ContactCountNumberOfCases'), E::ts('Count the number of cases related to the given CiviCRM ID.')); $this->addOutputHandler('fallback', new Definition('Civi\DataProcessor\FieldOutputHandler\FallbackFieldOutputHandler'), E::ts('Fallback field output')); + $this->addOutputHandler('token', new Definition('Civi\DataProcessor\FieldOutputHandler\TokenFieldOutputHandler'), E::ts('Rewrite with tokens')); $this->addFileFormat('csv', new Definition('Civi\DataProcessor\FileFormat\CSVFileFormat'), E::ts('CSV')); } diff --git a/Civi/DataProcessor/FieldOutputHandler/TokenFieldOutputHandler.php b/Civi/DataProcessor/FieldOutputHandler/TokenFieldOutputHandler.php new file mode 100644 index 0000000000000000000000000000000000000000..d80a3b7f1d961500fb133cf6b91b45aed5009ee4 --- /dev/null +++ b/Civi/DataProcessor/FieldOutputHandler/TokenFieldOutputHandler.php @@ -0,0 +1,161 @@ +<?php +/** + * Copyright (C) 2025 Jaap Jansma (jaap.jansma@civicoop.org) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program 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 Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see <https://www.gnu.org/licenses/>. + */ + +namespace Civi\DataProcessor\FieldOutputHandler; + +use Civi\DataProcessor\DataSpecification\FieldSpecification; +use CRM_Dataprocessor_ExtensionUtil as E; + +class TokenFieldOutputHandler extends AbstractFieldOutputHandler { + + /** + * @var \Civi\DataProcessor\DataSpecification\FieldSpecification + */ + protected $outputFieldSpec; + + protected $html; + + /** + * Initialize the processor + * + * @param String $alias + * @param String $title + * @param array $configuration + * @param \Civi\DataProcessor\ProcessorType\AbstractProcessorType $processorType + */ + public function initialize($alias, $title, $configuration) { + $this->outputFieldSpec = new FieldSpecification($alias, $this->getType(), $title); + $this->html = $configuration['html']; + } + + public function formatField($rawRecord, $formattedRecord) { + return new HTMLFieldOutput(); + } + + /** + * Returns the data type of this field + * + * @return String + */ + protected function getType() { + return 'String'; + } + + /** + * @return \Civi\DataProcessor\DataSpecification\FieldSpecification + */ + public function getOutputFieldSpecification() { + return $this->outputFieldSpec; + } + + /** + * Do post formatting. + * + * Post formatting is additional formatting of the formatted values of a + * record. + * + * @param \Civi\DataProcessor\FieldOutputHandler\FieldOutput[] $record + * + * @return \Civi\DataProcessor\FieldOutputHandler\FieldOutput[] + */ + public function postFormat(array $record): array { + $searchs = []; + $replcementsFormatted = []; + $replcementsRaw = []; + foreach($record as $fieldName => $value) { + $searchs[] = '{' . $fieldName . '}'; + $replcementsFormatted[] = $value->formattedValue; + $replcementsRaw[] = $value->rawValue; + } + $plaintext = \CRM_Utils_String::htmlToText($this->html); + $plaintext = str_replace($searchs, $replcementsRaw, $plaintext); + $html = str_replace($searchs, $replcementsFormatted, $this->html); + $record[$this->outputFieldSpec->alias] = new HTMLFieldOutput($plaintext); + $record[$this->outputFieldSpec->alias]->setHtmlOutput($html); + return $record; + } + + /** + * Returns true when this handler has additional configuration. + * + * @return bool + */ + public function hasConfiguration() { + return true; + } + + /** + * When this handler has additional configuration you can add + * the fields on the form with this function. + * + * @param \CRM_Core_Form $form + * @param array $field + */ + public function buildConfigurationForm(\CRM_Core_Form $form, $field=array()) { + $form->add('wysiwyg', 'html', E::ts('Rewrite HTML'), array('rows' => 6, 'cols' => 80, 'style="width: 100%;"')); + $fieldSelect = $this->getFieldOutputHandlers($field['data_processor_id']); + $form->assign('tokens', $fieldSelect); + if (isset($field['configuration'])) { + $configuration = $field['configuration']; + $defaults = []; + if (isset($configuration['html'])) { + $defaults['html'] = $configuration['html']; + } + if (isset($configuration['plaintext'])) { + $defaults['plaintext'] = $configuration['plaintext']; + } + $form->setDefaults($defaults); + } + } + + /** + * When this handler has configuration specify the template file name + * for the configuration form. + * + * @return false|string + */ + public function getConfigurationTemplateFileName() { + return "CRM/Dataprocessor/Form/Field/Configuration/TokenFieldOutputHandler.tpl"; + } + + + /** + * Process the submitted values and create a configuration array + * + * @param $submittedValues + * @return array + */ + public function processConfiguration($submittedValues) { + $configuration['html'] = $submittedValues['html']; + return $configuration; + } + + protected function getFieldOutputHandlers(int $dataProcessorId) { + $return = []; + $dataProcessor = civicrm_api3('DataProcessor', 'getsingle', array('id' => $dataProcessorId)); + $dataProcessorClass = \CRM_Dataprocessor_BAO_DataProcessor::dataProcessorToClass($dataProcessor); + $outputFieldHandlers = $dataProcessorClass->getDataFlow()->getOutputFieldHandlers(FALSE); + foreach ($outputFieldHandlers as $outputFieldHandler) { + if ($outputFieldHandler->getOutputFieldSpecification()->alias != $this->getOutputFieldSpecification()->alias) { + $return[$outputFieldHandler->getOutputFieldSpecification()->alias] = $outputFieldHandler->getOutputFieldSpecification()->title; + } + } + return $return; + } + +} diff --git a/templates/CRM/Dataprocessor/Form/Field/Configuration/TokenFieldOutputHandler.tpl b/templates/CRM/Dataprocessor/Form/Field/Configuration/TokenFieldOutputHandler.tpl new file mode 100644 index 0000000000000000000000000000000000000000..0b57a662f721ce214e22e6a363418f0dc917ee5d --- /dev/null +++ b/templates/CRM/Dataprocessor/Form/Field/Configuration/TokenFieldOutputHandler.tpl @@ -0,0 +1,17 @@ +{crmScope extensionKey='dataprocessor'} + {include file="CRM/Dataprocessor/Form/Field/Configuration/SimpleFieldOutputHandler.tpl"} + <div class="crm-section"> + <div class="label">{$form.html.label}</div> + <div class="content">{$form.html.html}</div> + <div class="clear"></div> + </div> + <div class="crm-section"> + <div class="label">{ts}Possible tokens{/ts}</div> + <div class="content"> + {foreach from=$tokens item=label key=token} + {literal}{{/literal}{$token}{literal}}{/literal} = {$label} <br /> + {/foreach} + </div> + <div class="clear"></div> + </div> +{/crmScope}