dolibarr 21.0.0-alpha
actions_massactions_mail.inc.php
1<?php
2/* Copyright (C) 2015-2017 Laurent Destailleur <eldy@users.sourceforge.net>
3 * Copyright (C) 2018-2021 Nicolas ZABOURI <info@inovea-conseil.com>
4 * Copyright (C) 2018 Juanjo Menent <jmenent@2byte.es>
5 * Copyright (C) 2019 Ferran Marcet <fmarcet@2byte.es>
6 * Copyright (C) 2019-2021 Frédéric France <frederic.france@netlogic.fr>
7 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 3 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program. If not, see <https://www.gnu.org/licenses/>.
21 * or see https://www.gnu.org/
22 */
23
30// $massaction must be defined
31// $objectclass and $objectlabel must be defined
32// $parameters, $object, $action must be defined for the hook.
33
34// $permissiontoread, $permissiontoadd, $permissiontodelete, $permissiontoclose may be defined
35// $uploaddir may be defined (example to $conf->project->dir_output."/";)
36// $toselect may be defined
37// $diroutputmassaction may be defined
38
39
40// Protection
41if (empty($objectclass) || empty($uploaddir)) {
42 dol_print_error(null, 'include of actions_massactions.inc.php is done but var $objectclass or $uploaddir was not defined');
43 exit;
44}
45
46'
47@phan-var-force string $massaction
48@phan-var-force string $objectclass
49@phan-var-force ?string $diroutputmassaction
50@phan-var-force ?string $uploaddir
51@phan-var-force string[] $toselect
52@phan-var-force array<string,mixed> $parameters
53';
54
55// Mass actions. Controls on number of lines checked.
56$maxformassaction = (!getDolGlobalString('MAIN_LIMIT_FOR_MASS_ACTIONS') ? 1000 : $conf->global->MAIN_LIMIT_FOR_MASS_ACTIONS);
57if (!empty($massaction) && is_array($toselect) && count($toselect) < 1) {
58 $error++;
59 setEventMessages($langs->trans("NoRecordSelected"), null, "warnings");
60}
61if (!$error && is_array($toselect) && count($toselect) > $maxformassaction) {
62 setEventMessages($langs->trans('TooManyRecordForMassAction', $maxformassaction), null, 'errors');
63 $error++;
64}
65
66if (!$error && $massaction == 'confirm_presend_attendees' && !GETPOST('sendmail')) { // If we do not choose button send (for example when we change template or limit), we must not send email, but keep on send email form
67 $massaction = 'presend_attendees';
68}
69
70if (!$error && $massaction == 'confirm_presend_attendees') {
71 $resaction = '';
72 $nbsent = 0;
73 $nbignored = 0;
74 $langs->load("mails");
75 include_once DOL_DOCUMENT_ROOT . '/core/lib/files.lib.php';
76
77 $listofobjectid = array();
78
79 $listofobjectref = array();
80 $oneemailperrecipient = (GETPOSTINT('oneemailperrecipient') ? 1 : 0);
81
82 $listofselectedid = array();
83 $listofselectedref = array();
84 if (!$error) {
85 require_once DOL_DOCUMENT_ROOT . '/eventorganization/class/conferenceorboothattendee.class.php';
86 $attendee = new ConferenceOrBoothAttendee($db);
87 $objecttmp = new $objectclass($db);
88 '@phan-var-force CommonObject $objecttmp';
89
90 foreach ($toselect as $toselectid) {
91 $result = $objecttmp->fetch($toselectid);
92 if ($result > 0) {
93 $attendees = $attendee->fetchAll();
94 if (is_array($attendees) && count($attendees) > 0) {
95 foreach ($attendees as $attmail) {
96 if (!empty($attmail->email)) {
97 $attmail->fetch_thirdparty();
98 $listofselectedid[$attmail->email] = $attmail;
99 $listofselectedref[$attmail->email] = $objecttmp;
100 }
101 }
102 }
103 }
104 }
105 }
106 '@phan-var-force CommonObject $objecttmp';
107 '@phan-var-force array<string,CommonObject> $listofselectedref';
108
109 // Check mandatory parameters
110 if (GETPOST('fromtype', 'alpha') === 'user' && empty($user->email)) {
111 $error++;
112 setEventMessages($langs->trans("NoSenderEmailDefined"), null, 'warnings');
113 $massaction = 'presend_attendees';
114 }
115
116 $receiver = GETPOST('receiver', 'alphawithlgt');
117 if (!is_array($receiver)) {
118 if (empty($receiver) || $receiver == '-1') {
119 $receiver = array();
120 } else {
121 $receiver = array($receiver);
122 }
123 }
124 if (!trim(GETPOST('sendto', 'alphawithlgt')) && count($receiver) == 0 && count($listofselectedid) == 0) { // if only one recipient, receiver is mandatory
125 $error++;
126 setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("Recipient")), null, 'warnings');
127 $massaction = 'presend_attendees';
128 }
129
130 if (!GETPOST('subject', 'restricthtml')) {
131 $error++;
132 setEventMessages($langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("MailTopic")), null, 'warnings');
133 $massaction = 'presend_attendees';
134 }
135
136 if (!$error && !empty($listofselectedid)) {
137 $objecttmp->fetch_thirdparty();
138 foreach ($listofselectedid as $email => $attendees) {
139 $sendto = '';
140 $sendtocc = '';
141 $sendtobcc = '';
142 $sendtoid = array();
143
144 // Define $sendto
145 $sendto = $attendees->thirdparty->name . '<' . trim($attendees->email) . '>';
146
147 // Define $sendtocc
148 $receivercc = GETPOST('receivercc', 'alphawithlgt');
149 if (!is_array($receivercc)) {
150 if ($receivercc == '-1') {
151 $receivercc = array();
152 } else {
153 $receivercc = array($receivercc);
154 }
155 }
156 $tmparray = array();
157 if (trim(GETPOST('sendtocc', 'alphawithlgt'))) {
158 $tmparray[] = trim(GETPOST('sendtocc', 'alphawithlgt'));
159 }
160 $sendtocc = implode(',', $tmparray);
161
162
163 $langs->load("commercial");
164
165 $reg = array();
166 $fromtype = GETPOST('fromtype');
167 if ($fromtype === 'user') {
168 $from = $user->getFullName($langs) . ' <' . $user->email . '>';
169 } elseif ($fromtype === 'company') {
170 $from = getDolGlobalString('MAIN_INFO_SOCIETE_NOM') . ' <' . getDolGlobalString('MAIN_INFO_SOCIETE_MAIL') . '>';
171 } elseif (preg_match('/user_aliases_(\d+)/', $fromtype, $reg)) {
172 $tmp = explode(',', $user->email_aliases);
173 $from = trim($tmp[((int) $reg[1] - 1)]);
174 } elseif (preg_match('/global_aliases_(\d+)/', $fromtype, $reg)) {
175 $tmp = explode(',', getDolGlobalString('MAIN_INFO_SOCIETE_MAIL_ALIASES'));
176 $from = trim($tmp[((int) $reg[1] - 1)]);
177 } elseif (preg_match('/senderprofile_(\d+)_(\d+)/', $fromtype, $reg)) {
178 $sql = "SELECT rowid, label, email FROM " . MAIN_DB_PREFIX . "c_email_senderprofile WHERE rowid = " . (int) $reg[1];
179 $resql = $db->query($sql);
180 $obj = $db->fetch_object($resql);
181 if ($obj) {
182 $from = dol_string_nospecial($obj->label, ' ', array(",")) . ' <' . $obj->email . '>';
183 }
184 } else {
185 $from = dol_string_nospecial(GETPOST('fromname'), ' ', array(",")) . ' <' . GETPOST('frommail') . '>';
186 }
187
188 $replyto = $from;
189 $subject = GETPOST('subject', 'restricthtml');
190 $message = GETPOST('message', 'restricthtml');
191
192 $sendtobcc = GETPOST('sendtoccc', 'alphawithlgt');
193
194 // $objecttmp is a real object or an empty object if we choose to send one email per thirdparty instead of one per object
195 // Make substitution in email content
196 $substitutionarray = getCommonSubstitutionArray($langs, 0, null, $attendees);
197 $url_link = null;
198 $html_link = null;
199
200 if (getDolGlobalString('MAIN_AGENDA_XCAL_EXPORTKEY')) {
201 $urlwithouturlroot = preg_replace('/' . preg_quote(DOL_URL_ROOT, '/') . '$/i', '', trim($dolibarr_main_url_root));
202 $urlwithroot = $urlwithouturlroot . DOL_URL_ROOT;
203 $url_link = $urlwithroot . '/public/agenda/agendaexport.php?format=ical' . ($conf->entity > 1 ? "&entity=" . $conf->entity : "");
204 $url_link .= '&exportkey=' . ($conf->global->MAIN_AGENDA_XCAL_EXPORTKEY ? urlencode(getDolGlobalString('MAIN_AGENDA_XCAL_EXPORTKEY')) : '...');
205 $url_link .= "&project=" . $listofselectedref[$email]->fk_project . '&module=' . urlencode('@eventorganization') . '&status=' . ConferenceOrBooth::STATUS_CONFIRMED;
206 $html_link = '<a href="' . $url_link . '">' . $langs->trans('DownloadICSLink') . '</a>';
207 }
208 $substitutionarray['__EVENTORGANIZATION_ICS_LINK__'] = $html_link;
209 $substitutionarray['__EVENTORGANIZATION_URL_LINK__'] = $url_link;
210
211 $parameters = array('mode' => 'formemail');
212
213 if (!empty($listofobjectref)) {
214 $parameters['listofobjectref'] = $listofobjectref;
215 }
216
217 complete_substitutions_array($substitutionarray, $langs, $attendees, $parameters);
218
219 $subjectreplaced = make_substitutions($subject, $substitutionarray);
220 $messagereplaced = make_substitutions($message, $substitutionarray);
221
222
223 if (empty($sendcontext)) {
224 $sendcontext = 'standard';
225 }
226
227 // Send mail (substitutionarray must be done just before this)
228 require_once DOL_DOCUMENT_ROOT . '/core/class/CMailFile.class.php';
229 $mailfile = new CMailFile($subjectreplaced, $sendto, $from, $messagereplaced, array(), array(), array(), $sendtocc, $sendtobcc, $deliveryreceipt, -1, '', '', "attendees_".$attendees->id, '', $sendcontext);
230 if ($mailfile->error) {
231 $resaction .= '<div class="error">' . $mailfile->error . '</div>';
232 } else {
233 $result = $mailfile->sendfile();
234 if ($result) {
235 $resaction .= $langs->trans('MailSuccessfulySent', $mailfile->getValidAddress($from, 2), $mailfile->getValidAddress($sendto, 2)) . '<br>'; // Must not contain "
236 $error = 0;
237
238 dol_syslog("Try to insert email event into agenda for objid=" . $attendees->id . " => objectobj=" . get_class($attendees));
239
240 $actionmsg = $langs->transnoentities('MailSentByTo', $from, $sendto);
241 if ($message) {
242 if ($sendtocc) {
243 $actionmsg = dol_concatdesc($actionmsg, $langs->transnoentities('Bcc') . ": " . $sendtocc);
244 }
245 $actionmsg = dol_concatdesc($actionmsg, $langs->transnoentities('MailTopic') . ": " . $subjectreplaced);
246 $actionmsg = dol_concatdesc($actionmsg, $langs->transnoentities('TextUsedInTheMessageBody') . ":");
247 $actionmsg = dol_concatdesc($actionmsg, $messagereplaced);
248 }
249 $actionmsg2 = '';
250
251 $objectobj2 = $listofselectedref[$email];
252 // Initialisation donnees
253 $objectobj2->actionmsg = $actionmsg; // Long text
254 $objectobj2->actionmsg2 = $actionmsg2; // Short text
255 $objectobj2->fk_element = $objectobj2->id;
256 $objectobj2->elementtype = $objectobj2->element;
257
258 $triggername = 'CONFERENCEORBOOTHATTENDEE_SENTBYMAIL';
259 if (!empty($triggername)) {
260 // Call trigger
261 $result = $objectobj2->call_trigger($triggername, $user);
262 if ($result < 0) {
263 $error++;
264 }
265 // End call triggers
266
267 if ($error) {
268 setEventMessages($db->lasterror(), $objectobj2->errors, 'errors');
269 dol_syslog("Error in trigger " . $triggername . ' ' . $db->lasterror(), LOG_ERR);
270 }
271 }
272
273 $nbsent++; // Nb of object sent
274 } else {
275 $langs->load("other");
276 if ($mailfile->error) {
277 $resaction .= $langs->trans('ErrorFailedToSendMail', $from, $sendto);
278 $resaction .= '<br><div class="error">' . $mailfile->error . '</div>';
279 } elseif (getDolGlobalString('MAIN_DISABLE_ALL_MAILS')) {
280 $resaction .= '<div class="warning">No mail sent. Feature is disabled by option MAIN_DISABLE_ALL_MAILS</div>';
281 } else {
282 $resaction .= $langs->trans('ErrorFailedToSendMail', $from, $sendto) . '<br><div class="error">(unhandled error)</div>';
283 }
284 }
285 }
286 } // foreach ($listofselectedid as $email => $attendees)
287 }
288 $resaction .= ($resaction ? '<br>' : $resaction);
289 $resaction .= '<strong>' . $langs->trans("ResultOfMailSending") . ':</strong><br>' . "\n";
290 $resaction .= $langs->trans("NbSelected") . ': ' . count($toselect) . "\n<br>";
291 $resaction .= $langs->trans("NbIgnored") . ': ' . ($nbignored ? $nbignored : 0) . "\n<br>";
292 $resaction .= $langs->trans("NbSent") . ': ' . ($nbsent ? $nbsent : 0) . "\n<br>";
293
294 if ($nbsent) {
295 $action = ''; // Do not show form post if there was at least one successful sent
296 //setEventMessages($langs->trans("EMailSentToNRecipients", $nbsent.'/'.count($toselect)), null, 'mesgs');
297 setEventMessages($langs->trans("EMailSentForNElements", $nbsent . '/' . count($toselect)), null, 'mesgs');
298 setEventMessages($resaction, null, 'mesgs');
299 } else {
300 //setEventMessages($langs->trans("EMailSentToNRecipients", 0), null, 'warnings'); // May be object has no generated PDF file
301 setEventMessages($resaction, null, 'warnings');
302 }
303
304 $action = 'list';
305 $massaction = '';
306}
307
308
309
310$parameters['toselect'] = $toselect;
311$parameters['uploaddir'] = $uploaddir;
312$parameters['massaction'] = $massaction;
313$parameters['diroutputmassaction'] = isset($diroutputmassaction) ? $diroutputmassaction : null;
314
315$reshook = $hookmanager->executeHooks('doMassActions', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks
316if ($reshook < 0) {
317 setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
318}
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Definition card.php:58
Class to send emails (with attachments or not) Usage: $mailfile = new CMailFile($subject,...
Class for ConferenceOrBoothAttendee.
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='', $noduplicate=0, $attop=0)
Set event messages in dol_events session object.
GETPOSTINT($paramname, $method=0)
Return the value of a $_GET or $_POST supervariable, converted into integer.
dol_string_nospecial($str, $newstr='_', $badcharstoreplace='', $badcharstoremove='', $keepspaces=0)
Clean a string from all punctuation characters to use it as a ref or login.
dol_concatdesc($text1, $text2, $forxml=false, $invert=false)
Concat 2 descriptions with a new line between them (second operand after first one with appropriate n...
complete_substitutions_array(&$substitutionarray, $outputlangs, $object=null, $parameters=null, $callfunc="completesubstitutionarray")
Complete the $substitutionarray with more entries coming from external module that had set the "subst...
make_substitutions($text, $substitutionarray, $outputlangs=null, $converttextinhtmlifnecessary=0)
Make substitution into a text string, replacing keys with vals from $substitutionarray (oldval=>newva...
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
getCommonSubstitutionArray($outputlangs, $onlykey=0, $exclude=null, $object=null, $include=null)
Return array of possible common substitutions.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.