Allow some mail customization for each template via note_private
This commit is contained in:
parent
36a17c4590
commit
3f14164994
4 changed files with 125 additions and 41 deletions
|
@ -1,11 +1,13 @@
|
||||||
# CHANGELOG SENDRECURRINGINVOICEBYMAIL FOR <a href="https://www.dolibarr.org">DOLIBARR ERP CRM</a>
|
# CHANGELOG SENDRECURRINGINVOICEBYMAIL FOR <a href="https://www.dolibarr.org">DOLIBARR ERP CRM</a>
|
||||||
|
|
||||||
|
## 0.2.7
|
||||||
|
Add the possibility to overwrite some email fields (recipients, subject, body) for each template.
|
||||||
|
|
||||||
## 0.2.5
|
## 0.2.5
|
||||||
Little cleanup : no reload needed for last_main_doc.
|
Little cleanup : no reload needed for last_main_doc.
|
||||||
|
|
||||||
## 0.2.3
|
## 0.2.3
|
||||||
Renaming from 'sendfacrecmail' to 'sendrecurringinvoicebymail'.
|
Renaming from 'sendfacrecmail' to 'sendrecurringinvoicebymail'.
|
||||||
|
|
||||||
|
|
||||||
## 0.1.2
|
## 0.1.2
|
||||||
First working version.
|
First working version.
|
||||||
|
|
27
README.md
27
README.md
|
@ -6,6 +6,31 @@ This module send the PDF generated with recurring invoices by email to the clien
|
||||||
|
|
||||||
You can customize the mail template in Home > Setup > Emails > Email templates.
|
You can customize the mail template in Home > Setup > Emails > Email templates.
|
||||||
|
|
||||||
|
Beta - test in progress : you can also customize for each template invoice, by adding some of those blocks in the private notes of the template.
|
||||||
|
```
|
||||||
|
This is a good client (this is outside of the %%% blocks so it won't appear in the mails :)
|
||||||
|
|
||||||
|
%%% sendrecurringinvoicebymail::body
|
||||||
|
Hello dear client,
|
||||||
|
|
||||||
|
Please find attached... invoice __REF__...
|
||||||
|
|
||||||
|
__(Sincerely)__,
|
||||||
|
|
||||||
|
__MYCOMPANY_NAME__
|
||||||
|
%%%
|
||||||
|
|
||||||
|
%%% sendrecurringinvoicebymail::subject
|
||||||
|
My custom subject
|
||||||
|
%%%
|
||||||
|
%%% sendrecurringinvoicebymail::sendto
|
||||||
|
test1@example.org, "Mr. Test2" <test2@example.com>
|
||||||
|
%%%
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
It requires Dolibarr version 10.0 at least (first version with the 'cron/afterCreationOfRecurringInvoice()' hook).
|
It requires Dolibarr version 10.0 at least (first version with the 'cron/afterCreationOfRecurringInvoice()' hook).
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
|
@ -16,11 +41,11 @@ Other modules are available on <a href="https://www.dolistore.com" target="_new"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<!--
|
||||||
### Translations
|
### Translations
|
||||||
|
|
||||||
Translations can be define manually by editing files into directories *langs*.
|
Translations can be define manually by editing files into directories *langs*.
|
||||||
|
|
||||||
<!--
|
|
||||||
This module contains also a sample configuration for Transifex, under the hidden directory [.tx](.tx), so it is possible to manage translation using this service.
|
This module contains also a sample configuration for Transifex, under the hidden directory [.tx](.tx), so it is possible to manage translation using this service.
|
||||||
|
|
||||||
For more informations, see the [translator's documentation](https://wiki.dolibarr.org/index.php/Translator_documentation).
|
For more informations, see the [translator's documentation](https://wiki.dolibarr.org/index.php/Translator_documentation).
|
||||||
|
|
|
@ -82,17 +82,12 @@ class Actionssendrecurringinvoicebymail
|
||||||
|
|
||||||
$facturerec = $parameters['facturerec'];
|
$facturerec = $parameters['facturerec'];
|
||||||
|
|
||||||
// On n'envoie la facture que si elle est validée
|
// We only send the mail when the invoice is not a draft
|
||||||
if ($object->brouillon) {
|
if ($object->brouillon) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
// On n'envoie évidemment pas s'il n'y a pas d'adresse email renseignée
|
|
||||||
if (empty($object->thirdparty->email)) {
|
|
||||||
dol_syslog("Empty email for thirdparty " . $object->thirdparty->id . ". Not sending facturerec " . $facturerec->ref . " (id:" . $facturerec->id . ").");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// récupération du template du mail
|
// Fetch the mail template
|
||||||
// (pas très précise mais je commence à en avoir marre de creuser tout dolibarr pour trouver les bonnes fonctions...)
|
// (pas très précise mais je commence à en avoir marre de creuser tout dolibarr pour trouver les bonnes fonctions...)
|
||||||
$result = $this->db->query("SELECT * FROM " . MAIN_DB_PREFIX . "c_email_templates WHERE module = 'sendrecurringinvoicebymail' AND active = 1 AND enabled = '1' ORDER BY tms DESC LIMIT 1");
|
$result = $this->db->query("SELECT * FROM " . MAIN_DB_PREFIX . "c_email_templates WHERE module = 'sendrecurringinvoicebymail' AND active = 1 AND enabled = '1' ORDER BY tms DESC LIMIT 1");
|
||||||
if ( ! $result or ! ($template = $this->db->fetch_object($result))) {
|
if ( ! $result or ! ($template = $this->db->fetch_object($result))) {
|
||||||
|
@ -102,29 +97,44 @@ class Actionssendrecurringinvoicebymail
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Préparation des remplacements dans le sujet et le corps du mail
|
// Prepare the substitions for mail's subject and body
|
||||||
$substitutionarray = getCommonSubstitutionArray($langs, 0, null, $object);
|
$substitutionarray = getCommonSubstitutionArray($langs, 0, null, $object);
|
||||||
complete_substitutions_array($substitutionarray, $langs, $object); // lourd et n'a rien ajouté lors de mes tests
|
complete_substitutions_array($substitutionarray, $langs, $object); // lourd et n'a rien ajouté lors de mes tests
|
||||||
|
|
||||||
// Par contre, il nous manque quelques trucs utiles...
|
// Adding some useful substitions of our own...
|
||||||
if ( ! empty($object->linkedObjects['contrat'])) {
|
if ( ! empty($object->linkedObjects['contrat'])) {
|
||||||
$contrat = reset($object->linkedObjects['contrat']); // on prend le premier qui vient.
|
$contrat = reset($object->linkedObjects['contrat']); // no deep search, we take the first linked contract
|
||||||
$substitutionarray['__CONTRACT_REF__'] = $contrat->ref;
|
$substitutionarray['__CONTRACT_REF__'] = $contrat->ref;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialisations and substitutions
|
// Initialisations
|
||||||
$sendto = $object->thirdparty->name . ' <' . $object->thirdparty->email . '>';
|
$mail_data = array(
|
||||||
$from = $conf->global->MAIN_MAIL_EMAIL_FROM;
|
'sendto' => $object->thirdparty->name . ' <' . $object->thirdparty->email . '>',
|
||||||
$errorsTo = $conf->global->MAIN_MAIL_ERRORS_TO;
|
'from' => $conf->global->MAIN_MAIL_EMAIL_FROM,
|
||||||
$replyTo = $conf->global->MAIN_MAIL_ERRORS_TO;
|
'errorsTo' => $conf->global->MAIN_MAIL_ERRORS_TO,
|
||||||
$subject = make_substitutions($template->topic, $substitutionarray, $langs);
|
'replyTo' => $conf->global->MAIN_MAIL_ERRORS_TO,
|
||||||
$body = make_substitutions($template->content, $substitutionarray, $langs);
|
'subject' => $template->topic,
|
||||||
if (method_exists($object, 'makeSubstitution')) {
|
'body' => $template->content,
|
||||||
$subject = $object->makeSubstitution($subject);
|
);
|
||||||
$body = $object->makeSubstitution($body);
|
|
||||||
|
// If the invoice has some custom parameters (subject, body, sendto, ...)
|
||||||
|
$mail_data = array_merge($mail_data, $this->getCustomFieldsMail($object));
|
||||||
|
|
||||||
|
// Check that we have a recipient, to avoid some frequent error...
|
||||||
|
if (empty($mail_data['sendto'])) {
|
||||||
|
dol_syslog("Empty recipient for thirdparty " . $object->thirdparty->id . ". Not sending facturerec " . $facturerec->ref . " (id:" . $facturerec->id . ").");
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// On regarde si on doit joindre le fichier
|
// Make the substitutions
|
||||||
|
foreach (array('subject', 'body') as $key) {
|
||||||
|
$mail_data[$key] = make_substitutions($mail_data[$key], $substitutionarray, $langs);
|
||||||
|
if (method_exists($object, 'makeSubstitution')) {
|
||||||
|
$mail_data[$key] = $object->makeSubstitution($mail_data[$key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if we have to attach the file
|
||||||
$filePath = array();
|
$filePath = array();
|
||||||
$fileMime = array();
|
$fileMime = array();
|
||||||
$fileName = array();
|
$fileName = array();
|
||||||
|
@ -134,12 +144,12 @@ class Actionssendrecurringinvoicebymail
|
||||||
$fileName = array(basename($object->last_main_doc));
|
$fileName = array(basename($object->last_main_doc));
|
||||||
}
|
}
|
||||||
|
|
||||||
// envoi du mail
|
// At last, send the mail
|
||||||
$mailfile = new CMailFile(
|
$mailfile = new CMailFile(
|
||||||
$subject, // sujet
|
$mail_data['subject'],
|
||||||
$sendto, // destinataire
|
$mail_data['sendto'],
|
||||||
$from, // expéditeur
|
$mail_data['from'],
|
||||||
$body, // corps du mail
|
$mail_data['body'],
|
||||||
$filePath,
|
$filePath,
|
||||||
$fileMime,
|
$fileMime,
|
||||||
$fileName,
|
$fileName,
|
||||||
|
@ -147,12 +157,12 @@ class Actionssendrecurringinvoicebymail
|
||||||
'', // BCC
|
'', // BCC
|
||||||
0, //deliveryreceipt
|
0, //deliveryreceipt
|
||||||
0, //msgishtml
|
0, //msgishtml
|
||||||
$errorsTo,
|
$mail_data['errorsTo'],
|
||||||
'', // css
|
'', // css
|
||||||
'', // trackid
|
'', // trackid
|
||||||
'', // moreinheader
|
'', // moreinheader
|
||||||
'standard', // sendcontext
|
'standard', // sendcontext
|
||||||
$replyTo);
|
$mail_data['replyTo']);
|
||||||
|
|
||||||
if ($mailfile->sendfile()) {
|
if ($mailfile->sendfile()) {
|
||||||
dol_syslog("Success sending email for " . $facturerec->ref . " (id:" . $facturerec->id . ").");
|
dol_syslog("Success sending email for " . $facturerec->ref . " (id:" . $facturerec->id . ").");
|
||||||
|
@ -160,18 +170,18 @@ class Actionssendrecurringinvoicebymail
|
||||||
// Adds info to object for trigger
|
// Adds info to object for trigger
|
||||||
// (maybe make a copy of the object instead of modifying it directly ?)
|
// (maybe make a copy of the object instead of modifying it directly ?)
|
||||||
$object->email_msgid = $mailfile->msgid;
|
$object->email_msgid = $mailfile->msgid;
|
||||||
$object->email_from = $from;
|
$object->email_from = $mail_data['from'];
|
||||||
$object->email_subject = $subject;
|
$object->email_subject = $mail_data['subject'];
|
||||||
$object->email_to = $sendto;
|
$object->email_to = $mail_data['sendto'];
|
||||||
//$object->email_tocc = $sendtocc;
|
//$object->email_tocc = $sendtocc;
|
||||||
//$object->email_tobcc = $sendtobcc;
|
//$object->email_tobcc = $sendtobcc;
|
||||||
$object->actiontypecode = 'AC_OTH_AUTO';
|
$object->actiontypecode = 'AC_OTH_AUTO';
|
||||||
$object->actionmsg2=$langs->transnoentities('MailSentBy').' '.CMailFile::getValidAddress($from,4,0,1).' '.$langs->transnoentities('To').' '.CMailFile::getValidAddress($sendto,4,0,1);
|
$object->actionmsg2=$langs->transnoentities('MailSentBy').' '.CMailFile::getValidAddress($mail_data['from'],4,0,1).' '.$langs->transnoentities('To').' '.CMailFile::getValidAddress($mail_data['sendto'],4,0,1);
|
||||||
$object->actionmsg = $langs->transnoentities('MailFrom').': '.dol_escape_htmltag($from);
|
$object->actionmsg = $langs->transnoentities('MailFrom').': '.dol_escape_htmltag($mail_data['from']);
|
||||||
$object->actionmsg = dol_concatdesc($object->actionmsg, $langs->transnoentities('MailTo').': '.dol_escape_htmltag($sendto));
|
$object->actionmsg = dol_concatdesc($object->actionmsg, $langs->transnoentities('MailTo').': '.dol_escape_htmltag($mail_data['sendto']));
|
||||||
$object->actionmsg = dol_concatdesc($object->actionmsg, $langs->transnoentities('MailTopic') . ": " . $subject);
|
$object->actionmsg = dol_concatdesc($object->actionmsg, $langs->transnoentities('MailTopic') . ": " . $mail_data['subject']);
|
||||||
$object->actionmsg = dol_concatdesc($object->actionmsg, $langs->transnoentities('TextUsedInTheMessageBody') . ":");
|
$object->actionmsg = dol_concatdesc($object->actionmsg, $langs->transnoentities('TextUsedInTheMessageBody') . ":");
|
||||||
$object->actionmsg = dol_concatdesc($object->actionmsg, $body);
|
$object->actionmsg = dol_concatdesc($object->actionmsg, $mail_data['body']);
|
||||||
|
|
||||||
// Launch triggers
|
// Launch triggers
|
||||||
$interface = new Interfaces($this->db);
|
$interface = new Interfaces($this->db);
|
||||||
|
@ -186,4 +196,51 @@ class Actionssendrecurringinvoicebymail
|
||||||
return ($error ? -1 : 0);
|
return ($error ? -1 : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FIXME: For the time being, we abuse of note_private to store our customizations
|
||||||
|
*/
|
||||||
|
public function getCustomFieldsMail($object)
|
||||||
|
{
|
||||||
|
return $this->parseCustomFieldsMail($object->note_private);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FIXME: For the time being, we abuse of note_private to store our customizations
|
||||||
|
*
|
||||||
|
* This expect something like this in note_private:
|
||||||
|
* This is a good client... (other private infos)
|
||||||
|
* %%% sendrecurringinvoicebymail::subject
|
||||||
|
* New invoice __REF__
|
||||||
|
* %%%
|
||||||
|
* %%% sendrecurringinvoicebymail::sendto
|
||||||
|
* recipient1@example.org, recipient2@example.org
|
||||||
|
* %%%
|
||||||
|
* %%% sendrecurringinvoicebymail::body
|
||||||
|
* Hello dear client,
|
||||||
|
* Please find attached...
|
||||||
|
* %%%
|
||||||
|
*/
|
||||||
|
public function parseCustomFieldsMail($data)
|
||||||
|
{
|
||||||
|
$output = [];
|
||||||
|
|
||||||
|
// Remove eventual windows' "\r"
|
||||||
|
$data = str_replace("\r", "", $data);
|
||||||
|
|
||||||
|
$regexps = array(
|
||||||
|
'subject' => '/(^|\n)%%% sendrecurringinvoicebymail::subject\n(?<subject>.*)%%%(\n|$)/sU',
|
||||||
|
'body' => '/(^|\n)%%% sendrecurringinvoicebymail::body\n(?<body>.*)%%%(\n|$)/sU',
|
||||||
|
'sendto' => '/(^|\n)%%% sendrecurringinvoicebymail::sendto\n(?<sendto>.*)%%%(\n|$)/sU',
|
||||||
|
);
|
||||||
|
foreach ($regexps as $key => $r) {
|
||||||
|
$result_regexp = [];
|
||||||
|
if (preg_match_all($r, $data, $result_regexp)) {
|
||||||
|
$output[$key] = trim($result_regexp[$key][0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $output;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -69,10 +69,10 @@ class modsendrecurringinvoicebymail extends DolibarrModules
|
||||||
$this->editor_url = 'https://code.bugness.org/Dolibarr/sendrecurringinvoicebymail';
|
$this->editor_url = 'https://code.bugness.org/Dolibarr/sendrecurringinvoicebymail';
|
||||||
|
|
||||||
// Possible values for version are: 'development', 'experimental', 'dolibarr', 'dolibarr_deprecated' or a version string like 'x.y.z'
|
// Possible values for version are: 'development', 'experimental', 'dolibarr', 'dolibarr_deprecated' or a version string like 'x.y.z'
|
||||||
$this->version = '0.2.6';
|
$this->version = '0.2.7';
|
||||||
|
|
||||||
//Url to the file with your last numberversion of this module
|
//Url to the file with your last numberversion of this module
|
||||||
//$this->url_last_version = 'http://www.example.com/versionmodule.txt';
|
//$this->url_last_version = 'http://www.example.com/versionmodule.txt';
|
||||||
// Key used in llx_const table to save module status enabled/disabled (where SENDRECURRINGINVOICEBYMAIL is value of property name of module in uppercase)
|
// Key used in llx_const table to save module status enabled/disabled (where SENDRECURRINGINVOICEBYMAIL is value of property name of module in uppercase)
|
||||||
$this->const_name = 'MAIN_MODULE_'.strtoupper($this->name);
|
$this->const_name = 'MAIN_MODULE_'.strtoupper($this->name);
|
||||||
// Name of image file used for this module.
|
// Name of image file used for this module.
|
||||||
|
|
Loading…
Reference in a new issue