dolibarr 21.0.0-alpha
doc_generic_ticket_odt.modules.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2010-2012 Laurent Destailleur <eldy@users.sourceforge.net>
3 * Copyright (C) 2012 Juanjo Menent <jmenent@2byte.es>
4 * Copyright (C) 2018-2024 Frédéric France <frederic.france@free.fr>
5 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
6*
7* This program is free software; you can redistribute it and/or modify
8* it under the terms of the GNU General Public License as published by
9* the Free Software Foundation; either version 3 of the License, or
10* (at your option) any later version.
11*
12* This program is distributed in the hope that it will be useful,
13* but WITHOUT ANY WARRANTY; without even the implied warranty of
14* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15* GNU General Public License for more details.
16*
17* You should have received a copy of the GNU General Public License
18* along with this program. If not, see <https://www.gnu.org/licenses/>.
19* or see https://www.gnu.org/
20*/
21
28require_once DOL_DOCUMENT_ROOT.'/core/modules/ticket/modules_ticket.php';
29require_once DOL_DOCUMENT_ROOT.'/core/lib/company.lib.php';
30require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
31require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
32require_once DOL_DOCUMENT_ROOT.'/core/lib/doc.lib.php';
33
34
39{
44 public $version = 'dolibarr';
45
46
52 public function __construct($db)
53 {
54 global $langs, $mysoc;
55
56 // Load translation files required by the page
57 $langs->loadLangs(array("main", "companies"));
58
59 $this->db = $db;
60 $this->name = "ODT templates";
61 $this->description = $langs->trans("DocumentModelOdt");
62 $this->scandir = 'TICKET_ADDON_PDF_ODT_PATH'; // Name of constant that is used to save list of directories to scan
63
64 // Page size for A4 format
65 $this->type = 'odt';
66 $this->page_largeur = 0;
67 $this->page_hauteur = 0;
68 $this->format = array($this->page_largeur, $this->page_hauteur);
69 $this->marge_gauche = 0;
70 $this->marge_droite = 0;
71 $this->marge_haute = 0;
72 $this->marge_basse = 0;
73
74 $this->option_logo = 1; // Display logo
75 $this->option_tva = 0; // Manage the vat option USER_TVAOPTION
76 $this->option_multilang = 1; // Available in several languages
77 $this->option_freetext = 0; // Support add of a personalised text
78 $this->option_draft_watermark = 0; // Support add of a watermark on drafts
79
80 if ($mysoc === null) {
81 dol_syslog(get_class($this).'::__construct() Global $mysoc should not be null.'. getCallerInfoString(), LOG_ERR);
82 return;
83 }
84
85 // Get source company
86 $this->emetteur = $mysoc;
87 if (!$this->emetteur->country_code) {
88 $this->emetteur->country_code = substr($langs->defaultlang, -2); // By default if not defined
89 }
90 }
91
92
99 public function info($langs)
100 {
101 global $langs;
102
103 // Load translation files required by the page
104 $langs->loadLangs(array('companies', 'errors'));
105
106 $form = new Form($this->db);
107
108 $texte = $this->description.".<br>\n";
109 $texte .= '<form action="'.$_SERVER["PHP_SELF"].'" method="POST" enctype="multipart/form-data">';
110 $texte .= '<input type="hidden" name="token" value="'.newToken().'">';
111 $texte .= '<input type="hidden" name="page_y" value="">';
112 $texte .= '<input type="hidden" name="action" value="setModuleOptions">';
113 $texte .= '<input type="hidden" name="param1" value="TICKET_ADDON_PDF_ODT_PATH">';
114
115 $texte .= '<table class="nobordernopadding" width="100%">';
116
117 // List of directories area
118 $texte .= '<tr><td>';
119 $texttitle = $langs->trans("ListOfDirectories");
120 $listofdir = explode(',', preg_replace('/[\r\n]+/', ',', trim(getDolGlobalString('TICKET_ADDON_PDF_ODT_PATH'))));
121 $listoffiles = array();
122 foreach ($listofdir as $key => $tmpdir) {
123 $tmpdir = trim($tmpdir);
124 $tmpdir = preg_replace('/DOL_DATA_ROOT/', DOL_DATA_ROOT, $tmpdir);
125 if (!$tmpdir) {
126 unset($listofdir[$key]);
127 continue;
128 }
129 if (!is_dir($tmpdir)) {
130 $texttitle .= img_warning($langs->trans("ErrorDirNotFound", $tmpdir), '');
131 } else {
132 $tmpfiles = dol_dir_list($tmpdir, 'files', 0, '\.(ods|odt)');
133 if (count($tmpfiles)) {
134 $listoffiles = array_merge($listoffiles, $tmpfiles);
135 }
136 }
137 }
138 $texthelp = $langs->trans("ListOfDirectoriesForModelGenODT");
139 $texthelp .= '<br><br><span class="opacitymedium">'.$langs->trans("ExampleOfDirectoriesForModelGen").'</span>';
140 // Add list of substitution keys
141 $texthelp .= '<br>'.$langs->trans("FollowingSubstitutionKeysCanBeUsed").'<br>';
142 $texthelp .= $langs->transnoentitiesnoconv("FullListOnOnlineDocumentation"); // This contains an url, we don't modify it
143
144 $texte .= $form->textwithpicto($texttitle, $texthelp, 1, 'help', '', 1, 3, $this->name);
145 $texte .= '<div><div style="display: inline-block; min-width: 100px; vertical-align: middle;">';
146 $texte .= '<textarea class="flat" cols="60" name="value1">';
147 $texte .= getDolGlobalString('TICKET_ADDON_PDF_ODT_PATH');
148 $texte .= '</textarea>';
149 $texte .= '</div><div style="display: inline-block; vertical-align: middle;">';
150 $texte .= '<input type="submit" class="button button-edit reposition smallpaddingimp" name="modify" value="'.dol_escape_htmltag($langs->trans("Modify")).'">';
151 $texte .= '<br></div></div>';
152
153 // Scan directories
154 if (count($listofdir)) {
155 $texte .= $langs->trans("NumberOfModelFilesFound").': <b>'.count($listoffiles).'</b>';
156
157 $texte .= '<div id="div_'.get_class($this).'" class="hiddenx">';
158 // Show list of found files
159 foreach ($listoffiles as $file) {
160 $texte .= '- '.$file['name'].' <a href="'.DOL_URL_ROOT.'/document.php?modulepart=doctemplates&file=tickets/'.urlencode(basename($file['name'])).'">'.img_picto('', 'listlight').'</a>';
161 $texte .= ' &nbsp; <a class="reposition" href="'.$_SERVER["PHP_SELF"].'?modulepart=doctemplates&keyforuploaddir=TICKET_ADDON_PDF_ODT_PATH&action=deletefile&token='.newToken().'&file='.urlencode(basename($file['name'])).'">'.img_picto('', 'delete').'</a>';
162 $texte .= '<br>';
163 }
164 $texte .= '</div>';
165 }
166 // Add input to upload a new template file.
167 $texte .= '<div>'.$langs->trans("UploadNewTemplate");
168 $maxfilesizearray = getMaxFileSizeArray();
169 $maxmin = $maxfilesizearray['maxmin'];
170 if ($maxmin > 0) {
171 $texte .= '<input type="hidden" name="MAX_FILE_SIZE" value="'.($maxmin * 1024).'">'; // MAX_FILE_SIZE must precede the field type=file
172 }
173 $texte .= ' <input type="file" name="uploadfile">';
174 $texte .= '<input type="hidden" value="TICKET_ADDON_PDF_ODT_PATH" name="keyforuploaddir">';
175 $texte .= '<input type="submit" class="button smallpaddingimp reposition" value="'.dol_escape_htmltag($langs->trans("Upload")).'" name="upload">';
176 $texte .= '</div>';
177
178 $texte .= '</td>';
179
180 $texte .= '</tr>';
181
182 $texte .= '</table>';
183 $texte .= '</form>';
184
185 return $texte;
186 }
187
188 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
200 public function write_file($object, $outputlangs, $srctemplatepath = '', $hidedetails = 0, $hidedesc = 0, $hideref = 0)
201 {
202 // phpcs:enable
203 global $user, $langs, $conf, $mysoc, $hookmanager;
204
205 if (empty($srctemplatepath)) {
206 dol_syslog("doc_generic_odt::write_file parameter srctemplatepath empty", LOG_WARNING);
207 return -1;
208 }
209
210 // Add odtgeneration hook
211 if (!is_object($hookmanager)) {
212 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
213 $hookmanager = new HookManager($this->db);
214 }
215 $hookmanager->initHooks(array('odtgeneration'));
216 global $action;
217
218 if (!is_object($outputlangs)) {
219 $outputlangs = $langs;
220 }
221 $sav_charset_output = $outputlangs->charset_output;
222 $outputlangs->charset_output = 'UTF-8';
223
224 // Load translation files required by the page
225 $outputlangs->loadLangs(array("main", "companies", "bills", "dict"));
226
227 if ($conf->ticket->dir_output) {
228 // If $object is id instead of object
229 if (!is_object($object)) {
230 $id = $object;
231 $object = new User($this->db);
232 $result = $object->fetch($id);
233 if ($result < 0) {
234 dol_print_error($this->db, $object->error);
235 return -1;
236 }
237 }
238
239 $object->fetch_thirdparty();
240
241 $dir = $conf->ticket->dir_output;
242 $objectref = dol_sanitizeFileName($object->ref);
243 if (!preg_match('/specimen/i', $objectref)) {
244 $dir .= "/".$objectref;
245 }
246 $file = $dir."/".$objectref.".odt";
247
248 if (!file_exists($dir)) {
249 if (dol_mkdir($dir) < 0) {
250 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
251 return -1;
252 }
253 }
254
255 if (file_exists($dir)) {
256 //print "srctemplatepath=".$srctemplatepath; // Src filename
257 $newfile = basename($srctemplatepath);
258 $newfiletmp = preg_replace('/\.od[ts]/i', '', $newfile);
259 $newfiletmp = preg_replace('/template_/i', '', $newfiletmp);
260 $newfiletmp = preg_replace('/modele_/i', '', $newfiletmp);
261
262 $newfiletmp = $objectref . '_' . $newfiletmp;
263
264 // Get extension (ods or odt)
265 $newfileformat = substr($newfile, strrpos($newfile, '.') + 1);
266 if (getDolGlobalString('MAIN_DOC_USE_TIMING')) {
267 $format = getDolGlobalString('MAIN_DOC_USE_TIMING');
268 if ($format == '1') {
269 $format = '%Y%m%d%H%M%S';
270 }
271 $filename = $newfiletmp . '-' . dol_print_date(dol_now(), $format) . '.' . $newfileformat;
272 } else {
273 $filename = $newfiletmp . '.' . $newfileformat;
274 }
275 $file = $dir . '/' . $filename;
276 //print "newdir=".$dir;
277 //print "newfile=".$newfile;
278 //print "file=".$file;
279 //print "conf->ticket->dir_temp=".$conf->ticket->dir_temp;
280
281 dol_mkdir($conf->ticket->dir_temp);
282 if (!is_writable($conf->ticket->dir_temp)) {
283 $this->error = $langs->transnoentities("ErrorFailedToWriteInTempDirectory", $conf->ticket->dir_temp);
284 dol_syslog('Error in write_file: ' . $this->error, LOG_ERR);
285 return -1;
286 }
287
288 // If CUSTOMER contact defined on user, we use it
289 $usecontact = false;
290 $arrayidcontact = $object->getIdContact('external', 'CUSTOMER');
291 if (count($arrayidcontact) > 0) {
292 $usecontact = true;
293 $result = $object->fetch_contact($arrayidcontact[0]);
294 }
295
296 // Recipient name
297 if (!empty($usecontact)) {
298 // We can use the company of contact instead of thirdparty company
299 if ($object->contact->socid != $object->thirdparty->id && (!isset($conf->global->MAIN_USE_COMPANY_NAME_OF_CONTACT) || getDolGlobalString('MAIN_USE_COMPANY_NAME_OF_CONTACT'))) {
300 $object->contact->fetch_thirdparty();
301 $socobject = $object->contact->thirdparty;
302 $contactobject = $object->contact;
303 } else {
304 $socobject = $object->thirdparty;
305 // if we have a CUSTOMER contact and we don't use it as thirdparty recipient we store the contact object for later use
306 $contactobject = $object->contact;
307 }
308 } else {
309 $socobject = $object->thirdparty;
310 }
311
312 // Open and load template
313 require_once ODTPHP_PATH.'odf.php';
314 try {
315 $odfHandler = new Odf(
316 $srctemplatepath,
317 array(
318 'PATH_TO_TMP' => $conf->ticket->dir_temp,
319 'ZIP_PROXY' => getDolGlobalString('MAIN_ODF_ZIP_PROXY', 'PclZipProxy'), // PhpZipProxy or PclZipProxy. Got "bad compression method" error when using PhpZipProxy.
320 'DELIMITER_LEFT' => '{',
321 'DELIMITER_RIGHT' => '}'
322 )
323 );
324 } catch (Exception $e) {
325 $this->error = $e->getMessage();
326 dol_syslog($e->getMessage(), LOG_WARNING);
327 return -1;
328 }
329
330 // Make substitutions into odt
331 $array_user = $this->get_substitutionarray_user($user, $outputlangs);
332 $array_soc = $this->get_substitutionarray_mysoc($mysoc, $outputlangs);
333 $array_thirdparty = $this->get_substitutionarray_thirdparty($socobject, $outputlangs);
334 $array_other = $this->get_substitutionarray_other($outputlangs);
335 // retrieve contact information for use in object as contact_xxx tags
336 $array_thirdparty_contact = array();
337 if ($usecontact && is_object($contactobject)) {
338 $array_thirdparty_contact = $this->get_substitutionarray_contact($contactobject, $outputlangs, 'contact');
339 }
340
341 $tmparray = array_merge($array_user, $array_soc, $array_thirdparty, $array_other, $array_thirdparty_contact);
342 complete_substitutions_array($tmparray, $outputlangs, $object);
343 $object->fetch_optionals();
344 // Call the ODTSubstitution hook
345 $parameters = array('file' => $file, 'object' => $object, 'outputlangs' => $outputlangs, 'substitutionarray' => &$tmparray);
346 $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
347 foreach ($tmparray as $key => $value) {
348 try {
349 if (preg_match('/logo$/', $key)) { // Image
350 if (file_exists($value)) {
351 $odfHandler->setImage($key, $value);
352 } else {
353 $odfHandler->setVars($key, 'ErrorFileNotFound', true, 'UTF-8');
354 }
355 } else { // Text
356 $odfHandler->setVars($key, $value, true, 'UTF-8');
357 }
358 } catch (OdfException $e) {
359 dol_syslog($e->getMessage(), LOG_WARNING);
360 }
361 }
362
363 // Replace labels translated
364 $tmparray = $outputlangs->get_translations_for_substitutions();
365 foreach ($tmparray as $key => $value) {
366 try {
367 $odfHandler->setVars($key, $value, true, 'UTF-8');
368 } catch (OdfException $e) {
369 dol_syslog($e->getMessage(), LOG_WARNING);
370 }
371 }
372
373 // Call the beforeODTSave hook
374 $parameters = array('odfHandler' => &$odfHandler, 'file' => $file, 'object' => $object, 'outputlangs' => $outputlangs);
375 $reshook = $hookmanager->executeHooks('beforeODTSave', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
376
377 // Write new file
378 if (getDolGlobalString('MAIN_ODT_AS_PDF')) {
379 try {
380 $odfHandler->exportAsAttachedPDF($file);
381 } catch (Exception $e) {
382 $this->error = $e->getMessage();
383 dol_syslog($e->getMessage(), LOG_WARNING);
384 return -1;
385 }
386 } else {
387 try {
388 $odfHandler->saveToDisk($file);
389 } catch (Exception $e) {
390 $this->error = $e->getMessage();
391 dol_syslog($e->getMessage(), LOG_WARNING);
392 return -1;
393 }
394 }
395
396 $reshook = $hookmanager->executeHooks('afterODTCreation', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
397
398 dolChmod($file);
399
400 $odfHandler = null; // Destroy object
401
402 $this->result = array('fullpath' => $file);
403
404 return 1; // Success
405 } else {
406 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
407 return -1;
408 }
409 }
410
411 return -1;
412 }
413
414 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
423 public function get_substitutionarray_object($object, $outputlangs, $array_key = 'object')
424 {
425 // phpcs:enable
426 if (!$object instanceof User) {
427 dol_syslog("Expected Ticket object, got ".gettype($object), LOG_ERR);
428 return array();
429 }
430
431 $array_other = array();
432 foreach ($object as $key => $value) {
433 if (!is_array($value) && !is_object($value)) {
434 $array_other[$array_key.'_'.$key] = $value;
435 }
436 }
437
438 return $array_other;
439 }
440}
$id
Definition account.php:39
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Definition card.php:58
get_substitutionarray_mysoc($mysoc, $outputlangs)
Define array with couple substitution key => substitution value.
get_substitutionarray_contact($object, $outputlangs, $array_key='object')
Define array with couple substitution key => substitution value.
get_substitutionarray_other($outputlangs)
Define array with couple substitution key => substitution value.
get_substitutionarray_thirdparty($object, $outputlangs, $array_key='company')
Define array with couple substitution key => substitution value For example {company_name}...
get_substitutionarray_user($user, $outputlangs)
Define array with couple substitution key => substitution value.
Class to manage generation of HTML components Only common components must be here.
Class to manage hooks.
Parent class for documents models.
Class to manage Dolibarr users.
Class to build documents using ODF templates generator.
write_file($object, $outputlangs, $srctemplatepath='', $hidedetails=0, $hidedesc=0, $hideref=0)
Function to build a document on disk using the generic odt module.
info($langs)
Return description of a module.
get_substitutionarray_object($object, $outputlangs, $array_key='object')
get substitution array for object
print $script_file $mode $langs defaultlang(is_numeric($duration_value) ? " delay=". $duration_value :"").(is_numeric($duration_value2) ? " after cd cd cd description as description
Only used if Module[ID]Desc translation string is not found.
dol_dir_list($utf8_path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0, $nbsecondsold=0)
Scan a directory and return a list of files/directories.
Definition files.lib.php:63
img_warning($titlealt='default', $moreatt='', $morecss='pictowarning')
Show warning logo.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
getCallerInfoString()
Get caller info as a string that can be appended to a log message.
dolChmod($filepath, $newmask='')
Change mod of a file.
dol_now($mode='auto')
Return date for now.
dol_print_date($time, $format='', $tzoutput='auto', $outputlangs=null, $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
newToken()
Return the value of token currently saved into session with name 'newtoken'.
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...
dol_sanitizeFileName($str, $newstr='_', $unaccent=1)
Clean a string to use it as a file name.
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
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.
dol_mkdir($dir, $dataroot='', $newmask='')
Creation of a directory (this can create recursive subdir)
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition repair.php:137
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition repair.php:140
getMaxFileSizeArray()
Return the max allowed for file upload.