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