4 * This class provides the common functionality for creating PDF letter for
5 * one or a group of contact ids.
7 class CRM_Contribute_Form_Task_PDFLetterCommon
extends CRM_Contact_Form_Task_PDFLetterCommon
{
10 * Build the form object.
12 * @var CRM_Core_Form $form
14 public static function buildQuickForm(&$form) {
15 // use contact form as a base
16 CRM_Contact_Form_Task_PDFLetterCommon
::buildQuickForm($form);
18 // Contribute PDF tasks allow you to email as well, so we need to add email address to those forms
19 $form->add('select', 'from_email_address', ts('From Email Address'), $form->_fromEmails
, TRUE);
20 parent
::buildQuickForm($form);
24 * Process the form after the input has been submitted and validated.
26 * @param CRM_Contribute_Form_Task $form
27 * @param array $formValues
29 public static function postProcess(&$form, $formValues = NULL) {
30 if (empty($formValues)) {
31 $formValues = $form->controller
->exportValues($form->getName());
33 list($formValues, $categories, $html_message, $messageToken, $returnProperties) = self
::processMessageTemplate($formValues);
36 if (!empty($formValues['email_options'])) {
37 $returnProperties['email'] = $returnProperties['on_hold'] = $returnProperties['is_deceased'] = $returnProperties['do_not_email'] = 1;
39 'subject' => $formValues['subject'] ??
NULL,
40 'from' => $formValues['from_email_address'] ??
NULL,
43 $emailParams['from'] = CRM_Utils_Mail
::formatFromAddress($emailParams['from']);
45 // We need display_name for emailLetter() so add to returnProperties here
46 $returnProperties['display_name'] = 1;
47 if (stristr($formValues['email_options'], 'pdfemail')) {
52 $receipt_update = $formValues['receipt_update'] ??
FALSE;
53 $thankyou_update = $formValues['thankyou_update'] ??
FALSE;
54 $nowDate = date('YmdHis');
55 $receipts = $thanks = $emailed = 0;
57 $task = 'CRM_Contribution_Form_Task_PDFLetterCommon';
58 $realSeparator = ', ';
61 'tr' => '</td></tr><tr><td>',
63 //the original thinking was mutliple options - but we are going with only 2 (comma & td) for now in case
64 // there are security (& UI) issues we need to think through
65 if (isset($formValues['group_by_separator'])) {
66 if (in_array($formValues['group_by_separator'], ['td', 'tr'])) {
67 $realSeparator = $tableSeparators[$formValues['group_by_separator']];
69 elseif ($formValues['group_by_separator'] == 'br') {
70 $realSeparator = "<br />";
73 // a placeholder in case the separator is common in the string - e.g ', '
74 $separator = '****~~~~';
75 $groupBy = $formValues['group_by'];
77 // skip some contacts ?
78 $skipOnHold = $form->skipOnHold ??
FALSE;
79 $skipDeceased = $form->skipDeceased ??
TRUE;
80 $contributionIDs = $form->getVar('_contributionIds');
81 if ($form->_includesSoftCredits
) {
82 //@todo - comment on what is stored there
83 $contributionIDs = $form->getVar('_contributionContactIds');
85 list($contributions, $contacts) = self
::buildContributionArray($groupBy, $contributionIDs, $returnProperties, $skipOnHold, $skipDeceased, $messageToken, $task, $separator, $form->_includesSoftCredits
);
87 $contactHtml = $emailedHtml = [];
88 foreach ($contributions as $contributionId => $contribution) {
89 $contact = &$contacts[$contribution['contact_id']];
93 $groupByID = empty($contribution[$groupBy]) ?
0 : $contribution[$groupBy];
94 $contribution = $contact['combined'][$groupBy][$groupByID];
98 if (empty($groupBy) ||
empty($contact['is_sent'][$groupBy][$groupByID])) {
99 $html[$contributionId] = self
::generateHtml($contact, $contribution, $groupBy, $contributions, $realSeparator, $tableSeparators, $messageToken, $html_message, $separator, $grouped, $groupByID);
100 $contactHtml[$contact['contact_id']][] = $html[$contributionId];
101 if (!empty($formValues['email_options'])) {
102 if (self
::emailLetter($contact, $html[$contributionId], $isPDF, $formValues, $emailParams)) {
104 if (!stristr($formValues['email_options'], 'both')) {
105 $emailedHtml[$contributionId] = TRUE;
109 $contact['is_sent'][$groupBy][$groupByID] = TRUE;
111 // Update receipt/thankyou dates
112 $contributionParams = ['id' => $contributionId];
113 if ($receipt_update) {
114 $contributionParams['receipt_date'] = $nowDate;
116 if ($thankyou_update) {
117 $contributionParams['thankyou_date'] = $nowDate;
119 if ($receipt_update ||
$thankyou_update) {
120 civicrm_api3('Contribution', 'create', $contributionParams);
121 $receipts = ($receipt_update ?
$receipts +
1 : $receipts);
122 $thanks = ($thankyou_update ?
$thanks +
1 : $thanks);
126 $contactIds = array_keys($contacts);
127 self
::createActivities($form, $html_message, $contactIds, CRM_Utils_Array
::value('subject', $formValues, ts('Thank you letter')), CRM_Utils_Array
::value('campaign_id', $formValues), $contactHtml);
128 $html = array_diff_key($html, $emailedHtml);
130 if (!empty($formValues['is_unit_test'])) {
136 $type = $formValues['document_type'];
138 if ($type == 'pdf') {
139 CRM_Utils_PDF_Utils
::html2pdf($html, "CiviLetter.pdf", FALSE, $formValues);
142 CRM_Utils_PDF_Document
::html2doc($html, "CiviLetter.$type", $formValues);
146 $form->postProcessHook();
149 $updateStatus = ts('Receipts have been emailed to %1 contributions.', [1 => $emailed]);
152 $updateStatus = ts('Receipt date has been updated for %1 contributions.', [1 => $receipts]);
155 $updateStatus .= ' ' . ts('Thank-you date has been updated for %1 contributions.', [1 => $thanks]);
159 CRM_Core_Session
::setStatus($updateStatus);
162 // ie. we have only sent emails - lets no show a white screen
163 CRM_Utils_System
::civiExit();
168 * Check whether any of the tokens exist in the html outside a table cell.
169 * If they do the table cell separator is not supported (return false)
170 * At this stage we are only anticipating contributions passed in this way but
171 * it would be easy to add others
177 public static function isValidHTMLWithTableSeparator($tokens, $html) {
178 $relevantEntities = ['contribution'];
179 foreach ($relevantEntities as $entity) {
180 if (isset($tokens[$entity]) && is_array($tokens[$entity])) {
181 foreach ($tokens[$entity] as $token) {
182 if (!self
::isHtmlTokenInTableCell($token, $entity, $html)) {
192 * Check that the token only appears in a table cell. The '</td><td>' separator cannot otherwise work
193 * Calculate the number of times it appears IN the cell & the number of times it appears - should be the same!
195 * @param string $token
196 * @param string $entity
197 * @param string $textToSearch
201 public static function isHtmlTokenInTableCell($token, $entity, $textToSearch) {
202 $tokenToMatch = $entity . '\.' . $token;
203 $pattern = '|<td(?![\w-])((?!</td>).)*\{' . $tokenToMatch . '\}.*?</td>|si';
204 $within = preg_match_all($pattern, $textToSearch);
205 $total = preg_match_all("|{" . $tokenToMatch . "}|", $textToSearch);
206 return ($within == $total);
211 * @param string $html_message
212 * @param array $contact
213 * @param array $contribution
214 * @param array $messageToken
215 * @param bool $grouped
216 * Does this letter represent more than one contribution.
217 * @param string $separator
218 * What is the preferred letter separator.
221 private static function resolveTokens($html_message, $contact, $contribution, $messageToken, $grouped, $separator) {
222 $categories = self
::getTokenCategories();
223 $domain = CRM_Core_BAO_Domain
::getDomain();
224 $tokenHtml = CRM_Utils_Token
::replaceDomainTokens($html_message, $domain, TRUE, $messageToken);
225 $tokenHtml = CRM_Utils_Token
::replaceContactTokens($tokenHtml, $contact, TRUE, $messageToken);
227 $tokenHtml = CRM_Utils_Token
::replaceMultipleContributionTokens($separator, $tokenHtml, $contribution, TRUE, $messageToken);
230 // no change to normal behaviour to avoid risk of breakage
231 $tokenHtml = CRM_Utils_Token
::replaceContributionTokens($tokenHtml, $contribution, TRUE, $messageToken);
233 $tokenHtml = CRM_Utils_Token
::replaceHookTokens($tokenHtml, $contact, $categories, TRUE);
234 if (defined('CIVICRM_MAIL_SMARTY') && CIVICRM_MAIL_SMARTY
) {
235 $smarty = CRM_Core_Smarty
::singleton();
236 // also add the tokens to the template
237 $smarty->assign_by_ref('contact', $contact);
238 $tokenHtml = $smarty->fetch("string:$tokenHtml");
244 * Generate the contribution array from the form, we fill in the contact details and determine any aggregation
245 * around contact_id of contribution_recur_id
247 * @param string $groupBy
248 * @param array $contributionIDs
249 * @param array $returnProperties
250 * @param bool $skipOnHold
251 * @param bool $skipDeceased
252 * @param array $messageToken
253 * @param string $task
254 * @param string $separator
255 * @param bool $isIncludeSoftCredits
259 public static function buildContributionArray($groupBy, $contributionIDs, $returnProperties, $skipOnHold, $skipDeceased, $messageToken, $task, $separator, $isIncludeSoftCredits) {
260 $contributions = $contacts = [];
261 foreach ($contributionIDs as $item => $contributionId) {
262 $contribution = CRM_Contribute_BAO_Contribution
::getContributionTokenValues($contributionId, $messageToken)['values'][$contributionId];
263 $contribution['campaign'] = $contribution['contribution_campaign_title'] ??
NULL;
264 $contributions[$contributionId] = $contribution;
266 if ($isIncludeSoftCredits) {
267 //@todo find out why this happens & add comments
268 list($contactID) = explode('-', $item);
269 $contactID = (int) $contactID;
272 $contactID = $contribution['contact_id'];
274 if (!isset($contacts[$contactID])) {
275 $contacts[$contactID] = [];
276 $contacts[$contactID]['contact_aggregate'] = 0;
277 $contacts[$contactID]['combined'] = $contacts[$contactID]['contribution_ids'] = [];
280 $contacts[$contactID]['contact_aggregate'] +
= $contribution['total_amount'];
281 $groupByID = empty($contribution[$groupBy]) ?
0 : $contribution[$groupBy];
283 $contacts[$contactID]['contribution_ids'][$groupBy][$groupByID][$contributionId] = TRUE;
284 if (!isset($contacts[$contactID]['combined'][$groupBy]) ||
!isset($contacts[$contactID]['combined'][$groupBy][$groupByID])) {
285 $contacts[$contactID]['combined'][$groupBy][$groupByID] = $contribution;
286 $contacts[$contactID]['aggregates'][$groupBy][$groupByID] = $contribution['total_amount'];
289 $contacts[$contactID]['combined'][$groupBy][$groupByID] = self
::combineContributions($contacts[$contactID]['combined'][$groupBy][$groupByID], $contribution, $separator);
290 $contacts[$contactID]['aggregates'][$groupBy][$groupByID] +
= $contribution['total_amount'];
293 // Assign the available contributions before calling tokens so hooks parsing smarty can access it.
294 // Note that in core code you can only use smarty here if enable if for the whole site, incl
295 // CiviMail, with a big performance impact.
296 // Hooks allow more nuanced smarty usage here.
297 CRM_Core_Smarty
::singleton()->assign('contributions', $contributions);
298 foreach ($contacts as $contactID => $contact) {
299 $tokenResolvedContacts = CRM_Utils_Token
::getTokenDetails(['contact_id' => $contactID],
307 $contacts[$contactID] = array_merge($tokenResolvedContacts[0][$contactID], $contact);
309 return [$contributions, $contacts];
313 * We combine the contributions by adding the contribution to each field with the separator in
314 * between the existing value and the new one. We put the separator there even if empty so it is clear what the
315 * value for previous contributions was
317 * @param array $existing
318 * @param array $contribution
319 * @param string $separator
323 public static function combineContributions($existing, $contribution, $separator) {
324 foreach ($contribution as $field => $value) {
325 $existing[$field] = isset($existing[$field]) ?
$existing[$field] . $separator : '';
326 $existing[$field] .= $value;
332 * We are going to retrieve the combined contribution and if smarty mail is enabled we
333 * will also assign an array of contributions for this contact to the smarty template
335 * @param array $contact
336 * @param array $contributions
338 * @param int $groupByID
340 public static function assignCombinedContributionValues($contact, $contributions, $groupBy, $groupByID) {
341 CRM_Core_Smarty
::singleton()->assign('contact_aggregate', $contact['contact_aggregate']);
342 CRM_Core_Smarty
::singleton()
343 ->assign('contributions', array_intersect_key($contributions, $contact['contribution_ids'][$groupBy][$groupByID]));
344 CRM_Core_Smarty
::singleton()->assign('contribution_aggregate', $contact['aggregates'][$groupBy][$groupByID]);
351 * @param array $contact
352 * @param string $html
355 * @param array $format
356 * @param array $params
360 public static function emailLetter($contact, $html, $is_pdf, $format = [], $params = []) {
362 if (empty($contact['email'])) {
365 $mustBeEmpty = ['do_not_email', 'is_deceased', 'on_hold'];
366 foreach ($mustBeEmpty as $emptyField) {
367 if (!empty($contact[$emptyField])) {
373 'toName' => $contact['display_name'],
374 'toEmail' => $contact['email'],
378 if (empty($params['from'])) {
379 $emails = CRM_Core_BAO_Email
::getFromEmail();
380 $emails = array_keys($emails);
381 $defaults['from'] = array_pop($emails);
384 $defaults['from'] = $params['from'];
386 if (!empty($params['subject'])) {
387 $defaults['subject'] = $params['subject'];
390 $defaults['subject'] = ts('Thank you for your contribution/s');
393 $defaults['html'] = ts('Please see attached');
394 $defaults['attachments'] = [CRM_Utils_Mail
::appendPDF('ThankYou.pdf', $html, $format)];
396 $params = array_merge($defaults);
397 return CRM_Utils_Mail
::send($params);
399 catch (CRM_Core_Exception
$e) {
407 * @param $contribution
409 * @param $contributions
410 * @param $realSeparator
411 * @param $tableSeparators
412 * @param $messageToken
413 * @param $html_message
416 * @param bool $grouped
417 * @param int $groupByID
421 protected static function generateHtml(&$contact, $contribution, $groupBy, $contributions, $realSeparator, $tableSeparators, $messageToken, $html_message, $separator, $grouped, $groupByID) {
422 static $validated = FALSE;
425 self
::assignCombinedContributionValues($contact, $contributions, $groupBy, $groupByID);
427 if (empty($groupBy) ||
empty($contact['is_sent'][$groupBy][$groupByID])) {
428 if (!$validated && in_array($realSeparator, $tableSeparators) && !self
::isValidHTMLWithTableSeparator($messageToken, $html_message)) {
429 $realSeparator = ', ';
430 CRM_Core_Session
::setStatus(ts('You have selected the table cell separator, but one or more token fields are not placed inside a table cell. This would result in invalid HTML, so comma separators have been used instead.'));
433 $html = str_replace($separator, $realSeparator, self
::resolveTokens($html_message, $contact, $contribution, $messageToken, $grouped, $separator));