dolibarr 18.0.6
doc_generic_member_odt.class.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-2021 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/member/modules_member.class.php';
28require_once DOL_DOCUMENT_ROOT.'/core/lib/company.lib.php';
29require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.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{
42 public $emetteur;
43
48 public $version = 'dolibarr';
49
50
56 public function __construct($db)
57 {
58 global $conf, $langs, $mysoc;
59
60 // Load translation files required by the page
61 $langs->loadLangs(array("main", "companies"));
62
63 $this->db = $db;
64 $this->name = "ODT templates";
65 $this->description = $langs->trans("DocumentModelOdt");
66 $this->scandir = 'MEMBER_ADDON_PDF_ODT_PATH'; // Name of constant that is used to save list of directories to scan
67
68 // Page size for A4 format
69 $this->type = 'odt';
70 $this->page_largeur = 0;
71 $this->page_hauteur = 0;
72 $this->format = array($this->page_largeur, $this->page_hauteur);
73 $this->marge_gauche = 0;
74 $this->marge_droite = 0;
75 $this->marge_haute = 0;
76 $this->marge_basse = 0;
77
78 $this->option_logo = 1; // Display logo
79 $this->option_tva = 0; // Manage the vat option FACTURE_TVAOPTION
80 $this->option_modereg = 0; // Display payment mode
81 $this->option_condreg = 0; // Display payment terms
82 $this->option_multilang = 1; // Available in several languages
83 $this->option_escompte = 0; // Displays if there has been a discount
84 $this->option_credit_note = 0; // Support credit notes
85 $this->option_freetext = 1; // Support add of a personalised text
86 $this->option_draft_watermark = 0; // Support add of a watermark on drafts
87
88 // Recupere emetteur
89 $this->emetteur = $mysoc;
90 if (!$this->emetteur->country_code) {
91 $this->emetteur->country_code = substr($langs->defaultlang, -2); // By default if not defined
92 }
93 }
94
95
102 public function info($langs)
103 {
104 global $conf, $langs;
105
106 // Load translation files required by the page
107 $langs->loadLangs(array('companies', 'errors'));
108
109 $form = new Form($this->db);
110
111 $texte = $this->description.".<br>\n";
112 $texte .= '<form action="'.$_SERVER["PHP_SELF"].'" method="POST" enctype="multipart/form-data">';
113 $texte .= '<input type="hidden" name="token" value="'.newToken().'">';
114 $texte .= '<input type="hidden" name="page_y" value="">';
115 $texte .= '<input type="hidden" name="action" value="setModuleOptions">';
116 $texte .= '<input type="hidden" name="param1" value="MEMBER_ADDON_PDF_ODT_PATH">';
117 $texte .= '<table class="nobordernopadding" width="100%">';
118
119 // List of directories area
120 $texte .= '<tr><td>';
121 $texttitle = $langs->trans("ListOfDirectories");
122 $listofdir = explode(',', preg_replace('/[\r\n]+/', ',', trim(getDolGlobalString('MEMBER_ADDON_PDF_ODT_PATH'))));
123 $listoffiles = array();
124 foreach ($listofdir as $key => $tmpdir) {
125 $tmpdir = trim($tmpdir);
126 $tmpdir = preg_replace('/DOL_DATA_ROOT/', DOL_DATA_ROOT, $tmpdir);
127 if (!$tmpdir) {
128 unset($listofdir[$key]);
129 continue;
130 }
131 if (!is_dir($tmpdir)) {
132 $texttitle .= img_warning($langs->trans("ErrorDirNotFound", $tmpdir), 0);
133 } else {
134 $tmpfiles = dol_dir_list($tmpdir, 'files', 0, '\.(ods|odt)');
135 if (count($tmpfiles)) {
136 $listoffiles = array_merge($listoffiles, $tmpfiles);
137 }
138 }
139 }
140 $texthelp = $langs->trans("ListOfDirectoriesForModelGenODT");
141 // Add list of substitution keys
142 $texthelp .= '<br>'.$langs->trans("FollowingSubstitutionKeysCanBeUsed").'<br>';
143 $texthelp .= $langs->transnoentitiesnoconv("FullListOnOnlineDocumentation"); // This contains an url, we don't modify it
144
145 $texte .= $form->textwithpicto($texttitle, $texthelp, 1, 'help', '', 1);
146 $texte .= '<div><div style="display: inline-block; min-width: 100px; vertical-align: middle;">';
147 $texte .= '<textarea class="flat" cols="60" name="value1">';
148 $texte .= getDolGlobalString('MEMBER_ADDON_PDF_ODT_PATH');
149 $texte .= '</textarea>';
150 $texte .= '</div><div style="display: inline-block; vertical-align: middle;">';
151 $texte .= '<input type="submit" class="button small reposition" name="modify" value="'.$langs->trans("Modify").'">';
152 $texte .= '<br></div></div>';
153
154 // Scan directories
155 if (count($listofdir)) {
156 $texte .= $langs->trans("NumberOfModelFilesFound").': <b>'.count($listoffiles).'</b>';
157
158 $texte .= '<div id="div_'.get_class($this).'" class="hiddenx">';
159 // Show list of found files
160 foreach ($listoffiles as $file) {
161 $texte .= '- '.$file['name'].' <a href="'.DOL_URL_ROOT.'/document.php?modulepart=doctemplates&file=members/'.urlencode(basename($file['name'])).'">'.img_picto('', 'listlight').'</a>';
162 $texte .= ' &nbsp; <a class="reposition" href="'.$_SERVER["PHP_SELF"].'?modulepart=doctemplates&keyforuploaddir=MEMBER_ADDON_PDF_ODT_PATH&action=deletefile&token='.newToken().'&file='.urlencode(basename($file['name'])).'">'.img_picto('', 'delete').'</a>';
163 $texte .= '<br>';
164 }
165 $texte .= '</div>';
166 }
167 // Add input to upload a new template file.
168 $texte .= '<div>'.$langs->trans("UploadNewTemplate");
169 $maxfilesizearray = getMaxFileSizeArray();
170 $maxmin = $maxfilesizearray['maxmin'];
171 if ($maxmin > 0) {
172 $texte .= '<input type="hidden" name="MAX_FILE_SIZE" value="'.($maxmin * 1024).'">'; // MAX_FILE_SIZE must precede the field type=file
173 }
174 $texte .= ' <input type="file" name="uploadfile">';
175 $texte .= '<input type="hidden" value="MEMBER_ADDON_PDF_ODT_PATH" name="keyforuploaddir">';
176 $texte .= '<input type="submit" class="button small reposition" value="'.dol_escape_htmltag($langs->trans("Upload")).'" name="upload">';
177 $texte .= '</div>';
178 $texte .= '</td>';
179
180 $texte .= '<td rowspan="2" class="tdtop hideonsmartphone">';
181 $texte .= '<span class="opacitymedium">';
182 $texte .= $langs->trans("ExampleOfDirectoriesForModelGen");
183 $texte .= '</span>';
184 $texte .= '</td>';
185 $texte .= '</tr>';
186
187 $texte .= '</table>';
188 $texte .= '</form>';
189
190 return $texte;
191 }
192
193 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
205 public function write_file($object, $outputlangs, $srctemplatepath, $mode = 'member', $nooutput = 0, $filename = 'tmp_cards')
206 {
207 // phpcs:enable
208 global $user, $langs, $conf, $mysoc, $hookmanager;
209
210 if (empty($srctemplatepath)) {
211 dol_syslog("doc_generic_odt::write_file parameter srctemplatepath empty", LOG_WARNING);
212 return -1;
213 }
214
215 // Add odtgeneration hook
216 if (!is_object($hookmanager)) {
217 include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
218 $hookmanager = new HookManager($this->db);
219 }
220 $hookmanager->initHooks(array('odtgeneration'));
221 global $action;
222
223 if (!is_object($outputlangs)) {
224 $outputlangs = $langs;
225 }
226 $sav_charset_output = $outputlangs->charset_output;
227 $outputlangs->charset_output = 'UTF-8';
228
229 // Load translation files required by the page
230 $outputlangs->loadLangs(array("main", "companies", "bills", "dict"));
231
232 if ($conf->adherent->dir_output) {
233 // If $object is id instead of object
234 if (!is_object($object)) {
235 $id = $object;
236 $object = new User($this->db);
237 $result = $object->fetch($id);
238 if ($result < 0) {
239 dol_print_error($this->db, $object->error);
240 return -1;
241 }
242 }
243
244 $object->fetch_thirdparty();
245
246 $dir = $conf->adherent->dir_output;
247 $objectref = dol_sanitizeFileName($object->ref);
248 if (!preg_match('/specimen/i', $objectref)) {
249 $dir .= "/".$objectref;
250 }
251 $file = $dir."/".$objectref.".odt";
252
253 if (!file_exists($dir)) {
254 if (dol_mkdir($dir) < 0) {
255 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
256 return -1;
257 }
258 }
259
260 if (file_exists($dir)) {
261 //print "srctemplatepath=".$srctemplatepath; // Src filename
262 $newfile = basename($srctemplatepath);
263 $newfiletmp = preg_replace('/\.od[ts]/i', '', $newfile);
264 $newfiletmp = preg_replace('/template_/i', '', $newfiletmp);
265 $newfiletmp = preg_replace('/modele_/i', '', $newfiletmp);
266
267 $newfiletmp = $objectref . '_' . $newfiletmp;
268
269 // Get extension (ods or odt)
270 $newfileformat = substr($newfile, strrpos($newfile, '.') + 1);
271 if (getDolGlobalString('MAIN_DOC_USE_TIMING')) {
272 $format = getDolGlobalString('MAIN_DOC_USE_TIMING');
273 if ($format == '1') {
274 $format = '%Y%m%d%H%M%S';
275 }
276 $filename = $newfiletmp . '-' . dol_print_date(dol_now(), $format) . '.' . $newfileformat;
277 } else {
278 $filename = $newfiletmp . '.' . $newfileformat;
279 }
280 $file = $dir . '/' . $filename;
281 //print "newdir=".$dir;
282 //print "newfile=".$newfile;
283 //print "file=".$file;
284 //print "conf->adherent->dir_temp=".$conf->adherent->dir_temp;
285
286 dol_mkdir($conf->adherent->dir_temp);
287 if (!is_writable($conf->adherent->dir_temp)) {
288 $this->error = $langs->transnoentities("ErrorFailedToWriteInTempDirectory", $conf->adherent->dir_temp);
289 dol_syslog('Error in write_file: ' . $this->error, LOG_ERR);
290 return -1;
291 }
292
293 // If CUSTOMER contact defined on member, we use it
294 $usecontact = false;
295 $arrayidcontact = $object->getIdContact('external', 'CUSTOMER');
296 if (count($arrayidcontact) > 0) {
297 $usecontact = true;
298 $result = $object->fetch_contact($arrayidcontact[0]);
299 }
300
301 // Recipient name
302 if (!empty($usecontact)) {
303 // We can use the company of contact instead of thirdparty company
304 if ($object->contact->socid != $object->thirdparty->id && (!isset($conf->global->MAIN_USE_COMPANY_NAME_OF_CONTACT) || !empty($conf->global->MAIN_USE_COMPANY_NAME_OF_CONTACT))) {
305 $object->contact->fetch_thirdparty();
306 $socobject = $object->contact->thirdparty;
307 $contactobject = $object->contact;
308 } else {
309 $socobject = $object->thirdparty;
310 // if we have a CUSTOMER contact and we dont use it as thirdparty recipient we store the contact object for later use
311 $contactobject = $object->contact;
312 }
313 } else {
314 $socobject = $object->thirdparty;
315 }
316
317 // Open and load template
318 require_once ODTPHP_PATH.'odf.php';
319 try {
320 $odfHandler = new Odf(
321 $srctemplatepath,
322 array(
323 'PATH_TO_TMP' => $conf->adherent->dir_temp,
324 'ZIP_PROXY' => 'PclZipProxy', // PhpZipProxy or PclZipProxy. Got "bad compression method" error when using PhpZipProxy.
325 'DELIMITER_LEFT' => '{',
326 'DELIMITER_RIGHT' => '}'
327 )
328 );
329 } catch (Exception $e) {
330 $this->error = $e->getMessage();
331 dol_syslog($e->getMessage(), LOG_WARNING);
332 return -1;
333 }
334
335 // Make substitutions into odt
336 $array_member = $this->getSubstitutionarrayMember($object, $outputlangs);
337 $array_soc = $this->get_substitutionarray_mysoc($mysoc, $outputlangs);
338 $array_thirdparty = $this->get_substitutionarray_thirdparty($socobject, $outputlangs);
339 $array_other = $this->get_substitutionarray_other($outputlangs);
340 // retrieve contact information for use in object as contact_xxx tags
341 $array_thirdparty_contact = array();
342 if ($usecontact && is_object($contactobject)) {
343 $array_thirdparty_contact = $this->get_substitutionarray_contact($contactobject, $outputlangs, 'contact');
344 }
345
346 $tmparray = array_merge($array_member, $array_soc, $array_thirdparty, $array_other, $array_thirdparty_contact);
347 complete_substitutions_array($tmparray, $outputlangs, $object);
348 // Call the ODTSubstitution hook
349 $parameters = array(
350 'file'=>$file,
351 'object'=>$object,
352 'outputlangs'=>$outputlangs,
353 'substitutionarray'=>&$tmparray
354 );
355 $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
356 foreach ($tmparray as $key => $value) {
357 try {
358 if (preg_match('/logo$/', $key)) {
359 // Image
360 if (file_exists($value)) {
361 $odfHandler->setImage($key, $value);
362 } else {
363 $odfHandler->setVars($key, 'ErrorFileNotFound', true, 'UTF-8');
364 }
365 } else {
366 // Text
367 $odfHandler->setVars($key, $value, true, 'UTF-8');
368 }
369 } catch (OdfException $e) {
370 dol_syslog($e->getMessage(), LOG_WARNING);
371 }
372 }
373
374 // Replace labels translated
375 $tmparray = $outputlangs->get_translations_for_substitutions();
376 foreach ($tmparray as $key => $value) {
377 try {
378 $odfHandler->setVars($key, $value, true, 'UTF-8');
379 } catch (OdfException $e) {
380 dol_syslog($e->getMessage(), LOG_WARNING);
381 }
382 }
383
384 // Call the beforeODTSave hook
385 $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs);
386 $reshook = $hookmanager->executeHooks('beforeODTSave', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
387
388 // Write new file
389 if (!empty($conf->global->MAIN_ODT_AS_PDF)) {
390 try {
391 $odfHandler->exportAsAttachedPDF($file);
392 } catch (Exception $e) {
393 $this->error = $e->getMessage();
394 dol_syslog($e->getMessage(), LOG_WARNING);
395 return -1;
396 }
397 } else {
398 try {
399 $odfHandler->saveToDisk($file);
400 } catch (Exception $e) {
401 $this->error = $e->getMessage();
402 dol_syslog($e->getMessage(), LOG_WARNING);
403 return -1;
404 }
405 }
406
407 $reshook = $hookmanager->executeHooks('afterODTCreation', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
408
409 dolChmod($file);
410
411 $odfHandler = null; // Destroy object
412
413 $this->result = array('fullpath'=>$file);
414
415 return 1; // Success
416 } else {
417 $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
418 return -1;
419 }
420 }
421
422 return -1;
423 }
424
425 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
434 public function get_substitutionarray_object($object, $outputlangs, $array_key = 'object')
435 {
436 // phpcs:enable
437 $array_other = array();
438 foreach ($object as $key => $value) {
439 if (!is_array($value) && !is_object($value)) {
440 $array_other[$array_key.'_'.$key] = $value;
441 }
442 }
443 return $array_other;
444 }
445}
getSubstitutionarrayMember($member, $outputlangs)
Define array with couple substitution key => substitution value.
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}...
Class to manage generation of HTML components Only common components must be here.
Class to manage hooks.
Parent class to manage intervention document templates.
Class to manage Dolibarr users.
Class to build documents using ODF templates generator.
info($langs)
Return description of a module.
get_substitutionarray_object($object, $outputlangs, $array_key='object')
get substitution array for object
write_file($object, $outputlangs, $srctemplatepath, $mode='member', $nooutput=0, $filename='tmp_cards')
Function to build a document on disk using the generic odt 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_print_error($db='', $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
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.