dolibarr  13.0.2
doc_generic_stock_odt.modules.php
Go to the documentation of this file.
1 <?php
2 /* Copyright (C) 2010-2012 Laurent Destailleur <eldy@stocks.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/stock/modules_stock.php';
28 require_once DOL_DOCUMENT_ROOT.'/product/stock/class/entrepot.class.php';
29 require_once DOL_DOCUMENT_ROOT.'/core/lib/company.lib.php';
30 require_once DOL_DOCUMENT_ROOT.'/core/lib/functions2.lib.php';
31 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
32 require_once DOL_DOCUMENT_ROOT.'/core/lib/doc.lib.php';
33 
34 
39 {
44  public $emetteur;
45 
50  public $phpmin = array(5, 6);
51 
55  public $version = 'dolibarr';
56 
57 
63  public function __construct($db)
64  {
65  global $conf, $langs, $mysoc;
66 
67  // Load translation files required by the page
68  $langs->loadLangs(array("main", "companies"));
69 
70  $this->db = $db;
71  $this->name = "ODT templates";
72  $this->description = $langs->trans("DocumentModelOdt");
73  $this->scandir = 'STOCK_ADDON_PDF_ODT_PATH'; // Name of constant that is used to save list of directories to scan
74 
75  // Page size for A4 format
76  $this->type = 'odt';
77  $this->page_largeur = 0;
78  $this->page_hauteur = 0;
79  $this->format = array($this->page_largeur, $this->page_hauteur);
80  $this->marge_gauche = 0;
81  $this->marge_droite = 0;
82  $this->marge_haute = 0;
83  $this->marge_basse = 0;
84 
85  $this->option_logo = 1; // Affiche logo
86  $this->option_tva = 0; // Gere option tva STOCK_TVAOPTION
87  $this->option_modereg = 0; // Affiche mode reglement
88  $this->option_condreg = 0; // Affiche conditions reglement
89  $this->option_codeproduitservice = 0; // Affiche code produit-service
90  $this->option_multilang = 1; // Dispo en plusieurs langues
91  $this->option_escompte = 0; // Affiche si il y a eu escompte
92  $this->option_credit_note = 0; // Support credit notes
93  $this->option_freetext = 1; // Support add of a personalised text
94  $this->option_draft_watermark = 0; // Support add of a watermark on drafts
95 
96  // Recupere emetteur
97  $this->emetteur = $mysoc;
98  if (!$this->emetteur->country_code) $this->emetteur->country_code = substr($langs->defaultlang, -2); // By default if not defined
99  }
100 
101 
108  public function info($langs)
109  {
110  global $conf, $langs;
111 
112  // Load translation files required by the page
113  $langs->loadLangs(array("errors", "companies"));
114 
115  $form = new Form($this->db);
116 
117  $texte = $this->description.".<br>\n";
118  $texte .= '<form action="'.$_SERVER["PHP_SELF"].'" method="POST" enctype="multipart/form-data">';
119  $texte .= '<input type="hidden" name="token" value="'.newToken().'">';
120  $texte .= '<input type="hidden" name="action" value="setModuleOptions">';
121  $texte .= '<input type="hidden" name="param1" value="STOCK_ADDON_PDF_ODT_PATH">';
122  $texte .= '<table class="nobordernopadding" width="100%">';
123 
124  // List of directories area
125  $texte .= '<tr><td>';
126  $texttitle = $langs->trans("ListOfDirectories");
127  $listofdir = explode(',', preg_replace('/[\r\n]+/', ',', trim($conf->global->STOCK_ADDON_PDF_ODT_PATH)));
128  $listoffiles = array();
129  foreach ($listofdir as $key=>$tmpdir)
130  {
131  $tmpdir = trim($tmpdir);
132  $tmpdir = preg_replace('/DOL_DATA_ROOT/', DOL_DATA_ROOT, $tmpdir);
133  if (!$tmpdir) {
134  unset($listofdir[$key]); continue;
135  }
136  if (!is_dir($tmpdir)) $texttitle .= img_warning($langs->trans("ErrorDirNotFound", $tmpdir), 0);
137  else {
138  $tmpfiles = dol_dir_list($tmpdir, 'files', 0, '\.(ods|odt)');
139  if (count($tmpfiles)) $listoffiles = array_merge($listoffiles, $tmpfiles);
140  }
141  }
142  $texthelp = $langs->trans("ListOfDirectoriesForModelGenODT");
143  // Add list of substitution keys
144  $texthelp .= '<br>'.$langs->trans("FollowingSubstitutionKeysCanBeUsed").'<br>';
145  $texthelp .= $langs->transnoentitiesnoconv("FullListOnOnlineDocumentation"); // This contains an url, we don't modify it
146 
147  $texte .= $form->textwithpicto($texttitle, $texthelp, 1, 'help', '', 1);
148  $texte .= '<div><div style="display: inline-block; min-width: 100px; vertical-align: middle;">';
149  $texte .= '<textarea class="flat" cols="60" name="value1">';
150  $texte .= $conf->global->STOCK_ADDON_PDF_ODT_PATH;
151  $texte .= '</textarea>';
152  $texte .= '</div><div style="display: inline-block; vertical-align: middle;">';
153  $texte .= '<input type="submit" class="button" value="'.$langs->trans("Modify").'" name="Button">';
154  $texte .= '<br></div></div>';
155 
156  // Scan directories
157  $nbofiles = count($listoffiles);
158  if (!empty($conf->global->STOCK_ADDON_PDF_ODT_PATH))
159  {
160  $texte .= $langs->trans("NumberOfModelFilesFound").': <b>';
161  //$texte.=$nbofiles?'<a id="a_'.get_class($this).'" href="#">':'';
162  $texte .= count($listoffiles);
163  //$texte.=$nbofiles?'</a>':'';
164  $texte .= '</b>';
165  }
166 
167  if ($nbofiles)
168  {
169  $texte .= '<div id="div_'.get_class($this).'" class="hiddenx">';
170  // Show list of found files
171  foreach ($listoffiles as $file) {
172  $texte .= '- '.$file['name'].' <a href="'.DOL_URL_ROOT.'/document.php?modulepart=doctemplates&file=stocks/'.urlencode(basename($file['name'])).'">'.img_picto('', 'listlight').'</a><br>';
173  }
174  $texte .= '</div>';
175  }
176  // Add input to upload a new template file.
177  $texte .= '<div>'.$langs->trans("UploadNewTemplate").' <input type="file" name="uploadfile">';
178  $texte .= '<input type="hidden" value="STOCK_ADDON_PDF_ODT_PATH" name="keyforuploaddir">';
179  $texte .= '<input type="submit" class="button" value="'.dol_escape_htmltag($langs->trans("Upload")).'" name="upload">';
180  $texte .= '</div>';
181 
182  $texte .= '</td>';
183 
184  $texte .= '<td rowspan="2" class="tdtop hideonsmartphone">';
185  $texte .= $langs->trans("ExampleOfDirectoriesForModelGen");
186  $texte .= '</td>';
187  $texte .= '</tr>';
188 
189  $texte .= '</table>';
190  $texte .= '</form>';
191 
192  return $texte;
193  }
194 
195  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
207  public function write_file($object, $outputlangs, $srctemplatepath, $hidedetails = 0, $hidedesc = 0, $hideref = 0)
208  {
209  // phpcs:enable
210  global $stock, $langs, $conf, $mysoc, $hookmanager, $user;
211 
212  if (empty($srctemplatepath))
213  {
214  dol_syslog("doc_generic_odt::write_file parameter srctemplatepath empty", LOG_WARNING);
215  return -1;
216  }
217 
218  // Add odtgeneration hook
219  if (!is_object($hookmanager))
220  {
221  include_once DOL_DOCUMENT_ROOT.'/core/class/hookmanager.class.php';
222  $hookmanager = new HookManager($this->db);
223  }
224  $hookmanager->initHooks(array('odtgeneration'));
225  global $action;
226 
227  if (!is_object($outputlangs)) $outputlangs = $langs;
228  $sav_charset_output = $outputlangs->charset_output;
229  $outputlangs->charset_output = 'UTF-8';
230 
231  // Load translation files required by the page
232  $outputlangs->loadLangs(array("main", "dict", "companies", "bills"));
233 
234  if ($conf->product->dir_output)
235  {
236  // If $object is id instead of object
237  if (!is_object($object))
238  {
239  $id = $object;
240  $object = new Entrepot($this->db);
241  $result = $object->fetch($id);
242  if ($result < 0)
243  {
244  dol_print_error($this->db, $object->error);
245  return -1;
246  }
247  }
248 
249  $stockFournisseur = new ProductFournisseur($this->db);
250  $supplierprices = $stockFournisseur->list_stock_fournisseur_price($object->id);
251  $object->supplierprices = $supplierprices;
252 
253  $dir = $conf->product->dir_output;
254  $objectref = dol_sanitizeFileName($object->ref);
255  if (!preg_match('/specimen/i', $objectref)) $dir .= "/".$objectref;
256  $file = $dir."/".$objectref.".odt";
257 
258  if (!file_exists($dir))
259  {
260  if (dol_mkdir($dir) < 0)
261  {
262  $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
263  return -1;
264  }
265  }
266 
267  if (file_exists($dir))
268  {
269  //print "srctemplatepath=".$srctemplatepath; // Src filename
270  $newfile = basename($srctemplatepath);
271  $newfiletmp = preg_replace('/\.od(t|s)/i', '', $newfile);
272  $newfiletmp = preg_replace('/template_/i', '', $newfiletmp);
273  $newfiletmp = preg_replace('/modele_/i', '', $newfiletmp);
274 
275  $newfiletmp = $objectref.'_'.$newfiletmp;
276 
277  // Get extension (ods or odt)
278  $newfileformat = substr($newfile, strrpos($newfile, '.') + 1);
279  if (!empty($conf->global->MAIN_DOC_USE_TIMING))
280  {
281  $format = $conf->global->MAIN_DOC_USE_TIMING;
282  if ($format == '1') $format = '%Y%m%d%H%M%S';
283  $filename = $newfiletmp.'-'.dol_print_date(dol_now(), $format).'.'.$newfileformat;
284  } else {
285  $filename = $newfiletmp.'.'.$newfileformat;
286  }
287  $file = $dir.'/'.$filename;
288  //print "newdir=".$dir;
289  //print "newfile=".$newfile;
290  //print "file=".$file;
291  //print "conf->product->dir_temp=".$conf->product->dir_temp;
292 
293  dol_mkdir($conf->product->dir_temp);
294 
295 
296  // If CUSTOMER contact defined on stock, we use it
297  $usecontact = false;
298  $arrayidcontact = $object->getIdContact('external', 'CUSTOMER');
299  if (count($arrayidcontact) > 0)
300  {
301  $usecontact = true;
302  $result = $object->fetch_contact($arrayidcontact[0]);
303  }
304 
305  // Recipient name
306  $contactobject = null;
307  if (!empty($usecontact))
308  {
309  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)))) {
310  $socobject = $object->contact;
311  } else {
312  $socobject = $object->thirdparty;
313  // if we have a CUSTOMER contact and we dont use it as recipient we store the contact object for later use
314  $contactobject = $object->contact;
315  }
316  } else {
317  $socobject = $object->thirdparty;
318  }
319  // Make substitution
320  $substitutionarray = array(
321  '__FROM_NAME__' => $this->emetteur->name,
322  '__FROM_EMAIL__' => $this->emetteur->email,
323  '__TOTAL_TTC__' => $object->total_ttc,
324  '__TOTAL_HT__' => $object->total_ht,
325  '__TOTAL_VAT__' => $object->total_tva
326  );
327  complete_substitutions_array($substitutionarray, $langs, $object);
328  // Call the ODTSubstitution hook
329  $parameters = array('file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$substitutionarray);
330  $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
331 
332  // Line of free text
333  $newfreetext = '';
334  $paramfreetext = 'stock_FREE_TEXT';
335  if (!empty($conf->global->$paramfreetext))
336  {
337  $newfreetext = make_substitutions($conf->global->$paramfreetext, $substitutionarray);
338  }
339 
340  // Open and load template
341  require_once ODTPHP_PATH.'odf.php';
342  try {
343  $odfHandler = new odf(
344  $srctemplatepath,
345  array(
346  'PATH_TO_TMP' => $conf->product->dir_temp,
347  'ZIP_PROXY' => 'PclZipProxy', // PhpZipProxy or PclZipProxy. Got "bad compression method" error when using PhpZipProxy.
348  'DELIMITER_LEFT' => '{',
349  'DELIMITER_RIGHT' => '}'
350  )
351  );
352  } catch (Exception $e)
353  {
354  $this->error = $e->getMessage();
355  dol_syslog($e->getMessage(), LOG_INFO);
356  return -1;
357  }
358  // After construction $odfHandler->contentXml contains content and
359  // [!-- BEGIN row.lines --]*[!-- END row.lines --] has been replaced by
360  // [!-- BEGIN lines --]*[!-- END lines --]
361  //print html_entity_decode($odfHandler->__toString());
362  //print exit;
363 
364  $object->fetch_optionals();
365 
366  // Make substitutions into odt of freetext
367  try {
368  $odfHandler->setVars('free_text', $newfreetext, true, 'UTF-8');
369  } catch (OdfException $e)
370  {
371  dol_syslog($e->getMessage(), LOG_INFO);
372  }
373 
374  // Define substitution array
375  $substitutionarray = getCommonSubstitutionArray($outputlangs, 0, null, $object);
376  $array_object_from_properties = $this->get_substitutionarray_each_var_object($object, $outputlangs);
377  //$array_objet=$this->get_substitutionarray_object($object,$outputlangs);
378  $array_user = $this->get_substitutionarray_user($user, $outputlangs);
379  $array_soc = $this->get_substitutionarray_mysoc($mysoc, $outputlangs);
380  $array_thirdparty = $this->get_substitutionarray_thirdparty($socobject, $outputlangs);
381  $array_other = $this->get_substitutionarray_other($outputlangs);
382  // retrieve contact information for use in stock as contact_xxx tags
383  $array_thirdparty_contact = array();
384  if ($usecontact && is_object($contactobject)) $array_thirdparty_contact = $this->get_substitutionarray_contact($contactobject, $outputlangs, 'contact');
385 
386  $tmparray = array_merge($substitutionarray, $array_object_from_properties, $array_user, $array_soc, $array_thirdparty, $array_other, $array_thirdparty_contact);
387  complete_substitutions_array($tmparray, $outputlangs, $object);
388 
389  // Call the ODTSubstitution hook
390  $parameters = array('file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$tmparray);
391  $reshook = $hookmanager->executeHooks('ODTSubstitution', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
392 
393  foreach ($tmparray as $key=>$value)
394  {
395  try {
396  if (preg_match('/logo$/', $key)) // Image
397  {
398  if (file_exists($value)) $odfHandler->setImage($key, $value);
399  else $odfHandler->setVars($key, 'ErrorFileNotFound', true, 'UTF-8');
400  } else // Text
401  {
402  $odfHandler->setVars($key, $value, true, 'UTF-8');
403  }
404  } catch (OdfException $e)
405  {
406  dol_syslog($e->getMessage(), LOG_INFO);
407  }
408  }
409  // Replace tags of lines
410  try {
411  $listlines = $odfHandler->setSegment('supplierprices');
412  if (!empty($object->supplierprices)) {
413  foreach ($object->supplierprices as $supplierprice)
414  {
415  $array_lines = $this->get_substitutionarray_each_var_object($supplierprice, $outputlangs);
416  complete_substitutions_array($array_lines, $outputlangs, $object, $supplierprice, "completesubstitutionarray_lines");
417  // Call the ODTSubstitutionLine hook
418  $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs, 'substitutionarray'=>&$array_lines, 'line'=>$supplierprice);
419  $reshook = $hookmanager->executeHooks('ODTSubstitutionLine', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
420  foreach ($array_lines as $key => $val)
421  {
422  try {
423  $listlines->setVars($key, $val, true, 'UTF-8');
424  } catch (OdfException $e)
425  {
426  dol_syslog($e->getMessage(), LOG_INFO);
427  } catch (SegmentException $e)
428  {
429  dol_syslog($e->getMessage(), LOG_INFO);
430  }
431  }
432  $listlines->merge();
433  }
434  }
435  $odfHandler->mergeSegment($listlines);
436  } catch (OdfException $e)
437  {
438  $this->error = $e->getMessage();
439  dol_syslog($this->error, LOG_WARNING);
440  return -1;
441  }
442 
443  // Replace labels translated
444  $tmparray = $outputlangs->get_translations_for_substitutions();
445  foreach ($tmparray as $key=>$value)
446  {
447  try {
448  $odfHandler->setVars($key, $value, true, 'UTF-8');
449  } catch (OdfException $e)
450  {
451  dol_syslog($e->getMessage(), LOG_INFO);
452  }
453  }
454 
455  // Call the beforeODTSave hook
456  $parameters = array('odfHandler'=>&$odfHandler, 'file'=>$file, 'object'=>$object, 'outputlangs'=>$outputlangs);
457  $reshook = $hookmanager->executeHooks('beforeODTSave', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
458 
459  // Write new file
460  if (!empty($conf->global->MAIN_ODT_AS_PDF)) {
461  try {
462  $odfHandler->exportAsAttachedPDF($file);
463  } catch (Exception $e) {
464  $this->error = $e->getMessage();
465  dol_syslog($e->getMessage(), LOG_INFO);
466  return -1;
467  }
468  } else {
469  try {
470  $odfHandler->saveToDisk($file);
471  } catch (Exception $e) {
472  $this->error = $e->getMessage();
473  dol_syslog($e->getMessage(), LOG_INFO);
474  return -1;
475  }
476  }
477 
478  $reshook = $hookmanager->executeHooks('afterODTCreation', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
479 
480  if (!empty($conf->global->MAIN_UMASK))
481  @chmod($file, octdec($conf->global->MAIN_UMASK));
482 
483  $odfHandler = null; // Destroy object
484 
485  $this->result = array('fullpath'=>$file);
486 
487  return 1; // Success
488  } else {
489  $this->error = $langs->transnoentities("ErrorCanNotCreateDir", $dir);
490  return -1;
491  }
492  }
493 
494  return -1;
495  }
496 }
info($langs)
Return description of a module.
Class to build documents using ODF templates generator.
</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.
$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.
getCommonSubstitutionArray($outputlangs, $onlykey=0, $exclude=null, $object=null)
Return array of possible common substitutions.
Class to manage hooks.
get_substitutionarray_contact($object, $outputlangs, $array_key= 'object')
Define array with couple substitution key =&gt; substitution value.
Class to manage generation of HTML components Only common components must be here.
get_substitutionarray_thirdparty($object, $outputlangs, $array_key= 'company')
Define array with couple substitution key =&gt; substitution value.
img_picto($titlealt, $picto, $moreatt= '', $pictoisfullpath=false, $srconly=0, $notitle=0, $alt= '', $morecss= '', $marginleftonlyshort=2)
Show picto whatever it&#39;s its name (generic function)
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
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...
write_file($object, $outputlangs, $srctemplatepath, $hidedetails=0, $hidedesc=0, $hideref=0)
Function to build a document on disk using the generic odt module.
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.
Parent class for stock models of doc generators.
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition: repair.php:105
Class to manage predefined suppliers products.
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...
Class to manage warehouses.