dolibarr 24.0.0-beta
fournisseur.facture-rec.class.php
1<?php
2/* Copyright (C) 2003-2005 Rodolphe Quiedeville <rodolphe@quiedeville.org>
3 * Copyright (C) 2004-2019 Laurent Destailleur <eldy@users.sourceforge.net>
4 * Copyright (C) 2009-2012 Regis Houssin <regis.houssin@inodbox.com>
5 * Copyright (C) 2010-2011 Juanjo Menent <jmenent@2byte.es>
6 * Copyright (C) 2012 Cedric Salvador <csalvador@gpcsolutions.fr>
7 * Copyright (C) 2013 Florian Henry <florian.henry@open-concept.pro>
8 * Copyright (C) 2015 Marcos García <marcosgdf@gmail.com>
9 * Copyright (C) 2017-2025 Frédéric France <frederic.france@free.fr>
10 * Copyright (C) 2024-2026 MDW <mdeweerd@users.noreply.github.com>
11 * Copyright (C) 2023-2024 Nick Fragoulis
12 *
13 * This program is free software; you can redistribute it and/or modify
14 * it under the terms of the GNU General Public License as published by
15 * the Free Software Foundation; either version 3 of the License, or
16 * (at your option) any later version.
17 *
18 * This program is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU General Public License for more details.
22 *
23 * You should have received a copy of the GNU General Public License
24 * along with this program. If not, see <https://www.gnu.org/licenses/>.
25 */
26
33require_once DOL_DOCUMENT_ROOT.'/core/class/notify.class.php';
34require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
35require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture.class.php';
36require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.facture-rec.ligne.class.php';
37require_once DOL_DOCUMENT_ROOT.'/core/lib/date.lib.php';
38
39
44{
49 public $TRIGGER_PREFIX = 'SUPPLIERBILLREC';
50
54 public $element = 'invoice_supplier_rec';
55
59 public $table_element = 'facture_fourn_rec';
60
64 public $table_element_line = 'facture_fourn_det_rec';
65
69 public $fk_element = 'fk_facture_fourn';
70
74 public $picto = 'supplier_invoice';
75
79 protected $table_ref_field = 'titre';
80
85 public $titre;
89 public $title;
90
94 public $ref_supplier;
98 public $socid;
99
104 public $fk_soc;
105
109 public $suspended; // status
110
115 public $libelle;
119 public $label;
120
125 public $amount;
130 public $remise;
131
135 public $vat_src_code;
139 public $localtax1;
143 public $localtax2;
144
148 public $user_author;
152 public $user_modif;
156 public $fk_project;
157
161 public $mode_reglement_id;
165 public $mode_reglement_code;
169 public $cond_reglement_code;
173 public $cond_reglement_doc;
177 public $cond_reglement_id;
178
182 public $date_lim_reglement;
183
187 public $usenewprice = 0;
191 public $frequency;
195 public $unit_frequency;
199 public $date_when;
203 public $date_last_gen;
204
208 public $nb_gen_done;
209
213 public $nb_gen_max;
214
218 public $auto_validate; //
222 public $generate_pdf; // 1 to generate PDF on invoice generation (default)
223
228 public $lines = array();
229
230
231 /* Override fields in CommonObject
232 public $total_ht;
233 public $total_tva;
234 public $total_ttc;
235 public $fk_account;
236 public $mode_reglement;
237 public $cond_reglement;
238 public $note_public;
239 public $note_private;
240 */
241
266 // BEGIN MODULEBUILDER PROPERTIES
270 public $fields = array(
271 'rowid' => array('type' => 'integer', 'label' => 'TechnicalID', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 10),
272 'titre' => array('type' => 'varchar(100)', 'label' => 'Titre', 'enabled' => 1, 'showoncombobox' => 1, 'visible' => -1, 'position' => 15),
273 'ref_supplier' => array('type' => 'varchar(180)', 'label' => 'RefSupplier', 'enabled' => 1, 'showoncombobox' => 1, 'visible' => -1, 'position' => 20),
274 'entity' => array('type' => 'integer', 'label' => 'Entity', 'default' => '1', 'enabled' => 1, 'visible' => -2, 'notnull' => 1, 'position' => 25, 'index' => 1),
275 'fk_soc' => array('type' => 'integer:Societe:societe/class/societe.class.php', 'label' => 'ThirdParty', 'enabled' => 'isModEnabled("societe")', 'visible' => -1, 'notnull' => 1, 'position' => 30),
276 'datec' => array('type' => 'datetime', 'label' => 'DateCreation', 'enabled' => 1, 'visible' => -1, 'position' => 35),
277 'tms' => array('type' => 'timestamp', 'label' => 'DateModification', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 40),
278 'suspended' => array('type' => 'integer', 'label' => 'Suspended', 'enabled' => 1, 'visible' => -1, 'position' => 225),
279 'libelle' => array('type' => 'varchar(100)', 'label' => 'Libelle', 'enabled' => 1, 'showoncombobox' => 0, 'visible' => -1, 'position' => 15),
280
281 'localtax1' => array('type' => 'double(24,8)', 'label' => 'Localtax1', 'enabled' => 1, 'visible' => -1, 'position' => 60, 'isameasure' => 1),
282 'localtax2' => array('type' => 'double(24,8)', 'label' => 'Localtax2', 'enabled' => 1, 'visible' => -1, 'position' => 65, 'isameasure' => 1),
283 'total_ht' => array('type' => 'double(24,8)', 'label' => 'Total', 'enabled' => 1, 'visible' => -1, 'position' => 70, 'isameasure' => 1),
284 'total_tva' => array('type' => 'double(24,8)', 'label' => 'Tva', 'enabled' => 1, 'visible' => -1, 'position' => 55, 'isameasure' => 1),
285 'total_ttc' => array('type' => 'double(24,8)', 'label' => 'Total ttc', 'enabled' => 1, 'visible' => -1, 'position' => 75, 'isameasure' => 1),
286
287 'fk_user_author' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserAuthor', 'enabled' => 1, 'visible' => -1, 'position' => 80),
288 'fk_user_modif' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserModif', 'enabled' => 1, 'visible' => -2, 'notnull' => -1, 'position' => 210),
289 'fk_projet' => array('type' => 'integer:Project:projet/class/project.class.php:1:fk_statut=1', 'label' => 'Fk projet', 'enabled' => "isModEnabled('project')", 'visible' => -1, 'position' => 85),
290 'fk_account' => array('type' => 'integer', 'label' => 'Fk account', 'enabled' => 'isModEnabled("bank")', 'visible' => -1, 'position' => 175),
291 'fk_cond_reglement' => array('type' => 'integer', 'label' => 'Fk cond reglement', 'enabled' => 1, 'visible' => -1, 'position' => 90),
292 'fk_mode_reglement' => array('type' => 'integer', 'label' => 'Fk mode reglement', 'enabled' => 1, 'visible' => -1, 'position' => 95),
293 'date_lim_reglement' => array('type' => 'date', 'label' => 'Date lim reglement', 'enabled' => 1, 'visible' => -1, 'position' => 100),
294
295 'note_private' => array('type' => 'html', 'label' => 'NotePublic', 'enabled' => 1, 'visible' => 0, 'position' => 105),
296 'note_public' => array('type' => 'html', 'label' => 'NotePrivate', 'enabled' => 1, 'visible' => 0, 'position' => 110),
297 'modelpdf' => array('type' => 'varchar(255)', 'label' => 'Modelpdf', 'enabled' => 1, 'visible' => -1, 'position' => 115),
298
299 'fk_multicurrency' => array('type' => 'integer', 'label' => 'Fk multicurrency', 'enabled' => 1, 'visible' => -1, 'position' => 180),
300 'multicurrency_code' => array('type' => 'varchar(255)', 'label' => 'Multicurrency code', 'enabled' => 1, 'visible' => -1, 'position' => 185),
301 'multicurrency_tx' => array('type' => 'double(24,8)', 'label' => 'Multicurrency tx', 'enabled' => 1, 'visible' => -1, 'position' => 190, 'isameasure' => 1),
302 'multicurrency_total_ht' => array('type' => 'double(24,8)', 'label' => 'Multicurrency total ht', 'enabled' => 1, 'visible' => -1, 'position' => 195, 'isameasure' => 1),
303 'multicurrency_total_tva' => array('type' => 'double(24,8)', 'label' => 'Multicurrency total tva', 'enabled' => 1, 'visible' => -1, 'position' => 200, 'isameasure' => 1),
304 'multicurrency_total_ttc' => array('type' => 'double(24,8)', 'label' => 'Multicurrency total ttc', 'enabled' => 1, 'visible' => -1, 'position' => 205, 'isameasure' => 1),
305
306 'usenewprice' => array('type' => 'integer', 'label' => 'UseNewPrice', 'enabled' => 1, 'visible' => 0, 'position' => 155),
307 'frequency' => array('type' => 'integer', 'label' => 'Frequency', 'enabled' => 1, 'visible' => -1, 'position' => 150),
308 'unit_frequency' => array('type' => 'varchar(2)', 'label' => 'Unit frequency', 'enabled' => 1, 'visible' => -1, 'position' => 125),
309
310 'date_when' => array('type' => 'datetime', 'label' => 'Date when', 'enabled' => 1, 'visible' => -1, 'position' => 130),
311 'date_last_gen' => array('type' => 'datetime', 'label' => 'Date last gen', 'enabled' => 1, 'visible' => -1, 'position' => 135),
312 'nb_gen_done' => array('type' => 'integer', 'label' => 'Nb gen done', 'enabled' => 1, 'visible' => -1, 'position' => 140),
313 'nb_gen_max' => array('type' => 'integer', 'label' => 'Nb gen max', 'enabled' => 1, 'visible' => -1, 'position' => 145),
314 'revenuestamp' => array('type' => 'double(24,8)', 'label' => 'RevenueStamp', 'enabled' => 1, 'visible' => -1, 'position' => 160, 'isameasure' => 1),
315 'auto_validate' => array('type' => 'integer', 'label' => 'Auto validate', 'enabled' => 1, 'visible' => -1, 'position' => 165),
316 'generate_pdf' => array('type' => 'integer', 'label' => 'Generate pdf', 'enabled' => 1, 'visible' => -1, 'position' => 170),
317 );
318 // END MODULEBUILDER PROPERTIES
319
320 const STATUS_NOTSUSPENDED = 0;
321 const STATUS_SUSPENDED = 1;
322
323
324
330 public function __construct($db)
331 {
332 $this->db = $db;
333 }
334
344 public function create($user, $facFournId, $notrigger = 0, $onlylines = array())
345 {
346 global $conf;
347
348 $error = 0;
349 $now = dol_now();
350
351 // Clean parameters
352 $this->titre = empty($this->titre) ? '' : $this->titre; // deprecated
353 $this->title = empty($this->title) ? '' : $this->title;
354 $keyforref = $this->table_ref_field;
355 $this->ref = $this->$keyforref;
356
357 $this->ref_supplier = empty($this->ref_supplier) ? '' : $this->ref_supplier;
358
359 $this->usenewprice = empty($this->usenewprice) ? 0 : $this->usenewprice;
360 $this->suspended = empty($this->suspended) ? 0 : $this->suspended;
361 // No frequency defined then no next date to execution
362 if (empty($this->frequency)) {
363 $this->frequency = 0;
364 $this->date_when = null;
365 }
366 $this->frequency = abs($this->frequency);
367 $this->nb_gen_done = 0;
368 $this->nb_gen_max = empty($this->nb_gen_max) ? 0 : $this->nb_gen_max;
369 $this->auto_validate = empty($this->auto_validate) ? 0 : $this->auto_validate;
370 $this->generate_pdf = empty($this->generate_pdf) ? 0 : $this->generate_pdf;
371
372 $this->db->begin();
373
374 // We load the supplier invoice from which we create the model/template supplier invoice
375 $facfourn_src = new FactureFournisseur($this->db);
376 $result = $facfourn_src->fetch($facFournId);
377 if ($result > 0) {
378 $sql = 'INSERT INTO '.MAIN_DB_PREFIX.'facture_fourn_rec (';
379 $sql .= 'titre';
380 $sql .= ", subtype";
381 $sql .= ', ref_supplier';
382 $sql .= ', entity';
383 $sql .= ', fk_soc';
384 $sql .= ', datec';
385 $sql .= ', suspended';
386 $sql .= ', libelle';
387 $sql .= ', total_ttc';
388 $sql .= ', fk_user_author';
389 $sql .= ', fk_projet';
390 $sql .= ', fk_account';
391 $sql .= ', fk_cond_reglement';
392 $sql .= ', fk_mode_reglement';
393 $sql .= ', date_lim_reglement';
394 $sql .= ', note_private';
395 $sql .= ', note_public';
396 $sql .= ', modelpdf';
397 $sql .= ', fk_multicurrency';
398 $sql .= ', multicurrency_code';
399 $sql .= ', multicurrency_tx';
400 $sql .= ', usenewprice';
401 $sql .= ', frequency';
402 $sql .= ', unit_frequency';
403 $sql .= ', date_when';
404 $sql .= ', date_last_gen';
405 $sql .= ', nb_gen_done';
406 $sql .= ', nb_gen_max';
407 $sql .= ', auto_validate';
408 $sql .= ', generate_pdf';
409 $sql .= ') VALUES (';
410 $sql .= "'".$this->db->escape($this->title)."'";
411 $sql .= ", ".(isset($this->subtype) ? (int) $this->subtype : "NULL");
412 $sql .= ", '".$this->db->escape($this->ref_supplier)."'";
413 $sql .= ", ".((int) $conf->entity);
414 $sql .= ", ".((int) $facfourn_src->socid);
415 $sql .= ", '".$this->db->idate($now)."'";
416 $sql .= ", ".((int) $this->suspended);
417 $sql .= ", '".$this->db->escape($this->libelle)."'";
418 $sql .= ", " .(!empty($facfourn_src->total_ttc) ? (float) $facfourn_src->total_ttc : '0'); // amount
419 $sql .= ", " .((int) $user->id);
420 $sql .= ", " .(!empty($this->fk_project) ? ((int) $this->fk_project) : 'NULL');
421 $sql .= ", " .(!empty($facfourn_src->fk_account) ? ((int) $facfourn_src->fk_account) : 'NULL');
422 $sql .= ", " .($this->cond_reglement_id > 0 ? (int) $this->cond_reglement_id : 'NULL');
423 $sql .= ", " .($this->mode_reglement_id > 0 ? (int) $this->mode_reglement_id : 'NULL');
424 $sql .= ", ".($facfourn_src->date_echeance > 0 ? "'".$this->db->idate($facfourn_src->date_echeance)."'" : 'NULL'); // date_lim_reglement
425 $sql .= ", " .(!empty($this->note_private) ? "'".$this->db->escape($this->note_private)."'" : 'NULL');
426 $sql .= ", " .(!empty($this->note_public) ? "'".$this->db->escape($this->note_public)."'" : 'NULL');
427 $sql .= ", " .(!empty($this->model_pdf) ? "'".$this->db->escape($this->model_pdf)."'" : 'NULL');
428 $sql .= ", " . (int) $facfourn_src->fk_multicurrency;
429 $sql .= ", '".$this->db->escape($facfourn_src->multicurrency_code)."'";
430 $sql .= ", " . (float) $facfourn_src->multicurrency_tx;
431 $sql .= ", " . (int) $this->usenewprice;
432 $sql .= ", " . (int) $this->frequency;
433 $sql .= ", '".$this->db->escape($this->unit_frequency)."'";
434 $sql .= ", " .(!empty($this->date_when) ? "'".$this->db->idate($this->date_when)."'" : 'NULL');
435 $sql .= ", " .(!empty($this->date_last_gen) ? "'".$this->db->idate($this->date_last_gen)."'" : 'NULL');
436 $sql .= ", " . (int) $this->nb_gen_done;
437 $sql .= ", " . (int) $this->nb_gen_max;
438 $sql .= ", " . (int) $this->auto_validate;
439 $sql .= ", " . (int) $this->generate_pdf;
440 $sql .= ')';
441
442 if ($this->db->query($sql)) {
443 $this->id = $this->db->last_insert_id(MAIN_DB_PREFIX. 'facture_fourn_rec');
444
445 // Fields used into addline later
446 $this->fk_multicurrency = $facfourn_src->fk_multicurrency;
447
448 $this->multicurrency_code = $facfourn_src->multicurrency_code;
449 $this->multicurrency_tx = $facfourn_src->multicurrency_tx;
450
451 // Add lines
452 $num = count($facfourn_src->lines);
453 for ($i = 0; $i < $num; $i++) {
454 $facfourn_line = $facfourn_src->lines[$i];
455 '@phan-var-force SupplierInvoiceLine $facfourn_line';
456 if (!empty($onlylines) && !in_array($facfourn_line->id, $onlylines)) {
457 continue; // Skip unselected lines
458 }
459
460 $tva_tx = $facfourn_line->tva_tx;
461 if (!empty($facfourn_line->vat_src_code) && !preg_match('/\‍(/', (string) $tva_tx)) {
462 $tva_tx .= ' ('.$facfourn_line->vat_src_code.')';
463 }
464
465 $result_insert = $this->addline(
466 $facfourn_line->fk_product,
467 $facfourn_line->ref_supplier,
468 $facfourn_line->product_label,
469 $facfourn_line->desc ? $facfourn_line->desc : $facfourn_line->description,
470 $facfourn_line->pu_ht,
471 $facfourn_line->pu_ttc,
472 $facfourn_line->qty,
473 $facfourn_line->remise_percent,
474 $tva_tx,
475 $facfourn_line->localtax1_tx,
476 $facfourn_line->localtax2_tx,
477 'HT',
478 $facfourn_line->product_type,
479 $facfourn_line->date_start,
480 $facfourn_line->date_end,
481 $facfourn_line->info_bits,
482 $facfourn_line->special_code,
483 $facfourn_line->rang,
484 $facfourn_line->fk_unit
485 );
486
487 if ($result_insert < 0) {
488 $error++;
489 } else {
490 $objectline = new FactureFournisseurLigneRec($this->db);
491
492 $result2 = $objectline->fetch($result_insert);
493 if ($result2 > 0) {
494 // Extrafields
495 if (method_exists($facfourn_line, 'fetch_optionals')) {
496 $facfourn_line->fetch_optionals($facfourn_line->id);
497 $objectline->array_options = $facfourn_line->array_options;
498 }
499
500 $result = $objectline->insertExtraFields();
501 if ($result < 0) {
502 $error++;
503 }
504 } elseif ($result2 < 0) {
505 $this->errors[] = $objectline->error;
506 $error++;
507 }
508 }
509 }
510
511 if (!empty($this->linkedObjectsIds) && empty($this->linked_objects)) { // To use new linkedObjectsIds instead of old linked_objects
512 $this->linked_objects = $this->linkedObjectsIds; // TODO Replace linked_objects with linkedObjectsIds
513 }
514
515 // Add object linked
516 if (!$error && $this->id && !empty($this->linked_objects) && is_array($this->linked_objects)) {
517 foreach ($this->linked_objects as $origin => $tmp_origin_id) {
518 if (is_array($tmp_origin_id)) { // New behaviour, if linked_object can have several links per type, so is something like array('contract'=>array(id1, id2, ...))
519 foreach ($tmp_origin_id as $origin_id) {
520 $ret = $this->add_object_linked($origin, $origin_id);
521 if (!$ret) {
522 $this->error = $this->db->lasterror();
523 $error++;
524 }
525 }
526 } else { // Old behaviour, if linked_object has only one link per type, so is something like array('contract'=>id1))
527 $origin_id = $tmp_origin_id;
528 $ret = $this->add_object_linked($origin, $origin_id);
529 if (!$ret) {
530 $this->error = $this->db->lasterror();
531 $error++;
532 }
533 }
534 }
535 }
536
537 if (!$error) {
538 $result = $this->insertExtraFields();
539 if ($result < 0) {
540 $error++;
541 }
542 }
543
544 if (!$error && !$notrigger) {
545 // Call trigger
546 $result = $this->call_trigger('SUPPLIERBILLREC_CREATE', $user);
547 if ($result < 0) {
548 $this->db->rollback();
549 return -2;
550 }
551 // End call triggers
552 }
553
554 if ($error) {
555 $this->db->rollback();
556 return -3;
557 } else {
558 $this->db->commit();
559 return $this->id;
560 }
561 } else {
562 $this->error = $this->db->lasterror();
563 $this->db->rollback();
564 return -2;
565 }
566 } else {
567 $this->db->rollback();
568 return -1;
569 }
570 }
571
572
580 public function update(User $user, $notrigger = 0)
581 {
582 $error = 0;
583
584 $sql = "UPDATE ".MAIN_DB_PREFIX."facture_fourn_rec SET";
585 $sql .= " titre = '" . (!empty($this->title) ? $this->db->escape($this->title) : "")."'," ;
586 $sql .= " subtype=".(isset($this->subtype) ? (int) $this->subtype : "NULL").",";
587 $sql .= " ref_supplier = '". (!empty($this->ref_supplier) ? $this->db->escape($this->ref_supplier) : "")."',";
588 $sql .= " entity = ". (!empty($this->entity) ? ((int) $this->entity) : 1) . ',';
589 if (!empty($this->socid) && $this->socid > 0) {
590 $sql .= " fk_soc = ". ((int) $this->socid). ',';
591 } elseif (!empty($this->fk_soc) && $this->fk_soc > 0) { // For backward compatibility
592 $sql .= " fk_soc = ". ((int) $this->fk_soc). ',';
593 }
594 $sql .= " suspended = ". (!empty($this->suspended) ? ((int) $this->suspended) : 0) . ',';
595 $sql .= " libelle = ". (!empty($this->libelle) ? "'".$this->db->escape($this->libelle)."'" : 'NULL') . ",";
596 $sql .= " vat_src_code = ". (!empty($this->vat_src_code) ? "'".$this->db->escape($this->vat_src_code)."'" : 'NULL') . ',';
597 $sql .= " localtax1 = ". (!empty($this->localtax1) ? ((float) $this->localtax1) : 0.00) . ',';
598 $sql .= " localtax2 = ". (!empty($this->localtax2) ? ((float) $this->localtax2) : 0.00) . ',';
599 $sql .= " total_ht = ". (!empty($this->total_ht) ? ((float) $this->total_ht) : 0.00) . ',';
600 $sql .= " total_tva = ". (!empty($this->total_tva) ? ((float) $this->total_tva) : 0.00) . ',';
601 $sql .= " total_ttc = ". (!empty($this->total_ttc) ? ((float) $this->total_ttc) : 0.00) . ',';
602 $sql .= " fk_user_modif = ". ((int) $user->id) . ',';
603 $sql .= " fk_projet = ". (!empty($this->fk_project) ? ((int) $this->fk_project) : 'NULL') . ',';
604 $sql .= " fk_account = ". (!empty($this->fk_account) ? ((int) $this->fk_account) : 'NULL') . ',';
605 $sql .= " fk_mode_reglement = ". (!empty($this->mode_reglement_id) ? ((int) $this->mode_reglement_id) : 'NULL') . ',';
606 $sql .= " fk_cond_reglement = ". (!empty($this->cond_reglement_id) ? ((int) $this->cond_reglement_id) : 'NULL') . ',';
607 $sql .= " date_lim_reglement = ". (!empty($this->date_lim_reglement) ? "'".$this->db->idate($this->date_lim_reglement)."'" : 'NULL') . ',';
608 $sql .= " note_private = '". (!empty($this->note_private) ? $this->db->escape($this->note_private) : '') . "',";
609 $sql .= " note_public = '". (!empty($this->note_public) ? $this->db->escape($this->note_public) : '') . "',";
610 $sql .= " modelpdf = ". (!empty($this->model_pdf) ? "'".$this->db->escape($this->model_pdf)."'" : 'NULL') . ",";
611 $sql .= " fk_multicurrency = ". (!empty($this->fk_multicurrency) ? ((int) $this->fk_multicurrency) : 'NULL') . ',';
612 $sql .= " multicurrency_code = ". (!empty($this->multicurrency_code) ? "'".$this->db->escape($this->multicurrency_code)."'" : 'NULL') . ",";
613 $sql .= " multicurrency_tx = ". (!empty($this->multicurrency_tx) ? ((float) $this->multicurrency_tx) : 1) . ',';
614 $sql .= " multicurrency_total_ht = ". (!empty($this->multicurrency_total_ht) ? ((float) $this->multicurrency_total_ht) : 0.00) . ',';
615 $sql .= " multicurrency_total_tva = ". (!empty($this->multicurrency_total_tva) ? ((float) $this->multicurrency_total_tva) : 0.00) . ',';
616 $sql .= " multicurrency_total_ttc = ". (!empty($this->multicurrency_total_ttc) ? ((float) $this->multicurrency_total_ttc) : 0.00) . ',';
617 $sql .= " usenewprice = ". (!empty($this->usenewprice) ? ((int) $this->usenewprice) : 0) . ',';
618 $sql .= " frequency = ". (!empty($this->frequency) ? ((int) $this->frequency) : 0). ',';
619 $sql .= " unit_frequency = '". (!empty($this->unit_frequency) ? $this->db->escape($this->unit_frequency) : ''). "',";
620 $sql .= " date_when = ". (!empty($this->date_when) ? "'".$this->db->idate($this->date_when)."'" : 'NULL') . ',';
621 $sql .= " date_last_gen = ". (!empty($this->date_last_gen) ? "'".$this->db->idate($this->date_last_gen)."'" : 'NULL') . ',';
622 $sql .= " nb_gen_done = ". (!empty($this->nb_gen_done) ? ((int) $this->nb_gen_done) : 0) . ',';
623 $sql .= " nb_gen_max = ". (!empty($this->nb_gen_max) ? ((int) $this->nb_gen_max) : 0) . ',';
624 $sql .= " auto_validate = ". (!empty($this->auto_validate) ? ((int) $this->auto_validate) : 0);
625 $sql .= " WHERE rowid = ". (int) $this->id;
626
627 $this->db->begin();
628
629 dol_syslog(get_class($this)."::update", LOG_DEBUG);
630 $resql = $this->db->query($sql);
631 if ($resql) {
632 if (!$error) {
633 $result = $this->insertExtraFields();
634 if ($result < 0) {
635 $error++;
636 }
637 }
638
639 if (!$error && !$notrigger) {
640 // Call trigger
641 $result = $this->call_trigger('SUPPLIERBILLREC_MODIFY', $user);
642 if ($result < 0) {
643 $this->db->rollback();
644 return -2;
645 }
646 // End call triggers
647 }
648 $this->db->commit();
649 return 1;
650 } else {
651 $this->error = $this->db->lasterror();
652 $this->db->rollback();
653 return -2;
654 }
655 }
656
665 public function fetch($rowid, $ref = '', $ref_ext = '')
666 {
667 $sql = 'SELECT f.rowid, f.titre as title, f.subtype, f.ref_supplier, f.entity, f.fk_soc';
668 $sql .= ', f.datec, f.tms, f.suspended';
669 $sql .= ', f.libelle as label';
670 $sql .= ', f.vat_src_code, f.localtax1, f.localtax2';
671 $sql .= ', f.total_tva, f.total_ht, f.total_ttc';
672 $sql .= ', f.fk_user_author, f.fk_user_modif';
673 $sql .= ', f.fk_projet as fk_project, f.fk_account';
674 $sql .= ', f.fk_mode_reglement, p.code as mode_reglement_code, p.libelle as mode_reglement_libelle';
675 $sql .= ', f.fk_cond_reglement, c.code as cond_reglement_code, c.libelle as cond_reglement_libelle, c.libelle_facture as cond_reglement_libelle_doc';
676 $sql .= ', f.date_lim_reglement';
677 $sql .= ', f.note_private, f.note_public, f.modelpdf as model_pdf';
678 $sql .= ', f.fk_multicurrency, f.multicurrency_code, f.multicurrency_tx, f.multicurrency_total_ht, f.multicurrency_total_tva, f.multicurrency_total_ttc';
679 $sql .= ', f.usenewprice, f.frequency, f.unit_frequency, f.date_when, f.date_last_gen, f.nb_gen_done, f.nb_gen_max, f.auto_validate';
680 $sql .= ', f.generate_pdf';
681 $sql .= ' FROM '.MAIN_DB_PREFIX.'facture_fourn_rec as f';
682 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_payment_term as c ON f.fk_cond_reglement = c.rowid';
683 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_paiement as p ON f.fk_mode_reglement = p.id';
684 $sql .= ' WHERE f.entity IN ('.getEntity('invoice').')';
685 if ($rowid) {
686 $sql .= ' AND f.rowid='. (int) $rowid;
687 } elseif ($ref) {
688 $sql .= " AND f.titre='".$this->db->escape($ref)."'";
689 } else {
690 $sql .= ' AND f.rowid = 0';
691 }
692
693 $result = $this->db->query($sql);
694 if ($result) {
695 if ($this->db->num_rows($result)) {
696 $obj = $this->db->fetch_object($result);
697
698 $keyforref = $this->table_ref_field;
699
700 $this->id = $obj->rowid;
701 $this->titre = $obj->title;
702 $this->title = $obj->title;
703 $this->subtype = $obj->subtype;
704 $this->ref = $obj->title;
705 $this->ref_supplier = $obj->ref_supplier;
706 $this->entity = $obj->entity;
707 $this->socid = $obj->fk_soc;
708 $this->date_creation = $obj->datec;
709 $this->date_modification = $obj->tms;
710 $this->status = $obj->suspended;
711 $this->suspended = $obj->suspended;
712 $this->libelle = $obj->label;
713 $this->label = $obj->label;
714 $this->vat_src_code = $obj->vat_src_code;
715 $this->total_localtax1 = $obj->localtax1;
716 $this->total_localtax2 = $obj->localtax2;
717 $this->total_ht = $obj->total_ht;
718 $this->total_tva = $obj->total_tva;
719 $this->total_ttc = $obj->total_ttc;
720 $this->user_creation_id = $obj->fk_user_author;
721 $this->user_modif = $obj->fk_user_modif;
722 $this->fk_project = $obj->fk_project;
723 $this->fk_account = $obj->fk_account;
724 $this->mode_reglement_id = $obj->fk_mode_reglement;
725 $this->mode_reglement_code = $obj->mode_reglement_code;
726 $this->mode_reglement = $obj->mode_reglement_libelle;
727 $this->cond_reglement_id = $obj->fk_cond_reglement;
728 $this->cond_reglement_code = $obj->cond_reglement_code;
729 $this->cond_reglement = $obj->cond_reglement_libelle;
730 $this->cond_reglement_doc = $obj->cond_reglement_libelle_doc;
731 $this->date_lim_reglement = $this->db->jdate($obj->date_lim_reglement);
732 $this->note_private = $obj->note_private;
733 $this->note_public = $obj->note_public;
734 $this->model_pdf = $obj->model_pdf;
735
736 // Multicurrency
737 $this->fk_multicurrency = $obj->fk_multicurrency;
738 $this->multicurrency_code = $obj->multicurrency_code;
739 $this->multicurrency_tx = $obj->multicurrency_tx;
740 $this->multicurrency_total_ht = $obj->multicurrency_total_ht;
741 $this->multicurrency_total_tva = $obj->multicurrency_total_tva;
742 $this->multicurrency_total_ttc = $obj->multicurrency_total_ttc;
743
744 $this->usenewprice = $obj->usenewprice;
745 $this->frequency = $obj->frequency;
746 $this->unit_frequency = $obj->unit_frequency;
747 $this->date_when = $this->db->jdate($obj->date_when);
748 $this->date_last_gen = $this->db->jdate($obj->date_last_gen);
749 $this->nb_gen_done = $obj->nb_gen_done;
750 $this->nb_gen_max = $obj->nb_gen_max;
751 $this->auto_validate = $obj->auto_validate;
752 $this->generate_pdf = $obj->generate_pdf;
753
754 // Retrieve all extrafield
755 // fetch optionals attributes and labels
756 $this->fetch_optionals();
757
758 /*
759 * Lines
760 */
761 $result = $this->fetch_lines();
762 if ($result < 0) {
763 $this->error = $this->db->lasterror();
764 return -3;
765 }
766 return 1;
767 } else {
768 $this->error = 'Bill with id '.$rowid.' or ref '.$ref.' not found';
769 dol_syslog('Facture::Fetch Error '.$this->error, LOG_ERR);
770 return -2;
771 }
772 } else {
773 $this->error = $this->db->error();
774 return -1;
775 }
776 }
777
778
784 public function getLinesArray()
785 {
786 return $this->fetch_lines();
787 }
788
789 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
795 public function fetch_lines()
796 {
797 // phpcs:enable
798 $this->lines = array();
799
800 // Retrieve all extrafield for line
801 // fetch optionals attributes and labels
802 /*if (!is_object($extrafields)) {
803 require_once DOL_DOCUMENT_ROOT.'/core/class/extrafields.class.php';
804 $extrafields = new ExtraFields($this->db);
805 }
806 $extrafields->fetch_name_optionals_label($this->table_element_line, true);
807 */
808
809 $sql = 'SELECT l.rowid,';
810 $sql .= ' l.fk_facture_fourn, l.fk_parent_line, l.fk_product, l.ref as ref_supplier, l.label, l.description as line_desc,';
811 $sql .= ' l.pu_ht, l.pu_ttc, l.qty, l.remise_percent, l.fk_remise_except, l.vat_src_code, l.tva_tx,';
812 $sql .= ' l.localtax1_tx, l.localtax2_tx, l.localtax1_type, l.localtax2_type,';
813 $sql .= ' l.total_ht, l.total_tva, l.total_ttc, total_localtax1, total_localtax2,';
814 $sql .= ' l.product_type, l.date_start, l.date_end,';
815 $sql .= ' l.info_bits, l.special_code, l.rang,';
816 $sql .= ' l.fk_unit, l.import_key, l.fk_user_author, l.fk_user_modif, l.extraparams,';
817 $sql .= ' l.fk_multicurrency, l.multicurrency_code, l.multicurrency_subprice, l.multicurrency_total_ht, l.multicurrency_total_tva, l.multicurrency_total_ttc,';
818 $sql .= ' p.ref as product_ref, p.fk_product_type as fk_product_type, p.label as product_label, p.description as product_desc';
819 $sql .= ' FROM '.MAIN_DB_PREFIX.'facture_fourn_det_rec as l';
820 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product as p ON l.fk_product = p.rowid';
821 $sql .= ' WHERE l.fk_facture_fourn = '. (int) $this->id;
822 $sql .= ' ORDER BY l.rang';
823
824 dol_syslog('FactureFournisseurRec::fetch_lines', LOG_DEBUG);
825
826 $result = $this->db->query($sql);
827 if ($result) {
828 $num = $this->db->num_rows($result);
829 $i = 0;
830 while ($i < $num) {
831 $objp = $this->db->fetch_object($result);
832
833 $line = new FactureFournisseurLigneRec($this->db);
834
835 $line->id = $objp->rowid;
836 $line->fk_facture_fourn = $objp->fk_facture_fourn;
837 $line->fk_parent = $objp->fk_parent_line;
838 $line->fk_product = $objp->fk_product;
839 $line->ref = $objp->product_ref; // Ref of product
840 $line->product_ref = $objp->product_ref; // Ref of product
841 $line->product_label = $objp->product_label;
842 $line->product_desc = $objp->product_desc;
843 $line->ref_supplier = $objp->ref_supplier;
844 $line->label = $objp->label;
845 $line->description = $objp->line_desc;
846 $line->desc = $objp->line_desc;
847 $line->pu_ht = $objp->pu_ht; // deprecated
848 $line->subprice = $objp->pu_ht;
849 $line->pu_ttc = $objp->pu_ttc; // deprecated
850 $line->subprice_ttc = $objp->pu_ttc;
851 $line->qty = $objp->qty;
852 $line->remise_percent = $objp->remise_percent;
853 $line->fk_remise_except = $objp->fk_remise_except;
854 $line->vat_src_code = $objp->vat_src_code;
855 $line->tva_tx = $objp->tva_tx;
856 $line->localtax1_tx = $objp->localtax1_tx;
857 $line->localtax1_type = $objp->localtax1_type;
858 $line->localtax2_tx = $objp->localtax2_tx;
859 $line->localtax2_type = $objp->localtax2_type;
860 $line->total_ht = $objp->total_ht;
861 $line->total_tva = $objp->total_tva;
862 $line->total_localtax1 = $objp->total_localtax1;
863 $line->total_localtax2 = $objp->total_localtax2;
864 $line->total_ttc = $objp->total_ttc;
865 $line->product_type = $objp->product_type;
866 $line->date_start = $this->db->jdate($objp->date_start);
867 $line->date_end = $this->db->jdate($objp->date_end);
868 $line->info_bits = $objp->info_bits ;
869 $line->special_code = $objp->special_code;
870 $line->rang = $objp->rang;
871 $line->fk_unit = $objp->fk_unit;
872 $line->import_key = $objp->import_key;
873 $line->fk_user_author = $objp->fk_user_author;
874 $line->fk_user_modif = $objp->fk_user_modif;
875 $this->extraparams = !empty($objp->extraparams) ? (array) json_decode($objp->extraparams, true) : array();
876 $line->fk_multicurrency = $objp->fk_multicurrency;
877 $line->multicurrency_code = $objp->multicurrency_code;
878 $line->multicurrency_subprice = $objp->multicurrency_subprice;
879 $line->multicurrency_total_ht = $objp->multicurrency_total_ht;
880 $line->multicurrency_total_tva = $objp->multicurrency_total_tva;
881 $line->multicurrency_total_ttc = $objp->multicurrency_total_ttc;
882
883 $line->fetch_optionals();
884
885 $this->lines[$i] = $line;
886
887 $i++;
888 }
889
890 $this->db->free($result);
891 return 1;
892 } else {
893 $this->error = $this->db->lasterror();
894 return -3;
895 }
896 }
897
898
907 public function delete(User $user, $notrigger = 0, $idwarehouse = -1)
908 {
909 $rowid = $this->id;
910
911 dol_syslog(get_class($this)."::delete rowid=".((int) $rowid), LOG_DEBUG);
912
913 $error = 0;
914 $this->db->begin();
915
916 $main = MAIN_DB_PREFIX.'facture_fourn_det_rec';
917 $ef = $main."_extrafields";
918
919 $sqlef = "DELETE FROM ".$ef." WHERE fk_object IN (SELECT rowid FROM ".$main." WHERE fk_facture_fourn = ". (int) $rowid .")";
920 $sql = "DELETE FROM ".MAIN_DB_PREFIX."facture_fourn_det_rec WHERE fk_facture_fourn = ". (int) $rowid;
921
922 if ($this->db->query($sqlef) && $this->db->query($sql)) {
923 $sql = "DELETE FROM ".MAIN_DB_PREFIX."facture_fourn_rec WHERE rowid = ". (int) $rowid;
924 dol_syslog($sql);
925 if ($this->db->query($sql)) {
926 // Delete linked object
927 $res = $this->deleteObjectLinked();
928 if ($res < 0) {
929 $error = -3;
930 }
931 // Delete extrafields
932 $res = $this->deleteExtraFields();
933 if ($res < 0) {
934 $error = -4;
935 }
936 } else {
937 $this->error = $this->db->lasterror();
938 $error = -1;
939 }
940 } else {
941 $this->error = $this->db->lasterror();
942 $error = -2;
943 }
944 if (!$error && !$notrigger) {
945 // Call trigger
946 $result = $this->call_trigger('SUPPLIERBILLREC_DELETE', $user);
947 if ($result < 0) {
948 $error++;
949 }
950 // End call triggers
951 }
952 if (! $error) {
953 $this->db->commit();
954 return 1;
955 } else {
956 $this->db->rollback();
957 return $error;
958 }
959 }
960
987 public function addline($fk_product, $ref, $label, $desc, $pu_ht, $pu_ttc, $qty, $remise_percent, $txtva, $txlocaltax1 = 0, $txlocaltax2 = 0, $price_base_type = 'HT', $type = 0, $date_start = 0, $date_end = 0, $info_bits = 0, $special_code = 0, $rang = -1, $fk_unit = null, $pu_ht_devise = 0)
988 {
989 global $mysoc, $user;
990
991 include_once DOL_DOCUMENT_ROOT.'/core/lib/price.lib.php';
992
993 $facid = $this->id; //Supplier invoice template ID linked to
994
995 dol_syslog(get_class($this)."::addline facid=$facid,desc=$desc,pu_ht=$pu_ht,qty=$qty,txtva=$txtva,txlocaltax1=$txlocaltax1,txlocaltax2=$txlocaltax2,fk_product=$fk_product,remise_percent=$remise_percent,info_bits=$info_bits,price_base_type=$price_base_type,pu_ttc=$pu_ttc,type=$type,fk_unit=$fk_unit,pu_ht_devise=$pu_ht_devise,date_start_fill=$date_start,date_end_fill=$date_end", LOG_DEBUG);
996
997 // Check if object of the line is product or service
998 if ($type < 0) {
999 return -1;
1000 }
1001
1002 $localtaxes_type = getLocalTaxesFromRate($txtva, 0, $this->thirdparty, $mysoc);
1003
1004 // Clean vat code
1005 $reg = array();
1006 $vat_src_code = '';
1007 if (preg_match('/\‍((.*)\‍)/', (string) $txtva, $reg)) {
1008 $vat_src_code = $reg[1];
1009 $txtva = preg_replace('/\s*\‍(.*\‍)/', '', (string) $txtva); // Remove code into vatrate.
1010 }
1011
1012 // Clean parameters
1013 $fk_product = empty($fk_product) ? 0 : $fk_product;
1014 $label = empty($label) ? '' : $label;
1015 $remise_percent = empty($remise_percent) ? 0 : price2num($remise_percent);
1016 $qty = price2num($qty);
1017 $pu_ht = price2num($pu_ht);
1018 $pu_ttc = price2num($pu_ttc);
1019 if (!preg_match('/\‍((.*)\‍)/', $txtva)) {
1020 $txtva = price2num($txtva); // $txtva can have format '5.0(XXX)' or '5'
1021 }
1022 $txlocaltax1 = price2num($txlocaltax1);
1023 $txlocaltax2 = price2num($txlocaltax2);
1024 $txtva = !empty($txtva) ? $txtva : 0;
1025 $txlocaltax1 = !empty($txlocaltax1) ? $txlocaltax1 : 0;
1026 $txlocaltax2 = !empty($txlocaltax2) ? $txlocaltax2 : 0;
1027 $info_bits = !empty($info_bits) ? $info_bits : 0;
1028 $info_bits = !empty($info_bits) ? $info_bits : 0;
1029
1030 $pu = ($price_base_type == 'HT' ? $pu_ht : $pu_ttc);
1031
1032 // Calculation of the gross total (TTC) and VAT for the line from qty, pu, remise_percent and txtva
1033 // VERY IMPORTANT: It's at the time of line insertion that we must store the net, VAT, and gross amounts,
1034 // and this is done at the line level, which has its own VAT rate
1035
1036 $tabprice = calcul_price_total((float) $qty, (float) $pu, $remise_percent, $txtva, $txlocaltax1, $txlocaltax2, 0, $price_base_type, $info_bits, $type, $mysoc, $localtaxes_type, 100, $this->multicurrency_tx, $pu_ht_devise);
1037 $total_ht = $tabprice[0];
1038 $total_tva = $tabprice[1];
1039 $total_ttc = $tabprice[2];
1040 $total_localtax1 = $tabprice[9];
1041 $total_localtax2 = $tabprice[10];
1042 $pu_ht = $tabprice[3];
1043
1044 // MultiCurrency
1045 $multicurrency_total_ht = $tabprice[16];
1046 $multicurrency_total_tva = $tabprice[17];
1047 $multicurrency_total_ttc = $tabprice[18];
1048 $pu_ht_devise = $tabprice[19];
1049
1050 $this->db->begin();
1051 $product_type = $type;
1052 if ($fk_product) {
1053 $product = new Product($this->db);
1054 $result = $product->fetch($fk_product);
1055 if ($result < 0) {
1056 return -1;
1057 }
1058 $product_type = $product->type;
1059 if (empty($label)) {
1060 $label = $product->label;
1061 }
1062 }
1063
1064 $sql = 'INSERT INTO ' . MAIN_DB_PREFIX . 'facture_fourn_det_rec (';
1065 $sql .= 'fk_facture_fourn';
1066 $sql .= ', fk_product';
1067 $sql .= ', ref';
1068 $sql .= ', label';
1069 $sql .= ', description';
1070 $sql .= ', pu_ht';
1071 $sql .= ', pu_ttc';
1072 $sql .= ', qty';
1073 $sql .= ', remise_percent';
1074 $sql .= ', fk_remise_except';
1075 $sql .= ', vat_src_code';
1076 $sql .= ', tva_tx';
1077 $sql .= ', localtax1_tx';
1078 $sql .= ', localtax1_type';
1079 $sql .= ', localtax2_tx';
1080 $sql .= ', localtax2_type';
1081 $sql .= ', total_ht';
1082 $sql .= ', total_tva';
1083 $sql .= ', total_localtax1';
1084 $sql .= ', total_localtax2';
1085 $sql .= ', total_ttc';
1086 $sql .= ', product_type';
1087 $sql .= ', date_start';
1088 $sql .= ', date_end';
1089 $sql .= ', info_bits';
1090 $sql .= ', special_code';
1091 $sql .= ', rang';
1092 $sql .= ', fk_unit';
1093 $sql .= ', fk_user_author';
1094 $sql .= ', fk_multicurrency, multicurrency_code, multicurrency_subprice, multicurrency_total_ht, multicurrency_total_tva, multicurrency_total_ttc';
1095 $sql .= ') VALUES (';
1096 $sql .= ' ' . (int) $facid; // source supplier invoice id
1097 $sql .= ', ' . (!empty($fk_product) ? ((int) $fk_product) : 'null');
1098 $sql .= ', ' . (!empty($ref) ? "'" . $this->db->escape($ref) . "'" : 'null');
1099 $sql .= ', ' . (!empty($label) ? "'" . $this->db->escape($label) . "'" : 'null');
1100 $sql .= ", '" . $this->db->escape($desc) . "'";
1101 $sql .= ', ' . price2num($pu_ht);
1102 $sql .= ', ' . price2num($pu_ttc);
1103 $sql .= ', ' . price2num($qty);
1104 $sql .= ', ' . price2num($remise_percent);
1105 $sql .= ', null';
1106 $sql .= ", '" . $this->db->escape($vat_src_code) . "'";
1107 $sql .= ', ' . price2num($txtva);
1108 $sql .= ', ' . price2num($txlocaltax1);
1109 $sql .= ", '" . $this->db->escape(isset($localtaxes_type[0]) ? $localtaxes_type[0] : '') . "'";
1110 $sql .= ', ' . price2num($txlocaltax2);
1111 $sql .= ", '" . $this->db->escape(isset($localtaxes_type[2]) ? $localtaxes_type[2] : '') . "'";
1112 $sql .= ', ' . price2num($total_ht);
1113 $sql .= ', ' . price2num($total_tva);
1114 $sql .= ', ' . price2num($total_localtax1);
1115 $sql .= ', ' . price2num($total_localtax2);
1116 $sql .= ', ' . price2num($total_ttc);
1117 $sql .= ', ' . (int) $product_type;
1118 $sql .= ', ' . ($date_start > 0 ? (int) $date_start : 'NULL');
1119 $sql .= ', ' . ($date_end > 0 ? (int) $date_end : 'NULL');
1120 $sql .= ', ' . (int) $info_bits;
1121 $sql .= ', ' . (int) $special_code;
1122 $sql .= ', ' . (int) $rang;
1123 $sql .= ', ' . ($fk_unit ? (int) $fk_unit : 'NULL');
1124 $sql .= ', ' . (int) $user->id;
1125 $sql .= ', ' . (int) $this->fk_multicurrency;
1126 $sql .= ", '" . $this->db->escape($this->multicurrency_code) . "'";
1127 $sql .= ', ' . price2num($pu_ht_devise, 'CU');
1128 $sql .= ', ' . price2num($multicurrency_total_ht, 'CT');
1129 $sql .= ', ' . price2num($multicurrency_total_tva, 'CT');
1130 $sql .= ', ' . price2num($multicurrency_total_ttc, 'CT');
1131 $sql .= ')';
1132
1133 dol_syslog(get_class($this). '::addline', LOG_DEBUG);
1134 if ($this->db->query($sql)) {
1135 $lineId = $this->db->last_insert_id(MAIN_DB_PREFIX. 'facture_fourn_det_rec');
1136 $this->update_price();
1137 $this->id = $facid;
1138 $this->db->commit();
1139 return $lineId;
1140 } else {
1141 $this->db->rollback();
1142 $this->error = $this->db->lasterror();
1143
1144 return -1;
1145 }
1146 }
1147
1175 public function updateline($rowid, $fk_product, $ref, $label, $desc, $pu_ht, $qty, $remise_percent, $txtva, $txlocaltax1 = 0, $txlocaltax2 = 0, $price_base_type = 'HT', $type = 0, $date_start = 0, $date_end = 0, $info_bits = 0, $special_code = 0, $rang = -1, $fk_unit = null, $pu_ht_devise = 0, $pu_ttc = 0)
1176 {
1177 global $mysoc, $user;
1178
1179 $facid = $this->id;
1180
1181 dol_syslog(get_class($this). '::updateline facid=' .$facid." rowid=$rowid, desc=$desc, pu_ht=$pu_ht, qty=$qty, txtva=$txtva, txlocaltax1=$txlocaltax1, txlocaltax2=$txlocaltax2, fk_product=$fk_product, remise_percent=$remise_percent, info_bits=$info_bits, price_base_type=$price_base_type, pu_ttc=$pu_ttc, type=$type, fk_unit=$fk_unit, pu_ht_devise=$pu_ht_devise", LOG_DEBUG);
1182 include_once DOL_DOCUMENT_ROOT.'/core/lib/price.lib.php';
1183
1184 // Check parameters
1185 if ($type < 0) {
1186 return -1;
1187 }
1188
1189 // Clean parameters
1190 $fk_product = empty($fk_product) ? 0 : $fk_product;
1191 $label = empty($label) ? '' : $label;
1192 $remise_percent = empty($remise_percent) ? 0 : price2num($remise_percent);
1193 $qty = price2num($qty);
1194 $info_bits = empty($info_bits) ? 0 : $info_bits;
1195 $pu_ht = price2num($pu_ht);
1196 $pu_ttc = price2num($pu_ttc);
1197 $pu_ht_devise = price2num($pu_ht_devise);
1198
1199 if (!preg_match('/\‍((.*)\‍)/', (string) $txtva)) {
1200 $txtva = price2num($txtva); // $txtva can have format '5.0(XXX)' or '5'
1201 }
1202
1203 $txlocaltax1 = empty($txlocaltax1) ? 0 : price2num($txlocaltax1);
1204 $txlocaltax2 = empty($txlocaltax2) ? 0 : price2num($txlocaltax2);
1205 $this->multicurrency_total_ht = empty($this->multicurrency_total_ht) ? 0 : $this->multicurrency_total_ht;
1206 $this->multicurrency_total_tva = empty($this->multicurrency_total_tva) ? 0 : $this->multicurrency_total_tva;
1207 $this->multicurrency_total_ttc = empty($this->multicurrency_total_ttc) ? 0 : $this->multicurrency_total_ttc;
1208
1209 $pu = ($price_base_type == 'HT' ? $pu_ht : $pu_ttc);
1210
1211
1212 // Calculation of the gross total (TTC) and VAT for the line from qty, pu, remise_percent and txtva
1213 // VERY IMPORTANT: It's at the time of line insertion that we must store the net, VAT, and gross amounts,
1214 // and this is done at the line level, which has its own VAT rate
1215
1216 $localtaxes_type = getLocalTaxesFromRate($txtva, 0, $this->thirdparty, $mysoc);
1217
1218 // Clean vat code
1219 $vat_src_code = '';
1220 $reg = array();
1221 if (preg_match('/\‍((.*)\‍)/', $txtva, $reg)) {
1222 $vat_src_code = $reg[1];
1223 $txtva = preg_replace('/\s*\‍(.*\‍)/', '', $txtva); // Remove code into vatrate.
1224 }
1225
1226 $tabprice = calcul_price_total((float) $qty, (float) $pu, $remise_percent, $txtva, $txlocaltax1, $txlocaltax2, 0, $price_base_type, $info_bits, $type, $mysoc, $localtaxes_type, 100, $this->multicurrency_tx, (float) $pu_ht_devise);
1227
1228 $total_ht = $tabprice[0];
1229 $total_tva = $tabprice[1];
1230 $total_ttc = $tabprice[2];
1231 $total_localtax1 = $tabprice[9];
1232 $total_localtax2 = $tabprice[10];
1233 $pu_ht = $tabprice[3];
1234 $pu_tva = $tabprice[4];
1235 $pu_ttc = $tabprice[5];
1236
1237 // MultiCurrency
1238 $multicurrency_total_ht = $tabprice[16];
1239 $multicurrency_total_tva = $tabprice[17];
1240 $multicurrency_total_ttc = $tabprice[18];
1241 $pu_ht_devise = $tabprice[19];
1242
1243 $product_type = $type;
1244 if ($fk_product) {
1245 $product = new Product($this->db);
1246 $result = $product->fetch($fk_product);
1247 $product_type = $product->type;
1248 }
1249
1250 $sql = 'UPDATE ' . MAIN_DB_PREFIX . 'facture_fourn_det_rec SET';
1251 $sql .= ' fk_facture_fourn = ' . ((int) $facid);
1252 $sql .= ', fk_product = ' . ($fk_product > 0 ? ((int) $fk_product) : 'null');
1253 $sql .= ", ref = '" . $this->db->escape($ref) . "'";
1254 $sql .= ", label = '" . $this->db->escape($label) . "'";
1255 $sql .= ", description = '" . $this->db->escape($desc) . "'";
1256 $sql .= ', pu_ht = ' . price2num($pu_ht);
1257 $sql .= ', qty = ' . price2num($qty);
1258 $sql .= ", remise_percent = '" . price2num($remise_percent) . "'";
1259 $sql .= ", vat_src_code = '" . $this->db->escape($vat_src_code) . "'";
1260 $sql .= ', tva_tx = ' . price2num($txtva);
1261 $sql .= ', localtax1_tx = ' . (float) $txlocaltax1;
1262 $sql .= ", localtax1_type = '" . $this->db->escape($localtaxes_type[0]) . "'";
1263 $sql .= ', localtax2_tx = ' . (float) $txlocaltax2;
1264 $sql .= ", localtax2_type = '" . $this->db->escape($localtaxes_type[2]) . "'";
1265 $sql .= ", total_ht = '" . price2num($total_ht) . "'";
1266 $sql .= ", total_tva = '" . price2num($total_tva) . "'";
1267 $sql .= ", total_localtax1 = '" . price2num($total_localtax1) . "'";
1268 $sql .= ", total_localtax2 = '" . price2num($total_localtax2) . "'";
1269 $sql .= ", total_ttc = '" . price2num($total_ttc) . "'";
1270 $sql .= ', product_type = ' . (int) $product_type;
1271 $sql .= ', date_start = ' . (empty($date_start) ? 'NULL' : (int) $date_start);
1272 $sql .= ', date_end = ' . (empty($date_end) ? 'NULL' : (int) $date_end);
1273 $sql .= ', info_bits = ' . (int) $info_bits;
1274 $sql .= ', special_code = ' . (int) $special_code;
1275 $sql .= ', rang = ' . (int) $rang;
1276 $sql .= ', fk_unit = ' . ($fk_unit ? "'" . $this->db->escape($fk_unit) . "'" : 'null');
1277 $sql .= ', fk_user_modif = ' . (int) $user;
1278 $sql .= ', multicurrency_subprice = '.price2num($pu_ht_devise);
1279 $sql .= ', multicurrency_total_ht = '.price2num($multicurrency_total_ht);
1280 $sql .= ', multicurrency_total_tva = '.price2num($multicurrency_total_tva);
1281 $sql .= ', multicurrency_total_ttc = '.price2num($multicurrency_total_ttc);
1282 $sql .= ' WHERE rowid = ' . (int) $rowid;
1283
1284 dol_syslog(get_class($this). '::updateline', LOG_DEBUG);
1285 if ($this->db->query($sql)) {
1286 $this->id = $facid;
1287 $this->update_price();
1288 return 1;
1289 } else {
1290 $this->error = $this->db->lasterror();
1291 return -1;
1292 }
1293 }
1294
1295
1303 public function getNextNumRef($soc, $mode = 'next')
1304 {
1305 // Not used for recurring invoices
1306 return '';
1307 }
1308
1314 public function getNextDate()
1315 {
1316 if (empty($this->date_when)) {
1317 return false;
1318 }
1319 return dol_time_plus_duree((int) $this->date_when, $this->frequency, $this->unit_frequency, 1);
1320 }
1321
1327 public function isMaxNbGenReached()
1328 {
1329 $ret = false;
1330 if ($this->nb_gen_max > 0 && ($this->nb_gen_done >= $this->nb_gen_max)) {
1331 $ret = true;
1332 }
1333 return $ret;
1334 }
1335
1342 public function strikeIfMaxNbGenReached($ret)
1343 {
1344 // Special case to strike the date
1345 return ($this->isMaxNbGenReached() ? '<strike>' : '').$ret.($this->isMaxNbGenReached() ? '</strike>' : '');
1346 }
1347
1358 public function createRecurringInvoices($restrictioninvoiceid = 0, $forcevalidation = 0)
1359 {
1360 global $conf, $langs, $db, $user, $hookmanager;
1361
1362 $error = 0;
1363 $nb_create = 0;
1364
1365 // Load translation files required by the page
1366 $langs->loadLangs(array('main', 'bills'));
1367
1368 $now = dol_now();
1369 $tmparray = dol_getdate($now);
1370 $today = dol_mktime(23, 59, 59, $tmparray['mon'], $tmparray['mday'], $tmparray['year']); // Today is last second of current day
1371
1372 dol_syslog('createRecurringInvoices restrictioninvoiceid=' .$restrictioninvoiceid. ' forcevalidation=' .$forcevalidation);
1373
1374 $sql = 'SELECT rowid FROM '.MAIN_DB_PREFIX.'facture_fourn_rec';
1375 $sql .= ' WHERE frequency > 0'; // A recurring supplier invoice is an invoice with a frequency
1376 $sql .= " AND (date_when IS NULL OR date_when <= '".$this->db->idate($today)."')";
1377 $sql .= ' AND (nb_gen_done < nb_gen_max OR nb_gen_max = 0)';
1378 $sql .= ' AND suspended = 0';
1379 $sql .= ' AND entity = '. (int) $conf->entity; // MUST STAY = $conf->entity here
1380 if ($restrictioninvoiceid > 0) {
1381 $sql .= ' AND rowid = '. (int) $restrictioninvoiceid;
1382 }
1383 $sql .= $this->db->order('entity', 'ASC');
1384 if (getDolGlobalInt('NB_REC_FACT_GEN_BY_CALL')) {
1385 $sql .= $this->db->plimit(getDolGlobalInt('NB_REC_FACT_GEN_BY_CALL'));
1386 }
1387 //print $sql;exit;
1388 $parameters = array(
1389 'restrictioninvoiceid' => $restrictioninvoiceid,
1390 'forcevalidation' => $forcevalidation,
1391 );
1392 $reshook = $hookmanager->executeHooks('beforeCreationOfRecurringInvoices', $parameters, $sql); // note that $sql might be modified by hooks
1393
1394 $resql = $this->db->query($sql);
1395 if ($resql) {
1396 $i = 0;
1397 $num = $this->db->num_rows($resql);
1398
1399 if ($num) {
1400 $this->output .= $langs->trans('FoundXQualifiedRecurringInvoiceTemplate', $num)."\n";
1401 } else {
1402 $this->output .= $langs->trans('NoQualifiedRecurringInvoiceTemplateFound');
1403 }
1404
1405 $saventity = $conf->entity;
1406 $laststep = "None";
1407
1408 while ($i < $num) { // Loop on each template invoice. If $num = 0, test is false at first pass.
1409 $line = $this->db->fetch_object($resql);
1410
1411 $this->db->begin();
1412
1413 $invoiceidgenerated = 0;
1414
1415 $new_fac_fourn = null;
1416 $facturerec = new FactureFournisseurRec($this->db);
1417 $laststep = "Fetch {$line->rowid}";
1418 $facturerec->fetch($line->rowid);
1419
1420 if ($facturerec->id > 0) {
1421 // Set entity context
1422 $conf->entity = $facturerec->entity;
1423
1424 dol_syslog('createRecurringInvoices Process invoice template id=' .$facturerec->id. ', ref=' .$facturerec->ref. ', entity=' .$facturerec->entity);
1425
1426 $new_fac_fourn = new FactureFournisseur($this->db);
1427 $new_fac_fourn->fac_rec = $facturerec->id; // We will create $facture from this recurring invoice
1428 $new_fac_fourn->fk_fac_rec_source = $facturerec->id; // We will create $facture from this recurring invoice
1429
1430 $new_fac_fourn->type = self::TYPE_STANDARD;
1431 $new_fac_fourn->subtype = $facturerec->subtype;
1432 $new_fac_fourn->statut = self::STATUS_DRAFT; // deprecated
1433 $new_fac_fourn->status = self::STATUS_DRAFT;
1434 $new_fac_fourn->date = empty($facturerec->date_when) ? $now : $facturerec->date_when; // We could also use dol_now here but we prefer date_when so invoice has real date when we would like even if we generate later.
1435 $new_fac_fourn->socid = $facturerec->socid;
1436 $new_fac_fourn->lines = $facturerec->lines;
1437 $new_fac_fourn->ref_supplier = $facturerec->ref_supplier;
1438 $new_fac_fourn->model_pdf = $facturerec->model_pdf;
1439 $new_fac_fourn->fk_project = $facturerec->fk_project;
1440 $new_fac_fourn->label = $facturerec->label;
1441 $new_fac_fourn->libelle = $facturerec->label; // deprecated
1442
1443 $invoiceidgenerated = $new_fac_fourn->create($user);
1444 $laststep = "Create invoiceidgenerated $invoiceidgenerated";
1445 if ($invoiceidgenerated <= 0) {
1446 $this->setErrorsFromObject($new_fac_fourn);
1447 $error++;
1448 }
1449 if (!$error && ($facturerec->auto_validate || $forcevalidation)) {
1450 $result = $new_fac_fourn->validate($user);
1451 $laststep = "Validate by user {$user->login}";
1452 if ($result <= 0) {
1453 $this->setErrorsFromObject($new_fac_fourn);
1454 $error++;
1455 }
1456 }
1457
1458 if (!$error && $facturerec->generate_pdf) {
1459 // We refresh the object in order to have all necessary data (like date_lim_reglement)
1460 $laststep = "Refresh ".$new_fac_fourn->id;
1461 $new_fac_fourn->fetch($new_fac_fourn->id);
1462 $laststep = "GenerateDocument ".$new_fac_fourn->id;
1463 $result = $new_fac_fourn->generateDocument($facturerec->model_pdf, $langs);
1464 if ($result < 0) {
1465 $this->setErrorsFromObject($new_fac_fourn);
1466 $error++;
1467 }
1468 }
1469 } else {
1470 $error++;
1471 $this->error = 'Failed to load invoice template with id=' .$line->rowid. ', entity=' .$conf->entity."\n";
1472 $this->errors[] = 'Failed to load invoice template with id=' .$line->rowid. ', entity=' .$conf->entity;
1473 dol_syslog('createRecurringInvoices Failed to load invoice template with id=' .$line->rowid. ', entity=' .$conf->entity);
1474 }
1475
1476 if (!$error && $invoiceidgenerated >= 0) {
1477 $facturerec->nb_gen_done++;
1478 $facturerec->date_last_gen = dol_now();
1479 $nextDate = $facturerec->getNextDate();
1480 $facturerec->date_when = (($nextDate === false) ? null : $nextDate);
1481 $facturerec->update($user);
1482 $this->db->commit('createRecurringInvoices Process invoice template id=' .$facturerec->id. ', title=' .$facturerec->title);
1483 dol_syslog('createRecurringInvoices Process invoice template ' .$facturerec->title. ' is finished with a success generation');
1484 $nb_create++;
1485 $this->output .= $langs->trans('InvoiceGeneratedFromTemplate', $new_fac_fourn->ref, $facturerec->title)."\n";
1486 } else {
1487 $this->db->rollback('createRecurringInvoices Process invoice template error='.$error.' invoiceidgenerated='.$invoiceidgenerated.' LastStep='.$laststep.' id=' .$facturerec->id. ', title=' .$facturerec->title);
1488 }
1489
1490 $parameters = array(
1491 'cpt' => $i,
1492 'total' => $num,
1493 'errorCount' => $error,
1494 'invoiceidgenerated' => $invoiceidgenerated,
1495 'facturerec' => $facturerec, // it's an object which PHP passes by "reference", so modifiable by hooks.
1496 'this' => $this, // it's an object which PHP passes by "reference", so modifiable by hooks.
1497 );
1498 // @phan-suppress-next-line PhanTypeMismatchArgumentNullable
1499 $reshook = $hookmanager->executeHooks('afterCreationOfRecurringInvoice', $parameters, $new_fac_fourn); // note: $facture can be modified by hooks (warning: $facture can be null)
1500
1501 $i++;
1502 }
1503
1504 $conf->entity = $saventity; // Restore entity context
1505 } else {
1506 dol_print_error($this->db);
1507 }
1508
1509 $this->output = trim($this->output);
1510
1511 return $error ? $error : 0;
1512 }
1513
1526 public function getNomUrl($withpicto = 0, $option = '', $max = 0, $short = 0, $moretitle = '', $notooltip = 0, $save_lastsearch_value = -1)
1527 {
1528 global $langs, $hookmanager;
1529
1530 $result = '';
1531
1532 $label = '<u>'.$langs->trans('RepeatableInvoice').'</u>';
1533 if (!empty($this->ref)) {
1534 $label .= '<br><b>'.$langs->trans('Ref').':</b> '.$this->ref;
1535 }
1536 if ($this->frequency > 0) {
1537 $label .= '<br><b>'.$langs->trans('Frequency').':</b> '.$langs->trans('FrequencyPer_'.$this->unit_frequency, $this->frequency);
1538 }
1539 if (!empty($this->date_last_gen)) {
1540 $label .= '<br><b>'.$langs->trans('DateLastGeneration').':</b> '.dol_print_date($this->date_last_gen, 'dayhour');
1541 }
1542 if ($this->frequency > 0) {
1543 if (!empty($this->date_when)) {
1544 $label .= '<br><b>'.$langs->trans('NextDateToExecution').':</b> ';
1545 $label .= (empty($this->suspended) ? '' : '<strike>').dol_print_date($this->date_when, 'day').(empty($this->suspended) ? '' : '</strike>'); // No hour for this property
1546 if (!empty($this->suspended)) {
1547 $label .= ' ('.$langs->trans('Disabled').')';
1548 }
1549 }
1550 }
1551
1552 $url = DOL_URL_ROOT.'/fourn/facture/card-rec.php?facid='.$this->id;
1553
1554 if ($short) {
1555 return $url;
1556 }
1557
1558 if ($option != 'nolink') {
1559 // Add param to save lastsearch_values or not
1560 $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
1561 if ($save_lastsearch_value == -1 && preg_match('/list\.php/', $_SERVER['PHP_SELF'])) {
1562 $add_save_lastsearch_values = 1;
1563 }
1564 if ($add_save_lastsearch_values) {
1565 $url .= '&save_lastsearch_values=1';
1566 }
1567 }
1568
1569 $linkstart = '<a href="'.$url.'" title="'.dol_escape_htmltag($label, 1).'" class="classfortooltip">';
1570 $linkend = '</a>';
1571
1572 $result .= $linkstart;
1573 if ($withpicto) {
1574 $result .= img_object(($notooltip ? '' : $label), ($this->picto ? $this->picto : 'generic'), ($notooltip ? (($withpicto != 2) ? 'class="paddingright"' : '') : 'class="'.(($withpicto != 2) ? 'paddingright ' : '').'classfortooltip"'), 0, 0, $notooltip ? 0 : 1);
1575 }
1576 if ($withpicto != 2) {
1577 $result .= $this->ref;
1578 }
1579 $result .= $linkend;
1580 global $action;
1581 $hookmanager->initHooks(array($this->element . 'dao'));
1582 $parameters = array('id' => $this->id, 'getnomurl' => &$result);
1583 $reshook = $hookmanager->executeHooks('getNomUrl', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
1584 if ($reshook > 0) {
1585 $result = $hookmanager->resPrint;
1586 } else {
1587 $result .= $hookmanager->resPrint;
1588 }
1589 return $result;
1590 }
1591
1599 public function getLibStatut($mode = 0, $alreadypaid = -1)
1600 {
1601 return $this->LibStatut($this->frequency ? 1 : 0, $this->suspended, $mode, $alreadypaid, empty($this->type) ? 0 : $this->type);
1602 }
1603
1604 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1616 public function LibStatut($recur, $status, $mode = 0, $alreadypaid = -1, $type = 0, $nbofopendirectdebitorcredittransfer = 0)
1617 {
1618 // phpcs:enable
1619 global $langs;
1620 $langs->load('bills');
1621
1622 $labelStatus = $langs->transnoentitiesnoconv('Active');
1623 $statusType = 'status0';
1624
1625 //print "$recur,$status,$mode,$alreadypaid,$type";
1626 if ($mode == 0) {
1627 if ($recur) {
1628 if ($status == self::STATUS_SUSPENDED) {
1629 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1630 } else {
1631 $labelStatus = $langs->transnoentitiesnoconv('Active');
1632 }
1633 } else {
1634 if ($status == self::STATUS_SUSPENDED) {
1635 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1636 } else {
1637 $labelStatus = $langs->transnoentitiesnoconv('Draft');
1638 }
1639 }
1640 } elseif ($mode == 1) {
1641 $prefix = 'Short';
1642 if ($recur) {
1643 if ($status == self::STATUS_SUSPENDED) {
1644 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1645 } else {
1646 $labelStatus = $langs->transnoentitiesnoconv('Active');
1647 }
1648 } else {
1649 if ($status == self::STATUS_SUSPENDED) {
1650 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1651 } else {
1652 $labelStatus = $langs->transnoentitiesnoconv('Draft');
1653 }
1654 }
1655 } elseif ($mode == 2) {
1656 if ($recur) {
1657 if ($status == self::STATUS_SUSPENDED) {
1658 $statusType = 'status6';
1659 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1660 } else {
1661 $statusType = 'status4';
1662 $labelStatus = $langs->transnoentitiesnoconv('Active');
1663 }
1664 } else {
1665 if ($status == self::STATUS_SUSPENDED) {
1666 $statusType = 'status6';
1667 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1668 } else {
1669 $statusType = 'status0';
1670 $labelStatus = $langs->transnoentitiesnoconv('Draft');
1671 }
1672 }
1673 } elseif ($mode == 3) {
1674 if ($recur) {
1675 $prefix = 'Short';
1676 if ($status == self::STATUS_SUSPENDED) {
1677 $statusType = 'status6';
1678 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1679 } else {
1680 $statusType = 'status4';
1681 $labelStatus = $langs->transnoentitiesnoconv('Active');
1682 }
1683 } else {
1684 if ($status == self::STATUS_SUSPENDED) {
1685 $statusType = 'status6';
1686 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1687 } else {
1688 $statusType = 'status0';
1689 $labelStatus = $langs->transnoentitiesnoconv('Draft');
1690 }
1691 }
1692 } elseif ($mode == 4) {
1693 $prefix = '';
1694 if ($recur) {
1695 if ($status == self::STATUS_SUSPENDED) {
1696 $statusType = 'status6';
1697 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1698 } else {
1699 $statusType = 'status4';
1700 $labelStatus = $langs->transnoentitiesnoconv('Active');
1701 }
1702 } else {
1703 if ($status == self::STATUS_SUSPENDED) {
1704 $statusType = 'status6';
1705 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1706 } else {
1707 $statusType = 'status0';
1708 $labelStatus = $langs->transnoentitiesnoconv('Draft');
1709 }
1710 }
1711 } elseif ($mode == 5 || $mode == 6) {
1712 $prefix = '';
1713 if ($mode == 5) {
1714 $prefix = 'Short';
1715 }
1716 if ($recur) {
1717 if ($status == self::STATUS_SUSPENDED) {
1718 $statusType = 'status6';
1719 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1720 } else {
1721 $statusType = 'status4';
1722 $labelStatus = $langs->transnoentitiesnoconv('Active');
1723 }
1724 } else {
1725 if ($status == self::STATUS_SUSPENDED) {
1726 $statusType = 'status6';
1727 $labelStatus = $langs->transnoentitiesnoconv('Disabled');
1728 } else {
1729 $statusType = 'status0';
1730 $labelStatus = $langs->transnoentitiesnoconv('Draft');
1731 }
1732 }
1733 }
1734
1735 $labelStatusShort = $labelStatus;
1736
1737 return dolGetStatus($labelStatus, $labelStatusShort, '', $statusType, $mode);
1738 }
1739
1748 public function initAsSpecimen($option = '')
1749 {
1750 global $user, $langs, $conf;
1751
1752 $now = dol_now();
1753 $arraynow = dol_getdate($now);
1754 $nownotime = dol_mktime(0, 0, 0, $arraynow['mon'], $arraynow['mday'], $arraynow['year']);
1755
1756 // Load array of products prodids
1757 $num_prods = 0;
1758 $prodids = array();
1759
1760 $sql = 'SELECT rowid';
1761 $sql .= ' FROM ' .MAIN_DB_PREFIX. 'product';
1762 $sql .= ' WHERE entity IN (' .getEntity('product'). ')';
1763 $sql .= $this->db->plimit(100);
1764
1765 $resql = $this->db->query($sql);
1766 if ($resql) {
1767 $num_prods = $this->db->num_rows($resql);
1768 $i = 0;
1769 while ($i < $num_prods) {
1770 $i++;
1771 $row = $this->db->fetch_row($resql);
1772 $prodids[$i] = $row[0];
1773 }
1774 }
1775
1776 // Initialize parameters
1777 $this->id = 0;
1778 $this->ref = 'SPECIMEN';
1779 $this->title = 'SPECIMEN';
1780 $this->specimen = 1;
1781 $this->socid = 1;
1782 $this->date = $nownotime;
1783 $this->date_lim_reglement = $nownotime + 3600 * 24 * 30;
1784 $this->cond_reglement_id = 1;
1785 $this->cond_reglement_code = 'RECEP';
1786 $this->date_lim_reglement = $this->calculate_date_lim_reglement();
1787 $this->mode_reglement_id = 0; // Not forced to show payment mode CHQ + VIR
1788 $this->mode_reglement_code = ''; // Not forced to show payment mode CHQ + VIR
1789 $this->note_public = 'This is a comment (public)';
1790 $this->note_private = 'This is a comment (private)';
1791 $this->note = 'This is a comment (private)';
1792 $this->fk_incoterms = 0;
1793 $this->location_incoterms = '';
1794
1795 if (empty($option) || $option != 'nolines') {
1796 // Lines
1797 $nbp = min(1000, GETPOSTINT('nblines') ? GETPOSTINT('nblines') : 5); // We can force the nb of lines to test from command line (but not more than 1000)
1798 $xnbp = 0;
1799 while ($xnbp < $nbp) {
1800 $line = new FactureLigne($this->db);
1801 $line->desc = $langs->trans('Description'). ' ' .$xnbp;
1802 $line->qty = 1;
1803 $line->subprice = 100;
1804 $line->tva_tx = 19.6;
1805 $line->localtax1_tx = 0;
1806 $line->localtax2_tx = 0;
1807 $line->remise_percent = 0;
1808 if ($xnbp == 1) { // Qty is negative (product line)
1809 $prodid = mt_rand(1, $num_prods);
1810 $line->fk_product = $prodids[$prodid];
1811 $line->qty = -1;
1812 $line->total_ht = -100;
1813 $line->total_ttc = -119.6;
1814 $line->total_tva = -19.6;
1815 } elseif ($xnbp == 2) { // UP is negative (free line)
1816 $line->subprice = -100;
1817 $line->total_ht = -100;
1818 $line->total_ttc = -119.6;
1819 $line->total_tva = -19.6;
1820 $line->remise_percent = 0;
1821 } elseif ($xnbp == 3) { // Discount is 50% (product line)
1822 $prodid = mt_rand(1, $num_prods);
1823 $line->fk_product = $prodids[$prodid];
1824 $line->total_ht = 50;
1825 $line->total_ttc = 59.8;
1826 $line->total_tva = 9.8;
1827 $line->remise_percent = 50;
1828 } else { // (product line)
1829 $prodid = mt_rand(1, $num_prods);
1830 $line->fk_product = $prodids[$prodid];
1831 $line->total_ht = 100;
1832 $line->total_ttc = 119.6;
1833 $line->total_tva = 19.6;
1834 $line->remise_percent = 00;
1835 }
1836
1837 $this->lines[$xnbp] = $line;
1838 $xnbp++;
1839
1840 $this->total_ht += $line->total_ht;
1841 $this->total_tva += $line->total_tva;
1842 $this->total_ttc += $line->total_ttc;
1843 }
1844 $this->revenuestamp = 0;
1845
1846 // Add a line "offered"
1847 $line = new FactureLigne($this->db);
1848 $line->desc = $langs->trans('Description'). ' (offered line)';
1849 $line->qty = 1;
1850 $line->subprice = 100;
1851 $line->tva_tx = 19.6;
1852 $line->localtax1_tx = 0;
1853 $line->localtax2_tx = 0;
1854 $line->remise_percent = 100;
1855 $line->total_ht = 0;
1856 $line->total_ttc = 0; // 90 * 1.196
1857 $line->total_tva = 0;
1858 $prodid = mt_rand(1, $num_prods);
1859 $line->fk_product = $prodids[$prodid];
1860
1861 $this->lines[$xnbp] = $line;
1862 $xnbp++;
1863 }
1864
1865 $this->usenewprice = 0;
1866
1867 return 1;
1868 }
1869
1878 public static function replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
1879 {
1880 $tables = array(
1881 'facture_fourn_rec'
1882 );
1883
1884 return CommonObject::commonReplaceThirdparty($dbs, $origin_id, $dest_id, $tables);
1885 }
1886
1894 public function setFrequencyAndUnit($frequency, $unit)
1895 {
1896 if (!$this->table_element) {
1897 dol_syslog(get_class($this). '::setFrequencyAndUnit was called on object with property table_element not defined', LOG_ERR);
1898 return -1;
1899 }
1900
1901 if (!empty($frequency) && empty($unit)) {
1902 dol_syslog(get_class($this). '::setFrequencyAndUnit was called on object with params frequency defined but unit not defined', LOG_ERR);
1903 return -2;
1904 }
1905
1906 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
1907 $sql .= " SET frequency = ".($frequency ? ((int) $frequency) : "NULL");
1908 if (!empty($unit)) {
1909 $sql .= ", unit_frequency = '".$this->db->escape($unit)."'";
1910 }
1911 $sql .= " WHERE rowid = ".((int) $this->id);
1912
1913 dol_syslog(get_class($this).'::setFrequencyAndUnit', LOG_DEBUG);
1914
1915 if ($this->db->query($sql)) {
1916 $this->frequency = $frequency;
1917 if (!empty($unit)) {
1918 $this->unit_frequency = $unit;
1919 }
1920 return 1;
1921 } else {
1922 $this->error = $this->db->lasterror();
1923 return -1;
1924 }
1925 }
1926
1934 public function setNextDate($date, $increment_nb_gen_done = 0)
1935 {
1936 if (!$this->table_element) {
1937 dol_syslog(get_class($this).'::setNextDate was called on object with property table_element not defined', LOG_ERR);
1938 return -1;
1939 }
1940 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
1941 $sql .= " SET date_when = " .($date ? "'".$this->db->idate($date)."'" : "NULL");
1942 if ($increment_nb_gen_done > 0) {
1943 $sql .= ", nb_gen_done = nb_gen_done + 1";
1944 }
1945 $sql .= " WHERE rowid = " . (int) $this->id;
1946
1947 dol_syslog(get_class($this).'::setNextDate', LOG_DEBUG);
1948
1949 if ($this->db->query($sql)) {
1950 $this->date_when = $date;
1951 if ($increment_nb_gen_done > 0) {
1952 $this->nb_gen_done++;
1953 }
1954 return 1;
1955 } else {
1956 $this->error = $this->db->lasterror();
1957 return -1;
1958 }
1959 }
1960
1967 public function setMaxPeriod($nb)
1968 {
1969 if (!$this->table_element) {
1970 dol_syslog(get_class($this).'::setMaxPeriod was called on object with property table_element not defined', LOG_ERR);
1971 return -1;
1972 }
1973
1974 if (empty($nb)) {
1975 $nb = 0;
1976 }
1977
1978 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
1979 $sql .= " SET nb_gen_max = ". (int) $nb;
1980 $sql .= " WHERE rowid = " . (int) $this->id;
1981
1982 dol_syslog(get_class($this).'::setMaxPeriod', LOG_DEBUG);
1983
1984 if ($this->db->query($sql)) {
1985 $this->nb_gen_max = $nb;
1986 return 1;
1987 } else {
1988 dol_print_error($this->db);
1989 return -1;
1990 }
1991 }
1992
1999 public function setAutoValidate($validate)
2000 {
2001 if (!$this->table_element) {
2002 dol_syslog(get_class($this).'::setAutoValidate was called on object with property table_element not defined', LOG_ERR);
2003 return -1;
2004 }
2005
2006 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
2007 $sql .= " SET auto_validate = ".((int) $validate);
2008 $sql .= " WHERE rowid = " . (int) $this->id;
2009
2010 dol_syslog(get_class($this).'::setAutoValidate', LOG_DEBUG);
2011
2012 if ($this->db->query($sql)) {
2013 $this->auto_validate = $validate;
2014 return 1;
2015 } else {
2016 dol_print_error($this->db);
2017 return -1;
2018 }
2019 }
2020
2027 public function setGeneratePdf($validate)
2028 {
2029 if (!$this->table_element) {
2030 dol_syslog(get_class($this).'::setGeneratePdf was called on object with property table_element not defined', LOG_ERR);
2031 return -1;
2032 }
2033
2034 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
2035 $sql .= " SET generate_pdf = ". (int) $validate;
2036 $sql .= " WHERE rowid = " . (int) $this->id;
2037
2038 dol_syslog(get_class($this).'::setGeneratePdf', LOG_DEBUG);
2039
2040 if ($this->db->query($sql)) {
2041 $this->generate_pdf = $validate;
2042 return 1;
2043 } else {
2044 dol_print_error($this->db);
2045 return -1;
2046 }
2047 }
2048
2055 public function setModelPdf($model)
2056 {
2057 if (!$this->table_element) {
2058 dol_syslog(get_class($this).'::setModelPdf was called on object with property table_element not defined', LOG_ERR);
2059 return -1;
2060 }
2061
2062 $sql = "UPDATE ".MAIN_DB_PREFIX.$this->table_element;
2063 $sql .= " SET modelpdf = '".$this->db->escape($model)."'";
2064 $sql .= " WHERE rowid = " . (int) $this->id;
2065
2066 dol_syslog(get_class($this).'::setModelPdf', LOG_DEBUG);
2067
2068 if ($this->db->query($sql)) {
2069 $this->model_pdf = $model;
2070 return 1;
2071 } else {
2072 dol_print_error($this->db);
2073 return -1;
2074 }
2075 }
2076}
$object ref
Definition info.php:90
Superclass for invoice classes.
const TYPE_STANDARD
Standard invoice.
calculate_date_lim_reglement($cond_reglement=0)
Returns an invoice payment deadline based on the invoice settlement conditions and billing date.
const STATUS_DRAFT
Draft status.
fetch_optionals($rowid=null, $optionsArray=null)
Function to get extra fields of an object into $this->array_options This method is in most cases call...
update_price($exclspec=0, $roundingadjust='auto', $nodatabaseupdate=0, $seller=null)
Update total_ht, total_ttc, total_vat, total_localtax1, total_localtax2 for an object (sum of lines).
add_object_linked($origin=null, $origin_id=null, $f_user=null, $notrigger=0)
Add an object link into llx_element_element.
deleteObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $rowid=0, $f_user=null, $notrigger=0)
Delete all links between an object $this.
setErrorsFromObject($object)
setErrorsFromObject
deleteExtraFields()
Delete all extra fields values for the current object.
static commonReplaceThirdparty(DoliDB $dbs, $origin_id, $dest_id, array $tables, $ignoreerrors=0)
Function used to replace a thirdparty id with another one.
insertExtraFields($trigger='', $userused=null)
Add/Update all extra fields values for the current object.
Class to manage Dolibarr database access.
Class to manage suppliers invoices.
Class to manage supplier invoice lines of templates.
Class to manage invoice templates.
setModelPdf($model)
Update the model for documents.
fetch($rowid, $ref='', $ref_ext='')
Load object and lines.
strikeIfMaxNbGenReached($ret)
Format string to output with by striking the string if max number of generation was reached.
setFrequencyAndUnit($frequency, $unit)
Update frequency and unit.
getNextNumRef($soc, $mode='next')
Return next reference of invoice not already used (or last reference)
setGeneratePdf($validate)
Update the auto generate documents.
getLinesArray()
Create an array of invoice lines.
initAsSpecimen($option='')
Initialise an instance with random values.
updateline($rowid, $fk_product, $ref, $label, $desc, $pu_ht, $qty, $remise_percent, $txtva, $txlocaltax1=0, $txlocaltax2=0, $price_base_type='HT', $type=0, $date_start=0, $date_end=0, $info_bits=0, $special_code=0, $rang=-1, $fk_unit=null, $pu_ht_devise=0, $pu_ttc=0)
Update a line to supplier invoice template.
isMaxNbGenReached()
Return if maximum number of generation is reached.
LibStatut($recur, $status, $mode=0, $alreadypaid=-1, $type=0, $nbofopendirectdebitorcredittransfer=0)
Return label of a status.
setAutoValidate($validate)
Update the auto validate flag of invoice.
static replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
Function used to replace a thirdparty id with another one.
getNextDate()
Return the next date of.
fetch_lines()
Get lines of template invoices into this->lines.
getLibStatut($mode=0, $alreadypaid=-1)
Return label of object status.
setNextDate($date, $increment_nb_gen_done=0)
Update the next date of execution.
setMaxPeriod($nb)
Update the maximum period.
createRecurringInvoices($restrictioninvoiceid=0, $forcevalidation=0)
Create all recurrents supplier invoices (for all entities if multicompany is used).
update(User $user, $notrigger=0)
Update fourn_invoice_rec.
addline($fk_product, $ref, $label, $desc, $pu_ht, $pu_ttc, $qty, $remise_percent, $txtva, $txlocaltax1=0, $txlocaltax2=0, $price_base_type='HT', $type=0, $date_start=0, $date_end=0, $info_bits=0, $special_code=0, $rang=-1, $fk_unit=null, $pu_ht_devise=0)
Add a line to recursive supplier invoice.
getNomUrl($withpicto=0, $option='', $max=0, $short=0, $moretitle='', $notooltip=0, $save_lastsearch_value=-1)
Return clickable name (with picto eventually)
create($user, $facFournId, $notrigger=0, $onlylines=array())
Create a predefined supplier invoice.
Class to manage invoice lines.
Class to manage products or services.
Class to manage Dolibarr users.
print $langs trans("Ref").' m titre as m m statut as status
Or an array listing all the potential status of the object: array: int of the status => translated la...
Definition index.php:168
global $mysoc
dol_time_plus_duree($time, $duration_value, $duration_unit, $ruleforendofmonth=0)
Add a delay to a date.
Definition date.lib.php:126
if(!isModEnabled('ai')||!getDolGlobalString('AI_ASSISTANT_ENABLED')) global $conf
The main.inc.php has been included so the following variable are now defined:
$date_start
Variables from include:
dol_now($mode='gmt')
Return date for now.
dol_mktime($hour, $minute, $second, $month, $day, $year, $gm='auto', $check=1)
Return a timestamp date built from detailed information (by default a local PHP server timestamp) Rep...
GETPOSTINT($paramname, $method=0)
Return the value of a $_GET or $_POST supervariable, converted into integer.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
img_object($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $allowothertags=array())
Show a picto called object_picto (generic function)
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
getLocalTaxesFromRate($vatrate, $local, $buyer, $seller, $firstparamisid=0)
Get type and rate of localtaxes for a particular vat rate/country of a thirdparty.
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...
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
dol_getdate($timestamp, $fast=false, $forcetimezone='')
Return an array with locale date info.
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0, $cleanalsojavascript=0)
Returns text escaped for inclusion in HTML alt or title or value tags, or into values of HTML input f...
calcul_price_total($qty, $pu, $remise_percent_ligne, $txtva, $uselocaltax1_rate, $uselocaltax2_rate, $remise_percent_global, $price_base_type, $info_bits, $type, $seller=null, $localtaxes_array=[], $progress=100, $multicurrency_tx=1, $pu_devise=0, $multicurrency_code='')
Calculate totals (net, vat, ...) of a line.
Definition price.lib.php:90
print $langs trans('Date')." left Ref Label right Qty right Price right TotalHT right TotalTTC right right right right right right right right right centpercent right TotalHT right n right VAT right n right TotalVAT right n No sujeto a RE IRPF right TotalLT1 right n right TotalLT2 right n right TotalTTC right n takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency right TotalTTC takeposcustomercurrency right takeposcustomercurrency n right Paid right PaymentTypeShortLIQ right SELECT p pos_change as p datep as date
Definition receipt.php:487
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:130