dolibarr 18.0.6
doc_generic_odt.modules.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2010-2011 Laurent Destailleur <ely@users.sourceforge.net>
3 * Copyright (C) 2016 Charlie Benke <charlie@patas-monkey.com>
4 * Copyright (C) 2018-2019 Frédéric France <frederic.france@netlogic.fr>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <https://www.gnu.org/licenses/>.
18 * or see https://www.gnu.org/
19 */
20
27require_once DOL_DOCUMENT_ROOT.'/core/modules/societe/modules_societe.class.php';
28require_once DOL_DOCUMENT_ROOT.'/societe/class/societe.class.php';
29require_once DOL_DOCUMENT_ROOT.'/core/lib/company.lib.php';
30require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
31require_once DOL_DOCUMENT_ROOT.'/core/lib/doc.lib.php';
32
33
38{
43 public $emetteur;
44
50 public function __construct($db)
51 {
52 global $langs, $mysoc;
53
54 // Load translation files required by the page
55 $langs->loadLangs(array("main", "companies"));
56
57 $this->db = $db;
58 $this->name = "ODT templates";
59 $this->description = $langs->trans("DocumentModelOdt");
60 $this->scandir = 'COMPANY_ADDON_PDF_ODT_PATH'; // Name of constant that is used to save list of directories to scan
61
62 // Page size for A4 format
63 $this->type = 'odt';
64 $this->page_largeur = 0;
65 $this->page_hauteur = 0;
66 $this->format = array($this->page_largeur, $this->page_hauteur);
67 $this->marge_gauche = 0;
68 $this->marge_droite = 0;
69 $this->marge_haute = 0;
70 $this->marge_basse = 0;
71
72 $this->option_logo = 1; // Display logo
73
74 // Retrieves transmitter
75 $this->emetteur = $mysoc;
76 if (!$this->emetteur->country_code) {
77 $this->emetteur->country_code = substr($langs->defaultlang, -2); // By default, if was not defined
78 }
79 }
80
81
88 public function info($langs)
89 {
90 global $conf, $langs;
91
92 // Load traductions files required by page
93 $langs->loadLangs(array("companies", "errors"));
94
95 $form = new Form($this->db);
96
97 $texte = $this->description.".<br>\n";
98 $texte .= '<!-- form for option of ODT templates -->';
99 $texte .= '<form action="'.$_SERVER["PHP_SELF"].'" method="POST" enctype="multipart/form-data">';
100 $texte .= '<input type="hidden" name="token" value="'.newToken().'">';
101 $texte .= '<input type="hidden" name="page_y" value="">';
102 $texte .= '<input type="hidden" name="action" value="setModuleOptions">';
103 $texte .= '<input type="hidden" name="param1" value="COMPANY_ADDON_PDF_ODT_PATH">';
104 $texte .= '<table class="nobordernopadding centpercent">';
105
106 // List of directories area
107 $texte .= '<tr><td>';
108 $texttitle = $langs->trans("ListOfDirectories");
109 $listofdir = explode(',', preg_replace('/[\r\n]+/', ',', trim($conf->global->COMPANY_ADDON_PDF_ODT_PATH)));
110 $listoffiles = array();
111 foreach ($listofdir as $key => $tmpdir) {
112 $tmpdir = trim($tmpdir);
113 $tmpdir = preg_replace('/DOL_DATA_ROOT/', DOL_DATA_ROOT, $tmpdir);
114 if (!$tmpdir) {
115 unset($listofdir[$key]);
116 continue;
117 }
118 if (!is_dir($tmpdir)) {
119 $texttitle .= img_warning($langs->trans("ErrorDirNotFound", $tmpdir), 0);
120 } else {
121 $tmpfiles = dol_dir_list($tmpdir, 'files', 0, '\.od(s|t)$', '', 'name', SORT_ASC, 0, true); // Disable hook for the moment
122 if (count($tmpfiles)) {
123 $listoffiles = array_merge($listoffiles, $tmpfiles);
124 }
125 }
126 }
127 $texthelp = $langs->trans("ListOfDirectoriesForModelGenODT");
128 // Add list of substitution keys
129 $texthelp .= '<br>'.$langs->trans("FollowingSubstitutionKeysCanBeUsed").'<br>';
130 $texthelp .= $langs->transnoentitiesnoconv("FullListOnOnlineDocumentation"); // This contains an url, we don't modify it
131
132 $texte .= $form->textwithpicto($texttitle, $texthelp, 1, 'help', '', 1);
133 $texte .= '<table><tr><td>';
134 $texte .= '<textarea class="flat" cols="60" name="value1">';
135 $texte .= $conf->global->COMPANY_ADDON_PDF_ODT_PATH;
136 $texte .= '</textarea>';
137 $texte .= '</td>';
138 $texte .= '<td class="center">&nbsp; ';
139 $texte .= '<input type="submit" class="button small reposition" name="modify" value="'.$langs->trans("Modify").'">';
140 $texte .= '</td>';
141 $texte .= '</tr>';
142 $texte .= '</table>';
143
144 // Scan directories
145 $nbofiles = count($listoffiles);
146 if (!empty($conf->global->COMPANY_ADDON_PDF_ODT_PATH)) {
147 $texte .= $langs->trans("NumberOfModelFilesFound").': <b>';
148 //$texte.=$nbofiles?'<a id="a_'.get_class($this).'" href="#">':'';
149 $texte .= $nbofiles;
150 //$texte.=$nbofiles?'</a>':'';
151 $texte .= '</b>';
152 }
153
154 if ($nbofiles) {
155 $texte .= '<div id="div_'.get_class($this).'" class="hiddenx">';
156 // Show list of found files
157 foreach ($listoffiles as $file) {
158 $texte .= '- '.$file['name'].' &nbsp; <a class="reposition" href="'.DOL_URL_ROOT.'/document.php?modulepart=doctemplates&file=thirdparties/'.urlencode(basename($file['name'])).'">'.img_picto('', 'listlight').'</a>';
159 $texte .= ' &nbsp; <a class="reposition" href="'.$_SERVER["PHP_SELF"].'?modulepart=doctemplates&keyforuploaddir=COMPANY_ADDON_PDF_ODT_PATH&action=deletefile&token='.newToken().'&file='.urlencode(basename($file['name'])).'">'.img_picto('', 'delete').'</a>';
160 $texte .= '<br>';
161 }
162 $texte .= '</div>';
163 }
164 // Add input to upload a new template file.
165 $texte .= '<div>'.$langs->trans("UploadNewTemplate");
166 $maxfilesizearray = getMaxFileSizeArray();
167 $maxmin = $maxfilesizearray['maxmin'];
168 if ($maxmin > 0) {
169 $texte .= '<input type="hidden" name="MAX_FILE_SIZE" value="'.($maxmin * 1024).'">'; // MAX_FILE_SIZE must precede the field type=file
170 }
171 $texte .= ' <input type="file" name="uploadfile">';
172 $texte .= '<input type="hidden" value="COMPANY_ADDON_PDF_ODT_PATH" name="keyforuploaddir">';
173 $texte .= '<input type="submit" class="button small reposition" value="'.dol_escape_htmltag($langs->trans("Upload")).'" name="upload">';
174 $texte .= '</div>';
175 $texte .= '</td>';
176
177 $texte .= '<td rowspan="2" class="tdtop hideonsmartphone">';
178 $texte .= '<span class="opacitymedium">';
179 $texte .= $langs->trans("ExampleOfDirectoriesForModelGen");
180 $texte .= '</span>';
181 $texte .= '</td>';
182 $texte .= '</tr>';
183
184 $texte .= '</table>';
185 $texte .= '</form>';
186
187 return $texte;
188 }
189
190 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
202 public function write_file($object, $outputlangs, $srctemplatepath, $hidedetails = 0, $hidedesc = 0, $hideref = 0)
203 {
204 // phpcs:enable
205 global $user, $langs, $conf, $mysoc, $hookmanager;
206 global $action;
207
208 if (empty($srctemplatepath)) {
209 dol_syslog("doc_generic_odt::write_file parameter srctemplatepath empty", LOG_WARNING);
210 return -1;
211 }
212
213 // Add odtgeneration hook
214 if (!is_object($hookmanager)) {
215 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
216 $hookmanager = new HookManager($this->db);
217 }
218 $hookmanager->initHooks(array('odtgeneration'));
219 global $action;
220
221 if (!is_object($outputlangs)) {
222 $outputlangs = $langs;
223 }
224 $sav_charset_output = $outputlangs->charset_output;
225 $outputlangs->charset_output = 'UTF-8';
226
227 // Load translation files required by the page
228 $outputlangs->loadLangs(array("main", "dict", "companies", "projects"));
229
230 if ($conf->societe->multidir_output[$object->entity]) {
231 $dir = $conf->societe->multidir_output[$object->entity];
232 $objectref = dol_sanitizeFileName($object->id);
233 if (!preg_match('/specimen/i', $objectref)) {
234 $dir .= "/".$objectref;
235 }
236
237 if (!file_exists($dir)) {
238 if (dol_mkdir($dir) < 0) {
239 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
240 return -1;
241 }
242 }
243
244 if (file_exists($dir)) {
245 //print "srctemplatepath=".$srctemplatepath; // Src filename
246 $newfile = basename($srctemplatepath);
247 $newfiletmp = preg_replace('/\.od(s|t)/i', '', $newfile);
248 $newfiletmp = preg_replace('/template_/i', '', $newfiletmp);
249 $newfiletmp = preg_replace('/modele_/i', '', $newfiletmp);
250 // Get extension (ods or odt)
251 $newfileformat = substr($newfile, strrpos($newfile, '.') + 1);
252 if (!empty($conf->global->MAIN_DOC_USE_OBJECT_THIRDPARTY_NAME)) {
253 $newfiletmp = dol_sanitizeFileName(dol_string_nospecial($object->name)) . '-' . $newfiletmp;
254 $newfiletmp = preg_replace('/__+/', '_', $newfiletmp); // Replace repeated _ into one _ (to avoid string with substitution syntax)
255 }
256 if (getDolGlobalString('MAIN_DOC_USE_TIMING')) {
257 $format = getDolGlobalString('MAIN_DOC_USE_TIMING');
258 if ($format == '1') {
259 $format = '%Y%m%d%H%M%S';
260 }
261 $filename = $newfiletmp . '-' . dol_print_date(dol_now(), $format) . '.' . $newfileformat;
262 } else {
263 $filename = $newfiletmp . '.' . $newfileformat;
264 }
265 $file = $dir . '/' . $filename;
266 $object->builddoc_filename = $filename; // For triggers
267 //print "newfileformat=".$newfileformat;
268 //print "newdir=".$dir;
269 //print "newfile=".$newfile;
270 //print "file=".$file;
271 //print "conf->societe->dir_temp=".$conf->societe->dir_temp;
272 //exit;
273
274 dol_mkdir($conf->societe->multidir_temp[$object->entity]);
275 if (!is_writable($conf->societe->multidir_temp[$object->entity])) {
276 $this->error = $langs->transnoentities("ErrorFailedToWriteInTempDirectory", $conf->societe->multidir_temp[$object->entity]);
277 dol_syslog('Error in write_file: ' . $this->error, LOG_ERR);
278 return -1;
279 }
280
281 // Open and load template
282 require_once ODTPHP_PATH.'odf.php';
283 try {
284 $odfHandler = new Odf(
285 $srctemplatepath,
286 array(
287 'PATH_TO_TMP' => $conf->societe->multidir_temp[$object->entity],
288 'ZIP_PROXY' => 'PclZipProxy', // PhpZipProxy or PclZipProxy. Got "bad compression method" error when using PhpZipProxy.
289 'DELIMITER_LEFT' => '{',
290 'DELIMITER_RIGHT' => '}'
291 )
292 );
293 } catch (Exception $e) {
294 $this->error = $e->getMessage();
295 dol_syslog($e->getMessage(), LOG_INFO);
296 return -1;
297 }
298 //print $odfHandler->__toString()."\n";
299
300 // Replace tags of lines for contacts
301 $contact_arrray = array();
302
303 $sql = "SELECT p.rowid";
304 $sql .= " FROM ".MAIN_DB_PREFIX."socpeople as p";
305 $sql .= " WHERE p.fk_soc = ".((int) $object->id);
306
307 $result = $this->db->query($sql);
308 $num = $this->db->num_rows($result);
309
310 if ($num) {
311 require_once DOL_DOCUMENT_ROOT.'/contact/class/contact.class.php';
312
313 $i = 0;
314 $contactstatic = new Contact($this->db);
315
316 while ($i < $num) {
317 $obj = $this->db->fetch_object($result);
318
319 $contact_arrray[$i] = $obj->rowid;
320 $i++;
321 }
322 }
323 if ((is_array($contact_arrray) && count($contact_arrray) > 0)) {
324 try {
325 $listlines = $odfHandler->setSegment('companycontacts');
326
327 foreach ($contact_arrray as $array_key => $contact_id) {
328 $res_contact = $contactstatic->fetch($contact_id);
329 if ((int) $res_contact > 0) {
330 $tmparray = $this->get_substitutionarray_contact($contactstatic, $outputlangs, 'contact');
331 foreach ($tmparray as $key => $val) {
332 try {
333 $listlines->setVars($key, $val, true, 'UTF-8');
334 } catch (OdfException $e) {
335 dol_syslog($e->getMessage(), LOG_INFO);
336 } catch (SegmentException $e) {
337 dol_syslog($e->getMessage(), LOG_INFO);
338 }
339 }
340 $listlines->merge();
341 } else {
342 $this->error = $contactstatic->error;
343 dol_syslog($this->error, LOG_WARNING);
344 }
345 }
346 $odfHandler->mergeSegment($listlines);
347 } catch (OdfException $e) {
348 $this->error = $e->getMessage();
349 dol_syslog($this->error, LOG_WARNING);
350 //return -1;
351 }
352 }
353
354 // Make substitutions into odt
355 $array_user = $this->get_substitutionarray_user($user, $outputlangs);
356 $array_soc = $this->get_substitutionarray_mysoc($mysoc, $outputlangs);
357 $array_thirdparty = $this->get_substitutionarray_thirdparty($object, $outputlangs);
358 $array_other = $this->get_substitutionarray_other($outputlangs);
359
360 $tmparray = array_merge($array_user, $array_soc, $array_thirdparty, $array_other);
361
362 complete_substitutions_array($tmparray, $outputlangs, $object);
363
364 // Call the ODTSubstitution hook
365 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
366 $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
367
368 // Replace variables into document
369 foreach ($tmparray as $key => $value) {
370 try {
371 if (preg_match('/logo$/', $key)) { // Image
372 if (file_exists($value)) {
373 $odfHandler->setImage($key, $value);
374 } else {
375 $odfHandler->setVars($key, 'ErrorFileNotFound', true, 'UTF-8');
376 }
377 } else // Text
378 {
379 $odfHandler->setVars($key, $value, true, 'UTF-8');
380 }
381 } catch (OdfException $e) {
382 // setVars failed, probably because key not found
383 dol_syslog($e->getMessage(), LOG_INFO);
384 }
385 }
386
387 // Replace labels translated
388 $tmparray = $outputlangs->get_translations_for_substitutions();
389 foreach ($tmparray as $key => $value) {
390 try {
391 $odfHandler->setVars($key, $value, true, 'UTF-8');
392 } catch (OdfException $e) {
393 dol_syslog($e->getMessage(), LOG_INFO);
394 }
395 }
396
397 // Call the beforeODTSave hook
398 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
399 $reshook = $hookmanager->executeHooks('beforeODTSave', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
400
401 // Write new file
402 if (!empty($conf->global->MAIN_ODT_AS_PDF)) {
403 try {
404 $odfHandler->exportAsAttachedPDF($file);
405 } catch (Exception $e) {
406 $this->error = $e->getMessage();
407 dol_syslog($e->getMessage(), LOG_INFO);
408 return -1;
409 }
410 } else {
411 try {
412 $odfHandler->creator = $user->getFullName($outputlangs);
413 $odfHandler->title = $object->builddoc_filename;
414 $odfHandler->subject = $object->builddoc_filename;
415
416 if (!empty($conf->global->ODT_ADD_DOLIBARR_ID)) {
417 $odfHandler->userdefined['dol_id'] = $object->id;
418 $odfHandler->userdefined['dol_element'] = $object->element;
419 }
420
421 $odfHandler->saveToDisk($file);
422 } catch (Exception $e) {
423 $this->error = $e->getMessage();
424 dol_syslog($e->getMessage(), LOG_INFO);
425 return -1;
426 }
427 }
428 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
429 $reshook = $hookmanager->executeHooks('afterODTCreation', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
430
431 dolChmod($file);
432
433 $odfHandler = null; // Destroy object
434
435 $this->result = array('fullpath'=>$file);
436
437 return 1; // Success
438 } else {
439 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
440 return -1;
441 }
442 }
443
444 $this->error = 'UnknownError';
445 return -1;
446 }
447}
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 contact/addresses.
Class to manage generation of HTML components Only common components must be here.
Class to manage hooks.
Parent class for third parties models of doc generators.
Class to build documents using ODF templates generator.
__construct($db)
Constructor.
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.
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($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:62
img_warning($titlealt='default', $moreatt='', $morecss='pictowarning')
Show warning logo.
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_print_date($time, $format='', $tzoutput='auto', $outputlangs='', $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
dolChmod($filepath, $newmask='')
Change mod of a file.
dol_now($mode='auto')
Return date for now.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=false, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
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.
getDolGlobalString($key, $default='')
Return 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:120
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition repair.php:123
getMaxFileSizeArray()
Return the max allowed for file upload.