dolibarr  13.0.2
doc_generic_usergroup_odt.modules.php
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 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 
27 require_once DOL_DOCUMENT_ROOT.'/core/modules/usergroup/modules_usergroup.class.php';
28 require_once DOL_DOCUMENT_ROOT.'/user/class/user.class.php';
29 require_once DOL_DOCUMENT_ROOT.'/user/class/usergroup.class.php';
30 require_once DOL_DOCUMENT_ROOT.'/core/lib/company.lib.php';
31 require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
32 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
33 require_once DOL_DOCUMENT_ROOT.'/core/lib/doc.lib.php';
34 
35 
40 {
45  public $emetteur;
46 
51  public $phpmin = array(5, 6);
52 
57  public $version = 'dolibarr';
58 
59 
65  public function __construct($db)
66  {
67  global $conf, $langs, $mysoc;
68 
69  // Load translation files required by the page
70  $langs->loadLangs(array("main", "companies"));
71 
72  $this->db = $db;
73  $this->name = "ODT templates";
74  $this->description = $langs->trans("DocumentModelOdt");
75  $this->scandir = 'USERGROUP_ADDON_PDF_ODT_PATH'; // Name of constant that is used to save list of directories to scan
76 
77  // Page size for A4 format
78  $this->type = 'odt';
79  $this->page_largeur = 0;
80  $this->page_hauteur = 0;
81  $this->format = array($this->page_largeur, $this->page_hauteur);
82  $this->marge_gauche = 0;
83  $this->marge_droite = 0;
84  $this->marge_haute = 0;
85  $this->marge_basse = 0;
86 
87  $this->option_logo = 1; // Affiche logo
88  $this->option_tva = 0; // Gere option tva USERGROUP_TVAOPTION
89  $this->option_modereg = 0; // Affiche mode reglement
90  $this->option_condreg = 0; // Affiche conditions reglement
91  $this->option_codeproduitservice = 0; // Affiche code produit-service
92  $this->option_multilang = 1; // Dispo en plusieurs langues
93  $this->option_escompte = 0; // Affiche si il y a eu escompte
94  $this->option_credit_note = 0; // Support credit notes
95  $this->option_freetext = 1; // Support add of a personalised text
96  $this->option_draft_watermark = 0; // Support add of a watermark on drafts
97 
98  // Recupere emetteur
99  $this->emetteur = $mysoc;
100  if (!$this->emetteur->country_code) $this->emetteur->country_code = substr($langs->defaultlang, -2); // By default if not defined
101  }
102 
103 
110  public function info($langs)
111  {
112  global $conf, $langs;
113 
114  // Load translation files required by the page
115  $langs->loadLangs(array("errors", "companies"));
116 
117  $form = new Form($this->db);
118 
119  $texte = $this->description.".<br>\n";
120  $texte .= '<form action="'.$_SERVER["PHP_SELF"].'" method="POST">';
121  $texte .= '<input type="hidden" name="token" value="'.newToken().'">';
122  $texte .= '<input type="hidden" name="action" value="setModuleOptions">';
123  $texte .= '<input type="hidden" name="param1" value="USERGROUP_ADDON_PDF_ODT_PATH">';
124  if ($conf->global->MAIN_PROPAL_CHOOSE_ODT_DOCUMENT > 0)
125  {
126  $texte .= '<input type="hidden" name="param2" value="USERGROUP_ADDON_PDF_ODT_DEFAULT">';
127  $texte .= '<input type="hidden" name="param3" value="USERGROUP_ADDON_PDF_ODT_TOBILL">';
128  $texte .= '<input type="hidden" name="param4" value="USERGROUP_ADDON_PDF_ODT_CLOSED">';
129  }
130  $texte .= '<table class="nobordernopadding" width="100%">';
131 
132  // List of directories area
133  $texte .= '<tr><td>';
134  $texttitle = $langs->trans("ListOfDirectories");
135  $listofdir = explode(',', preg_replace('/[\r\n]+/', ',', trim($conf->global->USERGROUP_ADDON_PDF_ODT_PATH)));
136  $listoffiles = array();
137  foreach ($listofdir as $key=>$tmpdir)
138  {
139  $tmpdir = trim($tmpdir);
140  $tmpdir = preg_replace('/DOL_DATA_ROOT/', DOL_DATA_ROOT, $tmpdir);
141  if (!$tmpdir) {
142  unset($listofdir[$key]); continue;
143  }
144  if (!is_dir($tmpdir)) $texttitle .= img_warning($langs->trans("ErrorDirNotFound", $tmpdir), 0);
145  else {
146  $tmpfiles = dol_dir_list($tmpdir, 'files', 0, '\.(ods|odt)');
147  if (count($tmpfiles)) $listoffiles = array_merge($listoffiles, $tmpfiles);
148  }
149  }
150  $texthelp = $langs->trans("ListOfDirectoriesForModelGenODT");
151  // Add list of substitution keys
152  $texthelp .= '<br>'.$langs->trans("FollowingSubstitutionKeysCanBeUsed").'<br>';
153  $texthelp .= $langs->transnoentitiesnoconv("FullListOnOnlineDocumentation"); // This contains an url, we don't modify it
154 
155  $texte .= $form->textwithpicto($texttitle, $texthelp, 1, 'help', '', 1);
156  $texte .= '<div><div style="display: inline-block; min-width: 100px; vertical-align: middle;">';
157  $texte .= '<textarea class="flat" cols="60" name="value1">';
158  $texte .= $conf->global->USERGROUP_ADDON_PDF_ODT_PATH;
159  $texte .= '</textarea>';
160  $texte .= '</div><div style="display: inline-block; vertical-align: middle;">';
161  $texte .= '<input type="submit" class="button" value="'.$langs->trans("Modify").'" name="Button">';
162  $texte .= '<br></div></div>';
163 
164  // Scan directories
165  if (count($listofdir))
166  {
167  $texte .= $langs->trans("NumberOfModelFilesFound").': <b>'.count($listoffiles).'</b>';
168 
169  if ($conf->global->MAIN_PROPAL_CHOOSE_ODT_DOCUMENT > 0)
170  {
171  // Model for creation
172  $liste = ModelePDFUserGroup::liste_modeles($this->db);
173  $texte .= '<table width="50%;">';
174  $texte .= '<tr>';
175  $texte .= '<td width="60%;">'.$langs->trans("DefaultModelPropalCreate").'</td>';
176  $texte .= '<td colspan="">';
177  $texte .= $form->selectarray('value2', $liste, $conf->global->USERGROUP_ADDON_PDF_ODT_DEFAULT);
178  $texte .= "</td></tr>";
179 
180  $texte .= '<tr>';
181  $texte .= '<td width="60%;">'.$langs->trans("DefaultModelPropalToBill").'</td>';
182  $texte .= '<td colspan="">';
183  $texte .= $form->selectarray('value3', $liste, $conf->global->USERGROUP_ADDON_PDF_ODT_TOBILL);
184  $texte .= "</td></tr>";
185  $texte .= '<tr>';
186 
187  $texte .= '<td width="60%;">'.$langs->trans("DefaultModelPropalClosed").'</td>';
188  $texte .= '<td colspan="">';
189  $texte .= $form->selectarray('value4', $liste, $conf->global->USERGROUP_ADDON_PDF_ODT_CLOSED);
190  $texte .= "</td></tr>";
191  $texte .= '</table>';
192  }
193  }
194 
195  $texte .= '</td>';
196 
197  $texte .= '<td rowspan="2" class="tdtop hideonsmartphone">';
198  $texte .= $langs->trans("ExampleOfDirectoriesForModelGen");
199  $texte .= '</td>';
200  $texte .= '</tr>';
201 
202  $texte .= '</table>';
203  $texte .= '</form>';
204 
205  return $texte;
206  }
207 
208  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
220  public function write_file($object, $outputlangs, $srctemplatepath, $hidedetails = 0, $hidedesc = 0, $hideref = 0)
221  {
222  // phpcs:enable
223  global $user, $langs, $conf, $mysoc, $hookmanager;
224 
225  if (empty($srctemplatepath))
226  {
227  dol_syslog("doc_generic_odt::write_file parameter srctemplatepath empty", LOG_WARNING);
228  return -1;
229  }
230 
231  // Add odtgeneration hook
232  if (!is_object($hookmanager))
233  {
234  include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
235  $hookmanager = new HookManager($this->db);
236  }
237  $hookmanager->initHooks(array('odtgeneration'));
238  global $action;
239 
240  if (!is_object($outputlangs)) $outputlangs = $langs;
241  $sav_charset_output = $outputlangs->charset_output;
242  $outputlangs->charset_output = 'UTF-8';
243 
244  // Load translation files required by the page
245  $outputlangs->loadLangs(array("main", "companies", "bills", "dict"));
246 
247  if ($conf->user->dir_output)
248  {
249  // If $object is id instead of object
250  if (!is_object($object))
251  {
252  $id = $object;
253  $object = new UserGroup($this->db);
254  $result = $object->fetch($id);
255  if ($result < 0)
256  {
257  dol_print_error($this->db, $object->error);
258  return -1;
259  }
260  }
261 
262  $dir = $conf->usergroup->dir_output;
263  $objectref = dol_sanitizeFileName($object->ref);
264  if (!preg_match('/specimen/i', $objectref)) $dir .= "/".$objectref;
265  $file = $dir."/".$objectref.".odt";
266 
267  if (!file_exists($dir))
268  {
269  if (dol_mkdir($dir) < 0)
270  {
271  $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
272  return -1;
273  }
274  }
275 
276  if (file_exists($dir))
277  {
278  //print "srctemplatepath=".$srctemplatepath; // Src filename
279  $newfile = basename($srctemplatepath);
280  $newfiletmp = preg_replace('/\.od(t|s)/i', '', $newfile);
281  $newfiletmp = preg_replace('/template_/i', '', $newfiletmp);
282  $newfiletmp = preg_replace('/modele_/i', '', $newfiletmp);
283 
284  $newfiletmp = $objectref.'_'.$newfiletmp;
285 
286  // Get extension (ods or odt)
287  $newfileformat = substr($newfile, strrpos($newfile, '.') + 1);
288  if (!empty($conf->global->MAIN_DOC_USE_TIMING))
289  {
290  $format = $conf->global->MAIN_DOC_USE_TIMING;
291  if ($format == '1') $format = '%Y%m%d%H%M%S';
292  $filename = $newfiletmp.'-'.dol_print_date(dol_now(), $format).'.'.$newfileformat;
293  } else {
294  $filename = $newfiletmp.'.'.$newfileformat;
295  }
296  $file = $dir.'/'.$filename;
297  //print "newdir=".$dir;
298  //print "newfile=".$newfile;
299  //print "file=".$file;
300  //print "conf->user->dir_temp=".$conf->user->dir_temp;
301 
302  dol_mkdir($conf->user->dir_temp);
303 
304 
305  // If CUSTOMER contact defined on user, we use it
306  $usecontact = false;
307  $arrayidcontact = $object->getIdContact('external', 'CUSTOMER');
308  if (count($arrayidcontact) > 0)
309  {
310  $usecontact = true;
311  $result = $object->fetch_contact($arrayidcontact[0]);
312  }
313 
314  // Recipient name
315  if (!empty($usecontact))
316  {
317  if ($usecontact && ($object->contact->fk_soc != $object->thirdparty->id && (!isset($conf->global->MAIN_USE_COMPANY_NAME_OF_CONTACT) || !empty($conf->global->MAIN_USE_COMPANY_NAME_OF_CONTACT)))) {
318  $socobject = $object->contact;
319  } else {
320  $socobject = $object->thirdparty;
321  // if we have a CUSTOMER contact and we dont use it as recipient we store the contact object for later use
322  $contactobject = $object->contact;
323  }
324  } else {
325  $socobject = $object->thirdparty;
326  }
327  // Make substitution
328  $substitutionarray = array(
329  '__FROM_NAME__' => $this->emetteur->name,
330  '__FROM_EMAIL__' => $this->emetteur->email,
331  '__TOTAL_TTC__' => $object->total_ttc,
332  '__TOTAL_HT__' => $object->total_ht,
333  '__TOTAL_VAT__' => $object->total_tva
334  );
335  complete_substitutions_array($substitutionarray, $langs, $object);
336  // Call the ODTSubstitution hook
337  $parameters = array('file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$substitutionarray);
338  $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
339 
340  // Line of free text
341  $newfreetext = '';
342  $paramfreetext = 'user_FREE_TEXT';
343  if (!empty($conf->global->$paramfreetext))
344  {
345  $newfreetext = make_substitutions($conf->global->$paramfreetext, $substitutionarray);
346  }
347 
348  // Open and load template
349  require_once ODTPHP_PATH.'odf.php';
350  try {
351  $odfHandler = new odf(
352  $srctemplatepath,
353  array(
354  'PATH_TO_TMP' => $conf->user->dir_temp,
355  'ZIP_PROXY' => 'PclZipProxy', // PhpZipProxy or PclZipProxy. Got "bad compression method" error when using PhpZipProxy.
356  'DELIMITER_LEFT' => '{',
357  'DELIMITER_RIGHT' => '}'
358  )
359  );
360  } catch (Exception $e) {
361  $this->error = $e->getMessage();
362  dol_syslog($e->getMessage(), LOG_WARNING);
363  return -1;
364  }
365  // After construction $odfHandler->contentXml contains content and
366  // [!-- BEGIN row.lines --]*[!-- END row.lines --] has been replaced by
367  // [!-- BEGIN lines --]*[!-- END lines --]
368  //print html_entity_decode($odfHandler->__toString());
369  //print exit;
370 
371 
372  // Make substitutions into odt of freetext
373  try {
374  $odfHandler->setVars('free_text', $newfreetext, true, 'UTF-8');
375  } catch (OdfException $e)
376  {
377  dol_syslog($e->getMessage(), LOG_WARNING);
378  }
379 
380  // Make substitutions into odt
381  $array_user = $this->get_substitutionarray_user($user, $outputlangs);
382  $array_global = $this->get_substitutionarray_each_var_object($object, $outputlangs);
383  $array_soc = $this->get_substitutionarray_mysoc($mysoc, $outputlangs);
384  $array_thirdparty = $this->get_substitutionarray_thirdparty($socobject, $outputlangs);
385  $array_objet = $this->get_substitutionarray_each_var_object($object, $outputlangs);
386  $array_other = $this->get_substitutionarray_other($outputlangs);
387  // retrieve contact information for use in object as contact_xxx tags
388  $array_thirdparty_contact = array();
389  if ($usecontact && is_object($contactobject)) $array_thirdparty_contact = $this->get_substitutionarray_contact($contactobject, $outputlangs, 'contact');
390 
391  $tmparray = array_merge($array_global, $array_user, $array_soc, $array_thirdparty, $array_objet, $array_other, $array_thirdparty_contact);
392  complete_substitutions_array($tmparray, $outputlangs, $object);
393  $object->fetch_optionals();
394  // Call the ODTSubstitution hook
395  $parameters = array('file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
396  $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
397  foreach ($tmparray as $key=>$value)
398  {
399  try {
400  if (preg_match('/logo$/', $key)) // Image
401  {
402  if (file_exists($value)) $odfHandler->setImage($key, $value);
403  else $odfHandler->setVars($key, 'ErrorFileNotFound', true, 'UTF-8');
404  } else // Text
405  {
406  $odfHandler->setVars($key, $value, true, 'UTF-8');
407  }
408  } catch (OdfException $e)
409  {
410  dol_syslog($e->getMessage(), LOG_WARNING);
411  }
412  }
413  // Replace tags of lines
414  try {
415  $foundtagforlines = 1;
416  try {
417  $listlines = $odfHandler->setSegment('lines');
418  } catch (OdfException $e)
419  {
420  // We may arrive here if tags for lines not present into template
421  $foundtagforlines = 0;
422  dol_syslog($e->getMessage(), LOG_INFO);
423  }
424  if ($foundtagforlines)
425  {
426  foreach ($object->members as $u)
427  {
428  $tmparray = $this->get_substitutionarray_each_var_object($u, $outputlangs);
429  unset($tmparray['object_pass']);
430  unset($tmparray['object_pass_indatabase']);
431  complete_substitutions_array($tmparray, $outputlangs, $object, $user, "completesubstitutionarray_users");
432  // Call the ODTSubstitutionLine hook
433  $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray, 'line'=>$u);
434  $reshook = $hookmanager->executeHooks('ODTSubstitutionLine', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
435  foreach ($tmparray as $key => $val)
436  {
437  try {
438  if (!is_array($val)) {
439  $listlines->setVars($key, $val, true, 'UTF-8');
440  }
441  } catch (OdfException $e)
442  {
443  dol_syslog($e->getMessage(), LOG_WARNING);
444  } catch (SegmentException $e)
445  {
446  dol_syslog($e->getMessage(), LOG_WARNING);
447  }
448  }
449  $listlines->merge();
450  }
451  $odfHandler->mergeSegment($listlines);
452  }
453  } catch (OdfException $e)
454  {
455  $this->error = $e->getMessage();
456  dol_syslog($this->error, LOG_WARNING);
457  return -1;
458  }
459 
460  // Replace labels translated
461  $tmparray = $outputlangs->get_translations_for_substitutions();
462  foreach ($tmparray as $key => $value)
463  {
464  try {
465  $odfHandler->setVars($key, $value, true, 'UTF-8');
466  } catch (OdfException $e)
467  {
468  dol_syslog($e->getMessage(), LOG_WARNING);
469  }
470  }
471 
472  // Call the beforeODTSave hook
473  $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs);
474  $reshook = $hookmanager->executeHooks('beforeODTSave', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
475 
476  // Write new file
477  if (!empty($conf->global->MAIN_ODT_AS_PDF)) {
478  try {
479  $odfHandler->exportAsAttachedPDF($file);
480  } catch (Exception $e) {
481  $this->error = $e->getMessage();
482  dol_syslog($e->getMessage(), LOG_WARNING);
483  return -1;
484  }
485  } else {
486  try {
487  $odfHandler->saveToDisk($file);
488  } catch (Exception $e) {
489  $this->error = $e->getMessage();
490  dol_syslog($e->getMessage(), LOG_WARNING);
491  return -1;
492  }
493  }
494 
495  $reshook = $hookmanager->executeHooks('afterODTCreation', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
496 
497  if (!empty($conf->global->MAIN_UMASK))
498  @chmod($file, octdec($conf->global->MAIN_UMASK));
499 
500  $odfHandler = null; // Destroy object
501 
502  $this->result = array('fullpath'=>$file);
503 
504  return 1; // Success
505  } else {
506  $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
507  return -1;
508  }
509  }
510 
511  return -1;
512  }
513 }
</td > param sortfield sortorder printFieldListOption< tdclass="liste_titremaxwidthsearchright"></td ></tr >< trclass="liste_titre">< inputtype="checkbox"onClick="toggle(this)"/> Ref p ref Label p label Duration p duration center DesiredStock p desiredstock right StockLimitShort p seuil_stock_alerte right stock_physique right stock_real_warehouse right Ordered right StockToBuy right SupplierRef right param sortfield sortorder printFieldListTitle warehouseinternal SELECT description FROM product_lang WHERE qty< br > qty qty qty StockTooLow StockTooLow help help help< trclass="oddeven">< td >< inputtype="checkbox"class="check"name="choose'.$i.'"></td >< tdclass="nowrap"> stock</td >< td >< inputtype="hidden"name="desc'.$i.'"value="'.dol_escape_htmltag($objp-> description
Only used if Module[ID]Desc translation string is not found.
Definition: replenish.php:750
get_substitutionarray_each_var_object(&$object, $outputlangs, $recursive=true)
Define array with couple substitution key =&gt; substitution value.
dol_now($mode= 'auto')
Return date for now.
get_substitutionarray_user($user, $outputlangs)
Define array with couple substitution key =&gt; substitution value.
Class to manage user groups.
info($langs)
Return description of a module.
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition: repair.php:108
$conf db
API class for accounts.
Definition: inc.php:54
img_warning($titlealt= 'default', $moreatt= '', $morecss= 'pictowarning')
Show warning logo.
Class to manage hooks.
get_substitutionarray_contact($object, $outputlangs, $array_key= 'object')
Define array with couple substitution key =&gt; substitution value.
Parent class to manage intervention document templates.
Class to manage generation of HTML components Only common components must be here.
write_file($object, $outputlangs, $srctemplatepath, $hidedetails=0, $hidedesc=0, $hideref=0)
Function to build a document on disk using the generic odt module.
Class to build documents using ODF templates generator.
get_substitutionarray_thirdparty($object, $outputlangs, $array_key= 'company')
Define array with couple substitution key =&gt; substitution value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename= '', $restricttologhandler= '', $logcontext=null)
Write log message into outputs.
get_substitutionarray_mysoc($mysoc, $outputlangs)
Define array with couple substitution key =&gt; substitution value.
dol_sanitizeFileName($str, $newstr= '_', $unaccent=1)
Clean a string to use it as a file name.
dol_dir_list($path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0)
Scan a directory and return a list of files/directories.
Definition: files.lib.php:60
static liste_modeles($db, $maxfilenamelength=0)
Return list of active generation modules.
dol_print_date($time, $format= '', $tzoutput= 'auto', $outputlangs= '', $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
dol_print_error($db= '', $error= '', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
make_substitutions($text, $substitutionarray, $outputlangs=null)
Make substitution into a text string, replacing keys with vals from $substitutionarray (oldval=&gt;newva...
get_substitutionarray_other($outputlangs)
Define array with couple substitution key =&gt; substitution value.
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition: repair.php:105
dol_mkdir($dir, $dataroot= '', $newmask=null)
Creation of a directory (this can create recursive subdir)
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 &quot;subst...