dolibarr 21.0.0-alpha
contrat.class.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2003 Rodolphe Quiedeville <rodolphe@quiedeville.org>
3 * Copyright (C) 2004-2012 Destailleur Laurent <eldy@users.sourceforge.net>
4 * Copyright (C) 2005-2014 Regis Houssin <regis.houssin@inodbox.com>
5 * Copyright (C) 2006 Andre Cianfarani <acianfa@free.fr>
6 * Copyright (C) 2008 Raphael Bertrand <raphael.bertrand@resultic.fr>
7 * Copyright (C) 2010-2016 Juanjo Menent <jmenent@2byte.es>
8 * Copyright (C) 2013 Christophe Battarel <christophe.battarel@altairis.fr>
9 * Copyright (C) 2013 Florian Henry <florian.henry@open-concept.pro>
10 * Copyright (C) 2014-2015 Marcos García <marcosgdf@gmail.com>
11 * Copyright (C) 2018 Nicolas ZABOURI <info@inovea-conseil.com>
12 * Copyright (C) 2018-2024 Frédéric France <frederic.france@free.fr>
13 * Copyright (C) 2015-2018 Ferran Marcet <fmarcet@2byte.es>
14 * Copyright (C) 2024 William Mead <william.mead@manchenumerique.fr>
15 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
16 *
17 * This program is free software; you can redistribute it and/or modify
18 * it under the terms of the GNU General Public License as published by
19 * the Free Software Foundation; either version 3 of the License, or
20 * (at your option) any later version.
21 *
22 * This program is distributed in the hope that it will be useful,
23 * but WITHOUT ANY WARRANTY; without even the implied warranty of
24 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
25 * GNU General Public License for more details.
26 *
27 * You should have received a copy of the GNU General Public License
28 * along with this program. If not, see <https://www.gnu.org/licenses/>.
29 */
30
37require_once DOL_DOCUMENT_ROOT.'/core/class/commonobject.class.php';
38require_once DOL_DOCUMENT_ROOT."/core/class/commonobjectline.class.php";
39require_once DOL_DOCUMENT_ROOT.'/core/lib/price.lib.php';
40require_once DOL_DOCUMENT_ROOT.'/margin/lib/margins.lib.php';
41
45class Contrat extends CommonObject
46{
50 public $element = 'contrat';
51
55 public $table_element = 'contrat';
56
60 public $table_element_line = 'contratdet';
61
65 public $fk_element = 'fk_contrat';
66
70 public $picto = 'contract';
71
76 public $restrictiononfksoc = 1;
77
81 protected $table_ref_field = 'ref';
82
87 public $ref_customer;
88 public $from;
89
94 public $ref_supplier;
95
100 public $entity;
101
106 public $socid;
107
113 public $fk_soc;
114
115
116 public $societe; // Object societe
117
123 public $statut = 0;
128 public $status = 0;
129
130 public $product;
131
135 public $fk_user_author;
136
142 public $user_author_id;
143
144
149 public $user_creation;
150
154 public $user_cloture;
155
159 public $date_contrat;
160
165 public $signed_status = 0;
166
167 public $commercial_signature_id;
168 public $fk_commercial_signature;
169 public $commercial_suivi_id;
170 public $fk_commercial_suivi;
171
177
178 public $extraparams = array();
179
183 public $lines = array();
184
185 public $nbofservices;
186 public $nbofserviceswait;
187 public $nbofservicesopened;
188 public $nbofservicesexpired;
189 public $nbofservicesclosed;
190 //public $lower_planned_end_date;
191 //public $higher_planner_end_date;
192
197 protected $lines_id_index_mapper = array();
198
199
224 // BEGIN MODULEBUILDER PROPERTIES
228 public $fields = array(
229 'rowid' => array('type' => 'integer', 'label' => 'TechnicalID', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 10),
230 'ref' => array('type' => 'varchar(50)', 'label' => 'Ref', 'enabled' => 1, 'visible' => -1, 'showoncombobox' => 1, 'position' => 15, 'searchall' => 1),
231 'ref_ext' => array('type' => 'varchar(255)', 'label' => 'Ref ext', 'enabled' => 1, 'visible' => 0, 'position' => 20),
232 'ref_customer' => array('type' => 'varchar(50)', 'label' => 'RefCustomer', 'enabled' => 1, 'visible' => -1, 'position' => 25, 'searchall' => 1),
233 'ref_supplier' => array('type' => 'varchar(50)', 'label' => 'RefSupplier', 'enabled' => 1, 'visible' => -1, 'position' => 26, 'searchall' => 1),
234 'entity' => array('type' => 'integer', 'label' => 'Entity', 'default' => '1', 'enabled' => 1, 'visible' => -2, 'notnull' => 1, 'position' => 30, 'index' => 1),
235 'tms' => array('type' => 'timestamp', 'label' => 'DateModification', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 35),
236 'datec' => array('type' => 'datetime', 'label' => 'DateCreation', 'enabled' => 1, 'visible' => -1, 'position' => 40),
237 'date_contrat' => array('type' => 'datetime', 'label' => 'Date contrat', 'enabled' => 1, 'visible' => -1, 'position' => 45),
238 'signed_status' => array('type' => 'smallint(6)', 'label' => 'SignedStatus', 'enabled' => 1, 'visible' => -1, 'position' => 50, 'arrayofkeyval' => array(0 => 'NoSignature', 1 => 'SignedSender', 2 => 'SignedReceiver', 9 => 'SignedAll')),
239 'fk_soc' => array('type' => 'integer:Societe:societe/class/societe.class.php', 'label' => 'ThirdParty', 'enabled' => 'isModEnabled("societe")', 'visible' => -1, 'notnull' => 1, 'position' => 70),
240 'fk_projet' => array('type' => 'integer:Project:projet/class/project.class.php:1:(fk_statut:=:1)', 'label' => 'Project', 'enabled' => "isModEnabled('project')", 'visible' => -1, 'position' => 75),
241 'fk_commercial_signature' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'SaleRepresentative Signature', 'enabled' => 1, 'visible' => -1, 'position' => 80),
242 'fk_commercial_suivi' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'SaleRepresentative follower', 'enabled' => 1, 'visible' => -1, 'position' => 85),
243 'fk_user_author' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserAuthor', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 90),
244 'note_public' => array('type' => 'html', 'label' => 'NotePublic', 'enabled' => 1, 'visible' => 0, 'position' => 105, 'searchall' => 1),
245 'note_private' => array('type' => 'html', 'label' => 'NotePrivate', 'enabled' => 1, 'visible' => 0, 'position' => 110, 'searchall' => 1),
246 'model_pdf' => array('type' => 'varchar(255)', 'label' => 'Model pdf', 'enabled' => 1, 'visible' => 0, 'position' => 115),
247 'import_key' => array('type' => 'varchar(14)', 'label' => 'ImportId', 'enabled' => 1, 'visible' => -2, 'position' => 120),
248 'extraparams' => array('type' => 'varchar(255)', 'label' => 'Extraparams', 'enabled' => 1, 'visible' => -1, 'position' => 125),
249 'fk_user_modif' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserModif', 'enabled' => 1, 'visible' => -2, 'notnull' => -1, 'position' => 135),
250 'last_main_doc' => array('type' => 'varchar(255)', 'label' => 'Last main doc', 'enabled' => 1, 'visible' => -1, 'position' => 140),
251 'statut' => array('type' => 'smallint(6)', 'label' => 'Statut', 'enabled' => 1, 'visible' => -1, 'position' => 500, 'notnull' => 1, 'arrayofkeyval' => array(0 => 'Draft', 1 => 'Validated', 2 => 'Closed'))
252 );
253 // END MODULEBUILDER PROPERTIES
254
255 const STATUS_DRAFT = 0;
256 const STATUS_VALIDATED = 1;
257 const STATUS_CLOSED = 2;
258
259 /*
260 * No signature
261 */
262 const STATUS_NO_SIGNATURE = 0;
263
264 /*
265 * Signed by sender
266 */
267 const STATUS_SIGNED_SENDER = 1;
268
269 /*
270 * Signed by receiver
271 */
272 const STATUS_SIGNED_RECEIVER = 2;
273
274 /*
275 * Signed by all
276 */
277 const STATUS_SIGNED_ALL = 9; // To handle future kind of signature (ex: tripartite contract)
278
279
285 public function __construct($db)
286 {
287 $this->db = $db;
288
289 $this->ismultientitymanaged = 1;
290 $this->isextrafieldmanaged = 1;
291 }
292
299 public function getNextNumRef($soc)
300 {
301 global $db, $langs, $conf;
302 $langs->load("contracts");
303
304 if (getDolGlobalString('CONTRACT_ADDON')) {
305 $mybool = false;
306
307 $file = getDolGlobalString('CONTRACT_ADDON') . ".php";
308 $classname = getDolGlobalString('CONTRACT_ADDON');
309
310 // Include file with class
311 $dirmodels = array_merge(array('/'), (array) $conf->modules_parts['models']);
312
313 foreach ($dirmodels as $reldir) {
314 $dir = dol_buildpath($reldir."core/modules/contract/");
315
316 // Load file with numbering class (if found)
317 $mybool = ((bool) @include_once $dir.$file) || $mybool;
318 }
319
320 if (!$mybool) {
321 dol_print_error(null, "Failed to include file ".$file);
322 return '';
323 }
324
325 $obj = new $classname();
326 '@phan-var-force CommonNumRefGenerator $obj';
327 $numref = $obj->getNextValue($soc, $this);
328
329 if ($numref != "") {
330 return $numref;
331 } else {
332 $this->error = $obj->error;
333 dol_print_error($db, get_class($this)."::getNextValue ".$obj->error);
334 return "";
335 }
336 } else {
337 $langs->load("errors");
338 print $langs->trans("Error")." ".$langs->trans("ErrorModuleSetupNotComplete", $langs->transnoentitiesnoconv("Contract"));
339 return "";
340 }
341 }
342
343 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
354 public function active_line($user, $line_id, $date_start, $date_end = '', $comment = '')
355 {
356 // phpcs:enable
357 $result = $this->lines[$this->lines_id_index_mapper[$line_id]]->active_line($user, $date_start, $date_end, $comment);
358 if ($result < 0) {
359 $this->error = $this->lines[$this->lines_id_index_mapper[$line_id]]->error;
360 $this->errors = $this->lines[$this->lines_id_index_mapper[$line_id]]->errors;
361 }
362 return $result;
363 }
364
365
366 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
376 public function close_line($user, $line_id, $date_end, $comment = '')
377 {
378 // phpcs:enable
379 $result = $this->lines[$this->lines_id_index_mapper[$line_id]]->close_line($user, $date_end, $comment);
380 if ($result < 0) {
381 $this->error = $this->lines[$this->lines_id_index_mapper[$line_id]]->error;
382 $this->errors = $this->lines[$this->lines_id_index_mapper[$line_id]]->errors;
383 }
384 return $result;
385 }
386
387
399 public function activateAll($user, $date_start = '', $notrigger = 0, $comment = '', $date_end = '')
400 {
401 if (empty($date_start)) {
402 $date_start = dol_now();
403 }
404
405 $this->db->begin();
406
407 $error = 0;
408
409 // Load lines
410 $this->fetch_lines();
411
412 foreach ($this->lines as $contratline) {
413 // Open lines not already open
414 if ($contratline->statut != ContratLigne::STATUS_OPEN) {
415 $contratline->context = $this->context;
416
417 $result = $contratline->active_line($user, $date_start, !empty($date_end) ? $date_end : -1, $comment); // This call trigger LINECONTRACT_ACTIVATE
418 if ($result < 0) {
419 $error++;
420 $this->error = $contratline->error;
421 $this->errors = $contratline->errors;
422 break;
423 }
424 }
425 }
426
427 if (!$error && $this->statut == 0) {
428 $result = $this->validate($user, '', $notrigger);
429 if ($result < 0) {
430 $error++;
431 }
432 }
433
434 if (!$error) {
435 $this->db->commit();
436 return 1;
437 } else {
438 $this->db->rollback();
439 return -1;
440 }
441 }
442
452 public function closeAll(User $user, $notrigger = 0, $comment = '')
453 {
454 $this->db->begin();
455
456 // Load lines
457 $this->fetch_lines();
458
459 $now = dol_now();
460
461 $error = 0;
462
463 foreach ($this->lines as $contratline) {
464 // Close lines not already closed
465 if ($contratline->statut != ContratLigne::STATUS_CLOSED) {
466 $contratline->date_end_real = $now;
467 $contratline->date_cloture = $now; // For backward compatibility
468 $contratline->user_closing_id = $user->id;
469 $contratline->statut = ContratLigne::STATUS_CLOSED;
470 $result = $contratline->close_line($user, $now, $comment, $notrigger);
471 if ($result < 0) {
472 $error++;
473 $this->error = $contratline->error;
474 $this->errors = $contratline->errors;
475 break;
476 }
477 }
478 }
479
480 if (!$error && $this->statut == 0) {
481 $result = $this->validate($user, '', $notrigger);
482 if ($result < 0) {
483 $error++;
484 }
485 }
486
487 if (!$error) {
488 $this->db->commit();
489 return 1;
490 } else {
491 $this->db->rollback();
492 return -1;
493 }
494 }
495
504 public function validate(User $user, $force_number = '', $notrigger = 0)
505 {
506 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
507 global $conf;
508
509 $now = dol_now();
510
511 $error = 0;
512 dol_syslog(get_class($this).'::validate user='.$user->id.', force_number='.$force_number);
513
514
515 $this->db->begin();
516
517 $this->fetch_thirdparty();
518
519 // A contract is validated so we can move thirdparty to status customer
520 if (!getDolGlobalString('CONTRACT_DISABLE_AUTOSET_AS_CLIENT_ON_CONTRACT_VALIDATION') && $this->thirdparty->fournisseur == 0) {
521 $result = $this->thirdparty->setAsCustomer();
522 }
523
524 // Define new ref
525 if ($force_number) {
526 $num = $force_number;
527 } elseif (!$error && (preg_match('/^[\‍(]?PROV/i', $this->ref) || empty($this->ref))) { // empty should not happened, but when it occurs, the test save life
528 $num = $this->getNextNumRef($this->thirdparty);
529 } else {
530 $num = $this->ref;
531 }
532 $this->newref = dol_sanitizeFileName($num);
533
534 if ($num) {
535 $sql = "UPDATE ".MAIN_DB_PREFIX."contrat SET ref = '".$this->db->escape($num)."', statut = 1";
536 //$sql.= ", fk_user_valid = ".$user->id.", date_valid = '".$this->db->idate($now)."'";
537 $sql .= " WHERE rowid = ".((int) $this->id)." AND statut = 0";
538
539 dol_syslog(get_class($this)."::validate", LOG_DEBUG);
540 $resql = $this->db->query($sql);
541 if (!$resql) {
542 dol_print_error($this->db);
543 $error++;
544 $this->error = $this->db->lasterror();
545 }
546
547 // Trigger calls
548 if (!$error && !$notrigger) {
549 // Call trigger
550 $result = $this->call_trigger('CONTRACT_VALIDATE', $user);
551 if ($result < 0) {
552 $error++;
553 }
554 // End call triggers
555 }
556
557 if (!$error) {
558 $this->oldref = $this->ref;
559
560 // Rename directory if dir was a temporary ref
561 if (preg_match('/^[\‍(]?PROV/i', $this->ref)) {
562 // Now we rename also files into index
563 $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filename = CONCAT('".$this->db->escape($this->newref)."', SUBSTR(filename, ".(strlen($this->ref) + 1).")), filepath = 'contract/".$this->db->escape($this->newref)."'";
564 $sql .= " WHERE filename LIKE '".$this->db->escape($this->ref)."%' AND filepath = 'contract/".$this->db->escape($this->ref)."' and entity = ".$conf->entity;
565 $resql = $this->db->query($sql);
566 if (!$resql) {
567 $error++;
568 $this->error = $this->db->lasterror();
569 }
570 $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filepath = 'contract/".$this->db->escape($this->newref)."'";
571 $sql .= " WHERE filepath = 'contract/".$this->db->escape($this->ref)."' and entity = ".$conf->entity;
572 $resql = $this->db->query($sql);
573 if (!$resql) {
574 $error++;
575 $this->error = $this->db->lasterror();
576 }
577
578 // We rename directory ($this->ref = old ref, $num = new ref) in order not to lose the attachments
579 $oldref = dol_sanitizeFileName($this->ref);
580 $newref = dol_sanitizeFileName($num);
581 $dirsource = $conf->contract->dir_output.'/'.$oldref;
582 $dirdest = $conf->contract->dir_output.'/'.$newref;
583 if (!$error && file_exists($dirsource)) {
584 dol_syslog(get_class($this)."::validate rename dir ".$dirsource." into ".$dirdest);
585
586 if (@rename($dirsource, $dirdest)) {
587 dol_syslog("Rename ok");
588 // Rename docs starting with $oldref with $newref
589 $listoffiles = dol_dir_list($conf->contract->dir_output.'/'.$newref, 'files', 1, '^'.preg_quote($oldref, '/'));
590 foreach ($listoffiles as $fileentry) {
591 $dirsource = $fileentry['name'];
592 $dirdest = preg_replace('/^'.preg_quote($oldref, '/').'/', $newref, $dirsource);
593 $dirsource = $fileentry['path'].'/'.$dirsource;
594 $dirdest = $fileentry['path'].'/'.$dirdest;
595 @rename($dirsource, $dirdest);
596 }
597 }
598 }
599 }
600 }
601
602 // Set new ref and define current statut
603 if (!$error) {
604 $this->ref = $num;
605 $this->status = self::STATUS_VALIDATED;
606 $this->statut = self::STATUS_VALIDATED; // deprecated
607 $this->date_validation = $now;
608 }
609 } else {
610 $error++;
611 }
612
613 if (!$error) {
614 $this->db->commit();
615 return 1;
616 } else {
617 $this->db->rollback();
618 return -1;
619 }
620 }
621
629 public function reopen($user, $notrigger = 0)
630 {
631 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
632
633 $now = dol_now();
634
635 $error = 0;
636 dol_syslog(get_class($this).'::reopen user='.$user->id);
637
638 $this->db->begin();
639
640 $this->fetch_thirdparty();
641
642 $sql = "UPDATE ".MAIN_DB_PREFIX."contrat SET statut = 0";
643 //$sql.= ", fk_user_valid = null, date_valid = null";
644 $sql .= " WHERE rowid = ".((int) $this->id)." AND statut = 1";
645
646 dol_syslog(get_class($this)."::validate", LOG_DEBUG);
647 $resql = $this->db->query($sql);
648 if (!$resql) {
649 dol_print_error($this->db);
650 $error++;
651 $this->error = $this->db->lasterror();
652 }
653
654 // Trigger calls
655 if (!$error && !$notrigger) {
656 // Call trigger
657 $result = $this->call_trigger('CONTRACT_REOPEN', $user);
658 if ($result < 0) {
659 $error++;
660 }
661 // End call triggers
662 }
663
664 // Set new ref and define current status
665 if (!$error) {
666 $this->statut = self::STATUS_DRAFT;
667 $this->status = self::STATUS_DRAFT;
668 $this->date_validation = $now;
669 }
670
671 if (!$error) {
672 $this->db->commit();
673 return 1;
674 } else {
675 $this->db->rollback();
676 return -1;
677 }
678 }
679
691 public function fetch($id, $ref = '', $ref_customer = '', $ref_supplier = '', $noextrafields = 0, $nolines = 0)
692 {
693 $sql = "SELECT rowid, statut as status, ref, fk_soc as thirdpartyid,";
694 $sql .= " ref_supplier, ref_customer,";
695 $sql .= " ref_ext,";
696 $sql .= " entity,";
697 $sql .= " date_contrat as datecontrat,";
698 $sql .= " fk_user_author,";
699 $sql .= " fk_projet as fk_project,";
700 $sql .= " fk_commercial_signature, fk_commercial_suivi,";
701 $sql .= " note_private, note_public, model_pdf, last_main_doc, extraparams";
702 $sql .= " FROM ".MAIN_DB_PREFIX."contrat";
703 if (!$id) {
704 $sql .= " WHERE entity IN (".getEntity('contract').")";
705 } else {
706 $sql .= " WHERE rowid = ".(int) $id;
707 }
708 if ($ref_customer) {
709 $sql .= " AND ref_customer = '".$this->db->escape($ref_customer)."'";
710 }
711 if ($ref_supplier) {
712 $sql .= " AND ref_supplier = '".$this->db->escape($ref_supplier)."'";
713 }
714 if ($ref) {
715 $sql .= " AND ref = '".$this->db->escape($ref)."'";
716 }
717
718 dol_syslog(get_class($this)."::fetch", LOG_DEBUG);
719 $resql = $this->db->query($sql);
720 if ($resql) {
721 $num = $this->db->num_rows($resql);
722 if ($num > 1) {
723 $this->error = 'Fetch found several records.';
724 dol_syslog($this->error, LOG_ERR);
725 $result = -2;
726 return 0;
727 } elseif ($num) { // $num = 1
728 $obj = $this->db->fetch_object($resql);
729 if ($obj) {
730 $this->id = $obj->rowid;
731 $this->ref = (!isset($obj->ref) || !$obj->ref) ? $obj->rowid : $obj->ref;
732 $this->ref_customer = $obj->ref_customer;
733 $this->ref_supplier = $obj->ref_supplier;
734 $this->ref_ext = $obj->ref_ext;
735 $this->entity = $obj->entity;
736 $this->statut = $obj->status;
737 $this->status = $obj->status;
738
739 $this->date_contrat = $this->db->jdate($obj->datecontrat);
740 $this->date_creation = $this->db->jdate($obj->datecontrat);
741
742 $this->user_author_id = $obj->fk_user_author;
743
744 $this->commercial_signature_id = $obj->fk_commercial_signature;
745 $this->commercial_suivi_id = $obj->fk_commercial_suivi;
746
747 $this->note_private = $obj->note_private;
748 $this->note_public = $obj->note_public;
749 $this->model_pdf = $obj->model_pdf;
750
751 $this->fk_projet = $obj->fk_project; // deprecated
752 $this->fk_project = $obj->fk_project;
753
754 $this->socid = $obj->thirdpartyid;
755 $this->fk_soc = $obj->thirdpartyid;
756 $this->last_main_doc = $obj->last_main_doc;
757 $this->extraparams = (isset($obj->extraparams) ? (array) json_decode($obj->extraparams, true) : null);
758
759 $this->db->free($resql);
760
761 // Retrieve all extrafields
762 // fetch optionals attributes and labels
763 if (empty($noextrafields)) {
764 $result = $this->fetch_optionals();
765 if ($result < 0) {
766 $this->error = $this->db->lasterror();
767 return -4;
768 }
769 }
770
771 // Lines
772 if (empty($nolines)) {
773 if ($result >= 0 && !empty($this->table_element_line)) {
774 $result = $this->fetch_lines();
775 if ($result < 0) {
776 $this->error = $this->db->lasterror();
777 return -3;
778 }
779 }
780 }
781
782 return $this->id;
783 } else {
784 dol_syslog(get_class($this)."::fetch Contract failed");
785 $this->error = "Fetch contract failed";
786 return -1;
787 }
788 } else {
789 dol_syslog(get_class($this)."::fetch Contract not found");
790 $this->error = "Contract not found";
791 return 0;
792 }
793 } else {
794 dol_syslog(get_class($this)."::fetch Error searching contract");
795 $this->error = $this->db->error();
796 return -1;
797 }
798 }
799
800 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
810 public function fetch_lines($only_services = 0, $loadalsotranslation = 0, $noextrafields = 0)
811 {
812 // phpcs:enable
813 $this->nbofservices = 0;
814 $this->nbofserviceswait = 0;
815 $this->nbofservicesopened = 0;
816 $this->nbofservicesexpired = 0;
817 $this->nbofservicesclosed = 0;
818
819 $total_ttc = 0;
820 $total_vat = 0;
821 $total_ht = 0;
822
823 $now = dol_now();
824
825 $this->lines = array();
826 $pos = 0;
827
828 // Selects contract lines related to a product
829 $sql = "SELECT p.label as product_label, p.description as product_desc, p.ref as product_ref, p.fk_product_type as product_type,";
830 $sql .= " d.rowid, d.fk_contrat, d.statut as status, d.description, d.price_ht, d.vat_src_code, d.tva_tx, d.localtax1_tx, d.localtax2_tx, d.localtax1_type, d.localtax2_type, d.qty, d.remise_percent, d.subprice, d.fk_product_fournisseur_price as fk_fournprice, d.buy_price_ht as pa_ht,";
831 $sql .= " d.total_ht,";
832 $sql .= " d.total_tva,";
833 $sql .= " d.total_localtax1,";
834 $sql .= " d.total_localtax2,";
835 $sql .= " d.total_ttc,";
836 $sql .= " d.info_bits, d.fk_product,";
837 $sql .= " d.date_ouverture_prevue as date_start,";
838 $sql .= " d.date_ouverture as date_start_real,";
839 $sql .= " d.date_fin_validite as date_end,";
840 $sql .= " d.date_cloture as date_end_real,";
841 $sql .= " d.fk_user_author,";
842 $sql .= " d.fk_user_ouverture,";
843 $sql .= " d.fk_user_cloture,";
844 $sql .= " d.fk_unit,";
845 $sql .= " d.product_type as type,";
846 $sql .= " d.rang";
847 $sql .= " FROM ".MAIN_DB_PREFIX."contratdet as d LEFT JOIN ".MAIN_DB_PREFIX."product as p ON d.fk_product = p.rowid";
848 $sql .= " WHERE d.fk_contrat = ".((int) $this->id);
849 if ($only_services == 1) {
850 $sql .= " AND d.product_type = 1";
851 }
852 $sql .= " ORDER by d.rang ASC";
853
854 dol_syslog(get_class($this)."::fetch_lines", LOG_DEBUG);
855 $result = $this->db->query($sql);
856 if ($result) {
857 $num = $this->db->num_rows($result);
858 $i = 0;
859
860 while ($i < $num) {
861 $objp = $this->db->fetch_object($result);
862
863 $line = new ContratLigne($this->db);
864
865 $line->id = $objp->rowid;
866 $line->ref = $objp->rowid;
867 $line->fk_contrat = $objp->fk_contrat;
868 $line->desc = $objp->description; // Description line
869 $line->qty = $objp->qty;
870 $line->vat_src_code = $objp->vat_src_code;
871 $line->tva_tx = $objp->tva_tx;
872 $line->localtax1_tx = $objp->localtax1_tx;
873 $line->localtax2_tx = $objp->localtax2_tx;
874 $line->localtax1_type = $objp->localtax1_type;
875 $line->localtax2_type = $objp->localtax2_type;
876 $line->subprice = $objp->subprice;
877 $line->statut = $objp->status;
878 $line->status = $objp->status;
879 $line->remise_percent = $objp->remise_percent;
880 $line->price_ht = $objp->price_ht;
881 $line->price = $objp->price_ht; // For backward compatibility
882 $line->total_ht = $objp->total_ht;
883 $line->total_tva = $objp->total_tva;
884 $line->total_localtax1 = $objp->total_localtax1;
885 $line->total_localtax2 = $objp->total_localtax2;
886 $line->total_ttc = $objp->total_ttc;
887 $line->fk_product = (($objp->fk_product > 0) ? $objp->fk_product : 0);
888 $line->info_bits = $objp->info_bits;
889 $line->type = $objp->type;
890
891 $line->fk_fournprice = $objp->fk_fournprice;
892 $marginInfos = getMarginInfos($objp->subprice, $objp->remise_percent, $objp->tva_tx, $objp->localtax1_tx, $objp->localtax2_tx, $objp->fk_fournprice, $objp->pa_ht);
893 $line->pa_ht = $marginInfos[0];
894
895 $line->fk_user_author = $objp->fk_user_author;
896 $line->fk_user_ouverture = $objp->fk_user_ouverture;
897 $line->fk_user_cloture = $objp->fk_user_cloture;
898 $line->fk_unit = $objp->fk_unit;
899
900 $line->ref = $objp->product_ref; // deprecated
901 $line->product_ref = $objp->product_ref; // Product Ref
902 $line->product_type = $objp->product_type; // Product Type
903 $line->product_desc = $objp->product_desc; // Product Description
904 $line->product_label = $objp->product_label; // Product Label
905
906 $line->description = $objp->description;
907
908 $line->date_start = $this->db->jdate($objp->date_start);
909 $line->date_start_real = $this->db->jdate($objp->date_start_real);
910 $line->date_end = $this->db->jdate($objp->date_end);
911 $line->date_end_real = $this->db->jdate($objp->date_end_real);
912 // For backward compatibility
913 //$line->date_ouverture_prevue = $this->db->jdate($objp->date_ouverture_prevue);
914 //$line->date_ouverture = $this->db->jdate($objp->date_ouverture);
915 //$line->date_fin_validite = $this->db->jdate($objp->date_fin_validite);
916 //$line->date_cloture = $this->db->jdate($objp->date_cloture);
917 //$line->date_debut_prevue = $this->db->jdate($objp->date_ouverture_prevue);
918 //$line->date_debut_reel = $this->db->jdate($objp->date_ouverture);
919 //$line->date_fin_prevue = $this->db->jdate($objp->date_fin_validite);
920 //$line->date_fin_reel = $this->db->jdate($objp->date_cloture);
921
922 $line->rang = $objp->rang;
923
924 // Retrieve all extrafields for contract line
925 // fetch optionals attributes and labels
926 if (empty($noextrafields)) {
927 $line->fetch_optionals();
928 }
929
930 // multilangs
931 if (getDolGlobalInt('MAIN_MULTILANGS') && !empty($objp->fk_product) && !empty($loadalsotranslation)) {
932 $tmpproduct = new Product($this->db);
933 $tmpproduct->fetch($objp->fk_product);
934 $tmpproduct->getMultiLangs();
935
936 $line->multilangs = $tmpproduct->multilangs;
937 }
938
939 $this->lines[$pos] = $line;
940
941 $this->lines_id_index_mapper[$line->id] = $pos;
942
943 //dol_syslog("1 ".$line->desc);
944 //dol_syslog("2 ".$line->product_desc);
945
946 if ($line->statut == ContratLigne::STATUS_INITIAL) {
947 $this->nbofserviceswait++;
948 }
949 if ($line->statut == ContratLigne::STATUS_OPEN && (empty($line->date_end) || $line->date_end >= $now)) {
950 $this->nbofservicesopened++;
951 }
952 if ($line->statut == ContratLigne::STATUS_OPEN && (!empty($line->date_end) && $line->date_end < $now)) {
953 $this->nbofservicesexpired++;
954 }
955 if ($line->statut == ContratLigne::STATUS_CLOSED) {
956 $this->nbofservicesclosed++;
957 }
958
959 $total_ttc += $objp->total_ttc; // TODO Not saved into database
960 $total_vat += $objp->total_tva;
961 $total_ht += $objp->total_ht;
962
963 $i++;
964 $pos++;
965 }
966 $this->db->free($result);
967 } else {
968 dol_syslog(get_class($this)."::Fetch Error when reading lines of contracts linked to products");
969 return -3;
970 }
971
972 // Now set the global properties on contract not stored into database.
973 $this->nbofservices = count($this->lines);
974 $this->total_ttc = (float) price2num($total_ttc);
975 $this->total_tva = (float) price2num($total_vat);
976 $this->total_ht = (float) price2num($total_ht);
977
978 return $this->lines;
979 }
980
987 public function create($user)
988 {
989 global $conf, $langs, $mysoc;
990
991 // Check parameters
992 $paramsok = 1;
993 if ($this->commercial_signature_id <= 0) {
994 $langs->load("commercial");
995 $this->error .= $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("SalesRepresentativeSignature"));
996 $paramsok = 0;
997 }
998 if ($this->commercial_suivi_id <= 0) {
999 $langs->load("commercial");
1000 $this->error .= ($this->error ? "<br>" : '');
1001 $this->error .= $langs->trans("ErrorFieldRequired", $langs->transnoentitiesnoconv("SalesRepresentativeFollowUp"));
1002 $paramsok = 0;
1003 }
1004 if (!$paramsok) {
1005 return -1;
1006 }
1007
1008
1009 $this->db->begin();
1010
1011 $now = dol_now();
1012
1013 // Insert contract
1014 $sql = "INSERT INTO ".MAIN_DB_PREFIX."contrat (datec, fk_soc, fk_user_author, date_contrat,";
1015 $sql .= " fk_commercial_signature, fk_commercial_suivi, fk_projet,";
1016 $sql .= " ref, entity, note_private, note_public, ref_customer, ref_supplier, ref_ext)";
1017 $sql .= " VALUES ('".$this->db->idate($now)."', ".((int) $this->socid).", ".((int) $user->id);
1018 $sql .= ", ".(dol_strlen($this->date_contrat) != 0 ? "'".$this->db->idate($this->date_contrat)."'" : "NULL");
1019 $sql .= ",".($this->commercial_signature_id > 0 ? ((int) $this->commercial_signature_id) : "NULL");
1020 $sql .= ",".($this->commercial_suivi_id > 0 ? ((int) $this->commercial_suivi_id) : "NULL");
1021 $sql .= ",".($this->fk_project > 0 ? ((int) $this->fk_project) : "NULL");
1022 $sql .= ", ".(dol_strlen($this->ref) <= 0 ? "null" : "'".$this->db->escape($this->ref)."'");
1023 $sql .= ", ".((int) $conf->entity);
1024 $sql .= ", ".(!empty($this->note_private) ? ("'".$this->db->escape($this->note_private)."'") : "NULL");
1025 $sql .= ", ".(!empty($this->note_public) ? ("'".$this->db->escape($this->note_public)."'") : "NULL");
1026 $sql .= ", ".(!empty($this->ref_customer) ? ("'".$this->db->escape($this->ref_customer)."'") : "NULL");
1027 $sql .= ", ".(!empty($this->ref_supplier) ? ("'".$this->db->escape($this->ref_supplier)."'") : "NULL");
1028 $sql .= ", ".(!empty($this->ref_ext) ? ("'".$this->db->escape($this->ref_ext)."'") : "NULL");
1029 $sql .= ")";
1030 $resql = $this->db->query($sql);
1031
1032 if ($resql) {
1033 $error = 0;
1034
1035 $this->id = $this->db->last_insert_id(MAIN_DB_PREFIX."contrat");
1036
1037 // Load object modContract
1038 $module = (getDolGlobalString('CONTRACT_ADDON') ? $conf->global->CONTRACT_ADDON : 'mod_contract_serpis');
1039 if (substr($module, 0, 13) == 'mod_contract_' && substr($module, -3) == 'php') {
1040 $module = substr($module, 0, dol_strlen($module) - 4);
1041 }
1042 $result = dol_include_once('/core/modules/contract/'.$module.'.php');
1043 if ($result > 0) {
1044 $modCodeContract = new $module();
1045 '@phan-var-force CommonNumRefGenerator $modCodeContrat';
1046
1047 if (!empty($modCodeContract->code_auto)) {
1048 // Force the ref to a draft value if numbering module is an automatic numbering
1049 $sql = 'UPDATE '.MAIN_DB_PREFIX."contrat SET ref='(PROV".$this->id.")' WHERE rowid=".((int) $this->id);
1050 if ($this->db->query($sql)) {
1051 if ($this->id) {
1052 $this->ref = "(PROV".$this->id.")";
1053 }
1054 }
1055 }
1056 }
1057
1058 if (!$error) {
1059 $result = $this->insertExtraFields();
1060 if ($result < 0) {
1061 $error++;
1062 }
1063 }
1064
1065 // Insert business contacts ('SALESREPSIGN','contrat')
1066 if (!$error) {
1067 $result = $this->add_contact($this->commercial_signature_id, 'SALESREPSIGN', 'internal');
1068 if ($result < 0) {
1069 $error++;
1070 }
1071 }
1072
1073 // Insert business contacts ('SALESREPFOLL','contrat')
1074 if (!$error) {
1075 $result = $this->add_contact($this->commercial_suivi_id, 'SALESREPFOLL', 'internal');
1076 if ($result < 0) {
1077 $error++;
1078 }
1079 }
1080
1081 if (!$error) {
1082 if (!empty($this->linkedObjectsIds) && empty($this->linked_objects)) { // To use new linkedObjectsIds instead of old linked_objects
1083 $this->linked_objects = $this->linkedObjectsIds; // TODO Replace linked_objects with linkedObjectsIds
1084 }
1085
1086 // Add object linked
1087 if (!$error && $this->id && !empty($this->linked_objects) && is_array($this->linked_objects)) {
1088 foreach ($this->linked_objects as $origin => $tmp_origin_id) {
1089 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, ...))
1090 foreach ($tmp_origin_id as $origin_id) {
1091 $ret = $this->add_object_linked($origin, $origin_id);
1092 if (!$ret) {
1093 $this->error = $this->db->lasterror();
1094 $error++;
1095 }
1096 }
1097 } else { // Old behaviour, if linked_object has only one link per type, so is something like array('contract'=>id1))
1098 $origin_id = $tmp_origin_id;
1099 $ret = $this->add_object_linked($origin, $origin_id);
1100 if (!$ret) {
1101 $this->error = $this->db->lasterror();
1102 $error++;
1103 }
1104 }
1105 }
1106 }
1107
1108 if (!$error && $this->id && getDolGlobalString('MAIN_PROPAGATE_CONTACTS_FROM_ORIGIN') && !empty($this->origin) && !empty($this->origin_id)) { // Get contact from origin object
1109 $originforcontact = $this->origin;
1110 $originidforcontact = $this->origin_id;
1111 if ($originforcontact == 'shipping') { // shipment and order share the same contacts. If creating from shipment we take data of order
1112 require_once DOL_DOCUMENT_ROOT.'/expedition/class/expedition.class.php';
1113 $exp = new Expedition($this->db);
1114 $exp->fetch($this->origin_id);
1115 $exp->fetchObjectLinked();
1116 if (count($exp->linkedObjectsIds['commande']) > 0) {
1117 foreach ($exp->linkedObjectsIds['commande'] as $key => $value) {
1118 $originforcontact = 'commande';
1119 $originidforcontact = $value;
1120 break; // We take first one
1121 }
1122 }
1123 }
1124
1125 $sqlcontact = "SELECT ctc.code, ctc.source, ec.fk_socpeople FROM ".MAIN_DB_PREFIX."element_contact as ec, ".MAIN_DB_PREFIX."c_type_contact as ctc";
1126 $sqlcontact .= " WHERE element_id = ".((int) $originidforcontact)." AND ec.fk_c_type_contact = ctc.rowid AND ctc.element = '".$this->db->escape($originforcontact)."'";
1127
1128 $resqlcontact = $this->db->query($sqlcontact);
1129 if ($resqlcontact) {
1130 while ($objcontact = $this->db->fetch_object($resqlcontact)) {
1131 if ($objcontact->source == 'internal' && in_array($objcontact->code, array('SALESREPSIGN', 'SALESREPFOLL'))) {
1132 continue; // ignore this, already forced previously
1133 }
1134
1135 $this->add_contact($objcontact->fk_socpeople, $objcontact->code, $objcontact->source); // May failed because of duplicate key or because code of contact type does not exists for new object
1136 }
1137 } else {
1138 dol_print_error($this->db, $resqlcontact);
1139 }
1140 }
1141 }
1142
1143 if (!$error) {
1144 // Call trigger
1145 $result = $this->call_trigger('CONTRACT_CREATE', $user);
1146 if ($result < 0) {
1147 $error++;
1148 }
1149 // End call triggers
1150
1151 if (!$error) {
1152 $this->db->commit();
1153 return $this->id;
1154 } else {
1155 dol_syslog(get_class($this)."::create - 30 - ".$this->error, LOG_ERR);
1156 $this->db->rollback();
1157 return -3;
1158 }
1159 } else {
1160 $this->error = "Failed to add contract";
1161 dol_syslog(get_class($this)."::create - 20 - ".$this->error, LOG_ERR);
1162 $this->db->rollback();
1163 return -2;
1164 }
1165 } else {
1166 $this->error = $langs->trans("UnknownError").": ".$this->db->error();
1167 dol_syslog(get_class($this)."::create - 10 - ".$this->error, LOG_ERR);
1168
1169 $this->db->rollback();
1170 return -1;
1171 }
1172 }
1173
1174
1181 public function delete($user)
1182 {
1183 global $conf;
1184
1185 require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
1186
1187 $error = 0;
1188
1189 $this->db->begin();
1190
1191 // Call trigger
1192 $result = $this->call_trigger('CONTRACT_DELETE', $user);
1193 if ($result < 0) {
1194 $error++;
1195 }
1196 // End call triggers
1197
1198 if (!$error) {
1199 // Delete linked contacts
1200 $res = $this->delete_linked_contact();
1201 if ($res < 0) {
1202 dol_syslog(get_class($this)."::delete error", LOG_ERR);
1203 $error++;
1204 }
1205 }
1206
1207 if (!$error) {
1208 // Delete linked object
1209 $res = $this->deleteObjectLinked();
1210 if ($res < 0) {
1211 $error++;
1212 }
1213 }
1214
1215 // Delete lines
1216 if (!$error) {
1217 // Delete contratdet extrafields
1218 $main = MAIN_DB_PREFIX.'contratdet';
1219 $ef = $main."_extrafields";
1220 $sql = "DELETE FROM ".$ef." WHERE fk_object IN (SELECT rowid FROM ".$main." WHERE fk_contrat = ".((int) $this->id).")";
1221
1222 dol_syslog(get_class($this)."::delete contratdet_extrafields", LOG_DEBUG);
1223 $resql = $this->db->query($sql);
1224 if (!$resql) {
1225 $this->error = $this->db->error();
1226 $error++;
1227 }
1228 }
1229
1230 if (!$error) {
1231 // Delete contratdet
1232 $sql = "DELETE FROM ".MAIN_DB_PREFIX."contratdet";
1233 $sql .= " WHERE fk_contrat=".((int) $this->id);
1234
1235 dol_syslog(get_class($this)."::delete contratdet", LOG_DEBUG);
1236 $resql = $this->db->query($sql);
1237 if (!$resql) {
1238 $this->error = $this->db->error();
1239 $error++;
1240 }
1241 }
1242
1243 // Delete llx_ecm_files
1244 if (!$error) {
1245 $sql = 'DELETE FROM '.MAIN_DB_PREFIX."ecm_files WHERE src_object_type = '".$this->db->escape($this->table_element.(empty($this->module) ? "" : "@".$this->module))."' AND src_object_id = ".((int) $this->id);
1246 $resql = $this->db->query($sql);
1247 if (!$resql) {
1248 $this->error = $this->db->lasterror();
1249 $this->errors[] = $this->error;
1250 $error++;
1251 }
1252 }
1253
1254 // Delete contract
1255 if (!$error) {
1256 $sql = "DELETE FROM ".MAIN_DB_PREFIX."contrat";
1257 $sql .= " WHERE rowid=".((int) $this->id);
1258
1259 dol_syslog(get_class($this)."::delete contrat", LOG_DEBUG);
1260 $resql = $this->db->query($sql);
1261 if (!$resql) {
1262 $this->error = $this->db->error();
1263 $error++;
1264 }
1265 }
1266
1267 // Removed extrafields
1268 if (!$error) {
1269 $result = $this->deleteExtraFields();
1270 if ($result < 0) {
1271 $error++;
1272 dol_syslog(get_class($this)."::delete error -3 ".$this->error, LOG_ERR);
1273 }
1274 }
1275
1276 if (!$error) {
1277 // We remove directory
1278 $ref = dol_sanitizeFileName($this->ref);
1279 if ($conf->contrat->dir_output) {
1280 $dir = $conf->contrat->multidir_output[$this->entity]."/".$ref;
1281 if (file_exists($dir)) {
1282 $res = @dol_delete_dir_recursive($dir);
1283 if (!$res) {
1284 $this->error = 'ErrorFailToDeleteDir';
1285 $error++;
1286 }
1287 }
1288 }
1289 }
1290
1291 if (!$error) {
1292 $this->db->commit();
1293 return 1;
1294 } else {
1295 $this->error = $this->db->lasterror();
1296 $this->db->rollback();
1297 return -1;
1298 }
1299 }
1300
1308 public function update($user, $notrigger = 0)
1309 {
1310 global $conf;
1311 $error = 0;
1312
1313 // Clean parameters
1314 if (empty($this->fk_commercial_signature) && $this->commercial_signature_id > 0) {
1315 $this->fk_commercial_signature = $this->commercial_signature_id;
1316 }
1317 if (empty($this->fk_commercial_suivi) && $this->commercial_suivi_id > 0) {
1318 $this->fk_commercial_suivi = $this->commercial_suivi_id;
1319 }
1320 if (empty($this->socid) && $this->fk_soc > 0) {
1321 $this->socid = (int) $this->fk_soc;
1322 }
1323 if (empty($this->fk_project) && $this->projet > 0) {
1324 $this->fk_project = (int) $this->projet;
1325 }
1326
1327 if (isset($this->ref)) {
1328 $this->ref = trim($this->ref);
1329 }
1330 if (isset($this->ref_customer)) {
1331 $this->ref_customer = trim($this->ref_customer);
1332 }
1333 if (isset($this->ref_supplier)) {
1334 $this->ref_supplier = trim($this->ref_supplier);
1335 }
1336 if (isset($this->ref_ext)) {
1337 $this->ref_ext = trim($this->ref_ext);
1338 }
1339 if (isset($this->entity)) {
1340 $this->entity = (int) $this->entity;
1341 }
1342 if (isset($this->statut)) {
1343 $this->statut = (int) $this->statut;
1344 }
1345 if (isset($this->status)) {
1346 $this->status = (int) $this->status;
1347 }
1348 if (isset($this->socid)) {
1349 $this->socid = (int) $this->socid;
1350 }
1351 if (isset($this->fk_commercial_signature)) {
1352 $this->fk_commercial_signature = trim($this->fk_commercial_signature);
1353 }
1354 if (isset($this->fk_commercial_suivi)) {
1355 $this->fk_commercial_suivi = trim($this->fk_commercial_suivi);
1356 }
1357 if (isset($this->note_private)) {
1358 $this->note_private = trim($this->note_private);
1359 }
1360 if (isset($this->note_public)) {
1361 $this->note_public = trim($this->note_public);
1362 }
1363 if (isset($this->import_key)) {
1364 $this->import_key = trim($this->import_key);
1365 }
1366 //if (isset($this->extraparams)) $this->extraparams=trim($this->extraparams);
1367
1368 // $this->oldcopy must have been set by the caller of update
1369
1370 // Update request
1371 $sql = "UPDATE ".MAIN_DB_PREFIX."contrat SET";
1372 $sql .= " ref=".(isset($this->ref) ? "'".$this->db->escape($this->ref)."'" : "null").",";
1373 $sql .= " ref_customer=".(isset($this->ref_customer) ? "'".$this->db->escape($this->ref_customer)."'" : "null").",";
1374 $sql .= " ref_supplier=".(isset($this->ref_supplier) ? "'".$this->db->escape($this->ref_supplier)."'" : "null").",";
1375 $sql .= " ref_ext=".(isset($this->ref_ext) ? "'".$this->db->escape($this->ref_ext)."'" : "null").",";
1376 $sql .= " entity=".$conf->entity.",";
1377 $sql .= " date_contrat=".(dol_strlen($this->date_contrat) != 0 ? "'".$this->db->idate($this->date_contrat)."'" : 'null').",";
1378 $sql .= " statut=".(isset($this->statut) ? $this->statut : (isset($this->status) ? $this->status : "null")).",";
1379 $sql .= " fk_soc=".($this->socid > 0 ? $this->socid : "null").",";
1380 $sql .= " fk_projet=".($this->fk_project > 0 ? $this->fk_project : "null").",";
1381 $sql .= " fk_commercial_signature=".(isset($this->fk_commercial_signature) ? $this->fk_commercial_signature : "null").",";
1382 $sql .= " fk_commercial_suivi=".(isset($this->fk_commercial_suivi) ? $this->fk_commercial_suivi : "null").",";
1383 $sql .= " note_private=".(isset($this->note_private) ? "'".$this->db->escape($this->note_private)."'" : "null").",";
1384 $sql .= " note_public=".(isset($this->note_public) ? "'".$this->db->escape($this->note_public)."'" : "null").",";
1385 $sql .= " import_key=".(isset($this->import_key) ? "'".$this->db->escape($this->import_key)."'" : "null");
1386 //$sql.= " extraparams=".(isset($this->extraparams)?"'".$this->db->escape($this->extraparams)."'":"null");
1387 $sql .= " WHERE rowid=".((int) $this->id);
1388
1389 $this->db->begin();
1390
1391 $resql = $this->db->query($sql);
1392 if (!$resql) {
1393 $error++;
1394 $this->errors[] = "Error ".$this->db->lasterror();
1395 }
1396
1397 if (!$error) {
1398 $result = $this->insertExtraFields(); // This delete and reinsert extrafields
1399 if ($result < 0) {
1400 $error++;
1401 }
1402 }
1403
1404 if (!$error && !$notrigger) {
1405 // Call triggers
1406 $result = $this->call_trigger('CONTRACT_MODIFY', $user);
1407 if ($result < 0) {
1408 $error++;
1409 }
1410 // End call triggers
1411 }
1412
1413 // Commit or rollback
1414 if ($error) {
1415 foreach ($this->errors as $errmsg) {
1416 dol_syslog(get_class($this)."::update ".$errmsg, LOG_ERR);
1417 $this->error .= ($this->error ? ', '.$errmsg : $errmsg);
1418 }
1419 $this->db->rollback();
1420 return -1 * $error;
1421 } else {
1422 $this->db->commit();
1423 return 1;
1424 }
1425 }
1426
1427
1451 public function addline($desc, $pu_ht, $qty, $txtva, $txlocaltax1, $txlocaltax2, $fk_product, $remise_percent, $date_start, $date_end, $price_base_type = 'HT', $pu_ttc = 0.0, $info_bits = 0, $fk_fournprice = null, $pa_ht = 0, $array_options = array(), $fk_unit = null, $rang = 0)
1452 {
1453 global $user, $langs, $conf, $mysoc;
1454 $error = 0;
1455
1456 dol_syslog(get_class($this)."::addline $desc, $pu_ht, $qty, $txtva, $txlocaltax1, $txlocaltax2, $fk_product, $remise_percent, $date_start, $date_end, $price_base_type, $pu_ttc, $info_bits, $rang");
1457
1458 // Check parameters
1459 if ($fk_product <= 0 && empty($desc)) {
1460 $this->error = "ErrorDescRequiredForFreeProductLines";
1461 return -1;
1462 }
1463
1464 if ($this->statut >= 0) {
1465 // Clean parameters
1466 $pu_ht = price2num($pu_ht);
1467 $pu_ttc = price2num($pu_ttc);
1468 $pa_ht = price2num($pa_ht);
1469
1470 // Clean vat code
1471 $reg = array();
1472 $vat_src_code = '';
1473 if (preg_match('/\‍((.*)\‍)/', (string) $txtva, $reg)) {
1474 $vat_src_code = $reg[1];
1475 $txtva = preg_replace('/\s*\‍(.*\‍)/', '', (string) $txtva); // Remove code into vatrate.
1476 }
1477 $txtva = price2num($txtva);
1478 $txlocaltax1 = price2num($txlocaltax1);
1479 $txlocaltax2 = price2num($txlocaltax2);
1480
1481 $remise_percent = price2num($remise_percent);
1482 $qty = price2num($qty);
1483 if (empty($qty)) {
1484 $qty = 1;
1485 }
1486 if (empty($info_bits)) {
1487 $info_bits = 0;
1488 }
1489 if (empty($pu_ht) || !is_numeric($pu_ht)) {
1490 $pu_ht = 0;
1491 }
1492 if (empty($pu_ttc)) {
1493 $pu_ttc = 0;
1494 }
1495 if (empty($txtva) || !is_numeric($txtva)) {
1496 $txtva = 0;
1497 }
1498 if (empty($txlocaltax1) || !is_numeric($txlocaltax1)) {
1499 $txlocaltax1 = 0;
1500 }
1501 if (empty($txlocaltax2) || !is_numeric($txlocaltax2)) {
1502 $txlocaltax2 = 0;
1503 }
1504
1505 if ($price_base_type == 'HT') {
1506 $pu = $pu_ht;
1507 } else {
1508 $pu = $pu_ttc;
1509 }
1510
1511 // Check parameters
1512 if (empty($remise_percent)) {
1513 $remise_percent = 0;
1514 }
1515 if (empty($rang)) {
1516 $rang = 0;
1517 }
1518
1519 if ($date_start && $date_end && $date_start > $date_end) {
1520 $langs->load("errors");
1521 $this->error = $langs->trans('ErrorStartDateGreaterEnd');
1522 return -1;
1523 }
1524
1525 $this->db->begin();
1526
1527 $localtaxes_type = getLocalTaxesFromRate($txtva.($vat_src_code ? ' ('.$vat_src_code.')' : ''), 0, $this->societe, $mysoc);
1528
1529 // Calcul du total TTC et de la TVA pour la ligne a partir de
1530 // qty, pu, remise_percent et txtva
1531 // TRES IMPORTANT: C'est au moment de l'insertion ligne qu'on doit stocker
1532 // la part ht, tva et ttc, et ce au niveau de la ligne qui a son propre taux tva.
1533
1534 $tabprice = calcul_price_total($qty, $pu, $remise_percent, $txtva, $txlocaltax1, $txlocaltax2, 0, $price_base_type, $info_bits, 1, $mysoc, $localtaxes_type);
1535 $total_ht = $tabprice[0];
1536 $total_tva = $tabprice[1];
1537 $total_ttc = $tabprice[2];
1538 $total_localtax1 = $tabprice[9];
1539 $total_localtax2 = $tabprice[10];
1540
1541 $localtax1_type = $localtaxes_type[0];
1542 $localtax2_type = $localtaxes_type[2];
1543
1544 // TODO A virer
1545 // Anciens indicateurs: $price, $remise (a ne plus utiliser)
1546 $remise = 0;
1547 $price = price2num(round($pu_ht, 2));
1548 if (dol_strlen($remise_percent) > 0) {
1549 $remise = round(($pu_ht * $remise_percent / 100), 2);
1550 $price = $pu_ht - $remise;
1551 }
1552
1553 if (empty($pa_ht)) {
1554 $pa_ht = 0;
1555 }
1556
1557
1558 // if buy price not defined, define buyprice as configured in margin admin
1559 if ($pa_ht == 0) {
1560 $result = $this->defineBuyPrice($pu_ht, $remise_percent, $fk_product);
1561 if ($result < 0) {
1562 return -1;
1563 } else {
1564 $pa_ht = $result;
1565 }
1566 }
1567
1568 // Insertion dans la base
1569 $sql = "INSERT INTO ".MAIN_DB_PREFIX."contratdet";
1570 $sql .= " (fk_contrat, label, description, fk_product, qty, tva_tx, vat_src_code,";
1571 $sql .= " localtax1_tx, localtax2_tx, localtax1_type, localtax2_type, remise_percent, subprice,";
1572 $sql .= " total_ht, total_tva, total_localtax1, total_localtax2, total_ttc,";
1573 $sql .= " info_bits,";
1574 $sql .= " price_ht, remise, fk_product_fournisseur_price, buy_price_ht";
1575 if ($date_start > 0) {
1576 $sql .= ",date_ouverture_prevue";
1577 }
1578 if ($date_end > 0) {
1579 $sql .= ",date_fin_validite";
1580 }
1581 $sql .= ", fk_unit";
1582 $sql .= ", rang";
1583 $sql .= ") VALUES (";
1584 $sql .= $this->id.", '', '".$this->db->escape($desc)."',";
1585 $sql .= ($fk_product > 0 ? $fk_product : "null").",";
1586 $sql .= " ".((float) $qty).",";
1587 $sql .= " ".((float) $txtva).",";
1588 $sql .= " ".($vat_src_code ? "'".$this->db->escape($vat_src_code)."'" : "null").",";
1589 $sql .= " ".((float) $txlocaltax1).",";
1590 $sql .= " ".((float) $txlocaltax2).",";
1591 $sql .= " '".$this->db->escape($localtax1_type)."',";
1592 $sql .= " '".$this->db->escape($localtax2_type)."',";
1593 $sql .= " ".price2num($remise_percent).",";
1594 $sql .= " ".price2num($pu_ht).",";
1595 $sql .= " ".price2num($total_ht).",".price2num($total_tva).",".price2num($total_localtax1).",".price2num($total_localtax2).",".price2num($total_ttc).",";
1596 $sql .= " ".((int) $info_bits).",";
1597 $sql .= " ".price2num($price).",".price2num($remise).",";
1598 if (isset($fk_fournprice)) {
1599 $sql .= ' '.((int) $fk_fournprice).',';
1600 } else {
1601 $sql .= ' null,';
1602 }
1603 if (isset($pa_ht)) {
1604 $sql .= ' '.price2num($pa_ht);
1605 } else {
1606 $sql .= ' null';
1607 }
1608 if ($date_start > 0) {
1609 $sql .= ",'".$this->db->idate($date_start)."'";
1610 }
1611 if ($date_end > 0) {
1612 $sql .= ",'".$this->db->idate($date_end)."'";
1613 }
1614 $sql .= ", ".($fk_unit ? "'".$this->db->escape($fk_unit)."'" : "null");
1615 $sql .= ", ".(!empty($rang) ? (int) $rang : "0");
1616 $sql .= ")";
1617
1618 $resql = $this->db->query($sql);
1619 if ($resql) {
1620 $contractlineid = $this->db->last_insert_id(MAIN_DB_PREFIX."contratdet");
1621
1622 if (!$error) {
1623 $contractline = new ContratLigne($this->db);
1624 $contractline->array_options = $array_options;
1625 $contractline->id = $contractlineid;
1626 $result = $contractline->insertExtraFields();
1627 if ($result < 0) {
1628 $this->errors = array_merge($this->errors, $contractline->errors);
1629 $this->error = $contractline->error;
1630 $error++;
1631 }
1632 }
1633
1634 if (empty($error)) {
1635 // Call trigger
1636 $this->context['line_id'] = $contractlineid;
1637 $result = $this->call_trigger('LINECONTRACT_INSERT', $user);
1638 if ($result < 0) {
1639 $error++;
1640 }
1641 // End call triggers
1642 }
1643
1644 if ($error) {
1645 $this->db->rollback();
1646 return -1;
1647 } else {
1648 $this->db->commit();
1649 return $contractlineid;
1650 }
1651 } else {
1652 $this->db->rollback();
1653 $this->error = $this->db->error()." sql=".$sql;
1654 return -1;
1655 }
1656 } else {
1657 dol_syslog(get_class($this)."::addline ErrorTryToAddLineOnValidatedContract", LOG_ERR);
1658 return -2;
1659 }
1660 }
1661
1686 public function updateline($rowid, $desc, $pu, $qty, $remise_percent, $date_start, $date_end, $tvatx, $localtax1tx = 0.0, $localtax2tx = 0.0, $date_start_real = '', $date_end_real = '', $price_base_type = 'HT', $info_bits = 0, $fk_fournprice = null, $pa_ht = 0, $array_options = array(), $fk_unit = null, $rang = 0)
1687 {
1688 global $user, $conf, $langs, $mysoc;
1689
1690 $error = 0;
1691
1692 // Clean parameters
1693 $qty = trim((string) $qty);
1694 $desc = trim($desc);
1695 $desc = trim($desc);
1696 $price = price2num($pu);
1697 $tvatx = price2num($tvatx);
1698 $localtax1tx = price2num($localtax1tx);
1699 $localtax2tx = price2num($localtax2tx);
1700 $pa_ht = price2num($pa_ht);
1701 if (empty($fk_fournprice)) {
1702 $fk_fournprice = 0;
1703 }
1704 if (empty($rang)) {
1705 $rang = 0;
1706 }
1707
1708 $subprice = $price;
1709 $remise = 0;
1710 if (dol_strlen($remise_percent) > 0) {
1711 $remise = round(($pu * $remise_percent / 100), 2);
1712 $price = $pu - $remise;
1713 } else {
1714 $remise_percent = 0;
1715 }
1716
1717 if ($date_start && $date_end && $date_start > $date_end) {
1718 $langs->load("errors");
1719 $this->error = $langs->trans('ErrorStartDateGreaterEnd');
1720 return -1;
1721 }
1722
1723 dol_syslog(get_class($this)."::updateline $rowid, $desc, $pu, $qty, $remise_percent, $date_start, $date_end, $date_start_real, $date_end_real, $tvatx, $localtax1tx, $localtax2tx, $price_base_type, $info_bits, $rang");
1724
1725 $this->db->begin();
1726
1727 // Calcul du total TTC et de la TVA pour la ligne a partir de
1728 // qty, pu, remise_percent et tvatx
1729 // TRES IMPORTANT: C'est au moment de l'insertion ligne qu'on doit stocker
1730 // la part ht, tva et ttc, et ce au niveau de la ligne qui a son propre taux tva.
1731
1732 $localtaxes_type = getLocalTaxesFromRate($tvatx, 0, $this->societe, $mysoc);
1733 $tvatx = preg_replace('/\s*\‍(.*\‍)/', '', $tvatx); // Remove code into vatrate.
1734
1735 $tabprice = calcul_price_total($qty, $pu, $remise_percent, (float) price2num($tvatx), $localtax1tx, $localtax2tx, 0, $price_base_type, $info_bits, 1, $mysoc, $localtaxes_type);
1736 $total_ht = $tabprice[0];
1737 $total_tva = $tabprice[1];
1738 $total_ttc = $tabprice[2];
1739 $total_localtax1 = $tabprice[9];
1740 $total_localtax2 = $tabprice[10];
1741
1742 $localtax1_type = (empty($localtaxes_type[0]) ? '' : $localtaxes_type[0]);
1743 $localtax2_type = (empty($localtaxes_type[2]) ? '' : $localtaxes_type[2]);
1744
1745 // TODO A virer
1746 // Anciens indicateurs: $price, $remise (a ne plus utiliser)
1747 $remise = 0;
1748 $price = price2num(round($pu, 2));
1749 if (dol_strlen($remise_percent) > 0) {
1750 $remise = round(($pu * $remise_percent / 100), 2);
1751 $price = $pu - $remise;
1752 }
1753
1754 if (empty($pa_ht)) {
1755 $pa_ht = 0;
1756 }
1757
1758 // if buy price not defined, define buyprice as configured in margin admin
1759 if ($pa_ht == 0) {
1760 $result = $this->defineBuyPrice($pu, $remise_percent);
1761 if ($result < 0) {
1762 return -1;
1763 } else {
1764 $pa_ht = $result;
1765 }
1766 }
1767
1768 $sql = "UPDATE ".MAIN_DB_PREFIX."contratdet set description = '".$this->db->escape($desc)."'";
1769 $sql .= ",price_ht = ".((float) price2num($price));
1770 $sql .= ",subprice = ".((float) price2num($subprice));
1771 $sql .= ",remise = ".((float) price2num($remise));
1772 $sql .= ",remise_percent = ".((float) price2num($remise_percent));
1773 $sql .= ",qty = ".((float) $qty);
1774 $sql .= ",tva_tx = ".((float) price2num($tvatx));
1775 $sql .= ",localtax1_tx = ".((float) price2num($localtax1tx));
1776 $sql .= ",localtax2_tx = ".((float) price2num($localtax2tx));
1777 $sql .= ",localtax1_type='".$this->db->escape($localtax1_type)."'";
1778 $sql .= ",localtax2_type='".$this->db->escape($localtax2_type)."'";
1779 $sql .= ", total_ht = ".((float) price2num($total_ht));
1780 $sql .= ", total_tva = ".((float) price2num($total_tva));
1781 $sql .= ", total_localtax1 = ".((float) price2num($total_localtax1));
1782 $sql .= ", total_localtax2 = ".((float) price2num($total_localtax2));
1783 $sql .= ", total_ttc = ".((float) price2num($total_ttc));
1784 $sql .= ", fk_product_fournisseur_price=".($fk_fournprice > 0 ? $fk_fournprice : "null");
1785 $sql .= ", buy_price_ht = ".((float) price2num($pa_ht));
1786 if ($date_start > 0) {
1787 $sql .= ",date_ouverture_prevue = '".$this->db->idate($date_start)."'";
1788 } else {
1789 $sql .= ",date_ouverture_prevue = null";
1790 }
1791 if ($date_end > 0) {
1792 $sql .= ",date_fin_validite = '".$this->db->idate($date_end)."'";
1793 } else {
1794 $sql .= ",date_fin_validite = null";
1795 }
1796 if ($date_start_real > 0) {
1797 $sql .= ",date_ouverture = '".$this->db->idate($date_start_real)."'";
1798 } else {
1799 $sql .= ",date_ouverture = null";
1800 }
1801 if ($date_end_real > 0) {
1802 $sql .= ",date_cloture = '".$this->db->idate($date_end_real)."'";
1803 } else {
1804 $sql .= ",date_cloture = null";
1805 }
1806 $sql .= ", fk_unit = ".($fk_unit > 0 ? ((int) $fk_unit) : "null");
1807 $sql .= ", rang = ".(!empty($rang) ? ((int) $rang) : "0");
1808 $sql .= " WHERE rowid = ".((int) $rowid);
1809
1810 dol_syslog(get_class($this)."::updateline", LOG_DEBUG);
1811 $result = $this->db->query($sql);
1812 if ($result) {
1813 if (is_array($array_options) && count($array_options) > 0) { // For avoid conflicts if trigger used
1814 $contractline = new ContratLigne($this->db);
1815 $contractline->fetch($rowid);
1816
1817 // We replace values in $contractline->array_options only for entries defined into $array_options
1818 foreach ($array_options as $key => $value) {
1819 $contractline->array_options[$key] = $array_options[$key];
1820 }
1821
1822 $result = $contractline->insertExtraFields();
1823 if ($result < 0) {
1824 $this->errors[] = $contractline->error;
1825 $error++;
1826 }
1827 }
1828
1829 if (empty($error)) {
1830 // Call trigger
1831 $this->context['line_id'] = $rowid;
1832 $result = $this->call_trigger('LINECONTRACT_MODIFY', $user);
1833 if ($result < 0) {
1834 $this->db->rollback();
1835 return -3;
1836 }
1837 // End call triggers
1838
1839 $this->db->commit();
1840 return 1;
1841 } else {
1842 $this->db->rollback();
1843 return -1;
1844 }
1845 } else {
1846 $this->db->rollback();
1847 $this->error = $this->db->error();
1848 dol_syslog(get_class($this)."::updateline Erreur -1");
1849 return -1;
1850 }
1851 }
1852
1860 public function deleteLine($idline, User $user)
1861 {
1862 $error = 0;
1863
1864 if ($this->statut >= 0) {
1865 // Call trigger
1866 $this->context['line_id'] = $idline;
1867 $result = $this->call_trigger('LINECONTRACT_DELETE', $user);
1868 if ($result < 0) {
1869 return -1;
1870 }
1871 // End call triggers
1872
1873 $this->db->begin();
1874
1875 $sql = "DELETE FROM ".MAIN_DB_PREFIX.$this->table_element_line;
1876 $sql .= " WHERE rowid = ".((int) $idline);
1877
1878 dol_syslog(get_class($this)."::deleteline", LOG_DEBUG);
1879 $resql = $this->db->query($sql);
1880 if (!$resql) {
1881 $this->error = "Error ".$this->db->lasterror();
1882 $error++;
1883 }
1884
1885 if (!$error) {
1886 // Remove extrafields
1887 $contractline = new ContratLigne($this->db);
1888 $contractline->id = $idline;
1889 $result = $contractline->deleteExtraFields();
1890 if ($result < 0) {
1891 $error++;
1892 $this->error = "Error ".get_class($this)."::deleteline deleteExtraFields error -4 ".$contractline->error;
1893 }
1894 }
1895
1896 if (empty($error)) {
1897 $this->db->commit();
1898 return 1;
1899 } else {
1900 dol_syslog(get_class($this)."::deleteline ERROR:".$this->error, LOG_ERR);
1901 $this->db->rollback();
1902 return -1;
1903 }
1904 } else {
1905 $this->error = 'ErrorDeleteLineNotAllowedByObjectStatus';
1906 return -2;
1907 }
1908 }
1909
1910
1911 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1919 public function update_statut($user)
1920 {
1921 // phpcs:enable
1922 dol_syslog(__METHOD__." is deprecated", LOG_WARNING);
1923
1924 // If draft, we keep it (should not happen)
1925 if ($this->statut == 0) {
1926 return 1;
1927 }
1928
1929 // Load $this->lines array
1930 // $this->fetch_lines();
1931
1932 // $newstatut=1;
1933 // foreach($this->lines as $key => $contractline)
1934 // {
1935 // // if ($contractline) // Loop on each service
1936 // }
1937
1938 return 1;
1939 }
1940
1941
1948 public function getLibStatut($mode)
1949 {
1950 return $this->LibStatut($this->statut, $mode);
1951 }
1952
1953 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1961 public function LibStatut($status, $mode)
1962 {
1963 // phpcs:enable
1964 global $langs;
1965
1966 if (empty($this->labelStatus) || empty($this->labelStatusShort)) {
1967 global $langs;
1968 $langs->load("contracts");
1969 $this->labelStatus[self::STATUS_DRAFT] = $langs->transnoentitiesnoconv('ContractStatusDraft');
1970 $this->labelStatus[self::STATUS_VALIDATED] = $langs->transnoentitiesnoconv('ContractStatusValidated');
1971 $this->labelStatus[self::STATUS_CLOSED] = $langs->transnoentitiesnoconv('ContractStatusClosed');
1972 $this->labelStatusShort[self::STATUS_DRAFT] = $langs->transnoentitiesnoconv('ContractStatusDraft');
1973 $this->labelStatusShort[self::STATUS_VALIDATED] = $langs->transnoentitiesnoconv('ContractStatusValidated');
1974 $this->labelStatusShort[self::STATUS_CLOSED] = $langs->transnoentitiesnoconv('ContractStatusClosed');
1975 }
1976
1977 $statusType = 'status'.$status;
1978 if ($status == self::STATUS_VALIDATED) {
1979 $statusType = 'status6';
1980 }
1981
1982 if ($mode == 4 || $mode == 6 || $mode == 7) {
1983 $text = '';
1984 if ($mode == 4) {
1985 $text = '<span class="hideonsmartphone">';
1986 $text .= ($this->nbofserviceswait + $this->nbofservicesopened + $this->nbofservicesexpired + $this->nbofservicesclosed);
1987 $text .= ' '.$langs->trans("Services");
1988 $text .= ': &nbsp; &nbsp; ';
1989 $text .= '</span>';
1990 }
1991 $text .= ($mode == 7 ? '<span class="nowraponall">' : '');
1992 $text .= ($mode != 7 || $this->nbofserviceswait > 0) ? ($this->nbofserviceswait.ContratLigne::LibStatut(0, 3, -1, 'class="marginleft2"')).(($mode != 7 || $this->nbofservicesopened || $this->nbofservicesexpired || $this->nbofservicesclosed) ? ' &nbsp; ' : '') : '';
1993 $text .= ($mode == 7 ? '</span><span class="nowraponall">' : '');
1994 $text .= ($mode != 7 || $this->nbofservicesopened > 0) ? ($this->nbofservicesopened.ContratLigne::LibStatut(4, 3, 0, 'class="marginleft2"')).(($mode != 7 || $this->nbofservicesexpired || $this->nbofservicesclosed) ? ' &nbsp; ' : '') : '';
1995 $text .= ($mode == 7 ? '</span><span class="nowraponall">' : '');
1996 $text .= ($mode != 7 || $this->nbofservicesexpired > 0) ? ($this->nbofservicesexpired.ContratLigne::LibStatut(4, 3, 1, 'class="marginleft2"')).(($mode != 7 || $this->nbofservicesclosed) ? ' &nbsp; ' : '') : '';
1997 $text .= ($mode == 7 ? '</span><span class="nowraponall">' : '');
1998 $text .= ($mode != 7 || $this->nbofservicesclosed > 0) ? ($this->nbofservicesclosed.ContratLigne::LibStatut(5, 3, -1, 'class="marginleft2"')) : '';
1999 $text .= ($mode == 7 ? '</span>' : '');
2000 return $text;
2001 } else {
2002 return dolGetStatus($this->labelStatus[$status], $this->labelStatusShort[$status], '', $statusType, $mode);
2003 }
2004 }
2005
2012 public function getTooltipContentArray($params)
2013 {
2014 global $conf, $langs, $user;
2015
2016 $langs->load('contracts');
2017
2018 $datas = [];
2019 $nofetch = !empty($params['nofetch']);
2020
2021 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2022 return ['optimize' => $langs->trans("ShowContract")];
2023 }
2024 if ($user->hasRight('contrat', 'lire')) {
2025 $datas['picto'] = img_picto('', $this->picto).' <u class="paddingrightonly">'.$langs->trans("Contract").'</u>';
2026 /* Status of a contract is status of all services, so disabled
2027 if (isset($this->statut)) {
2028 $label .= ' '.$this->getLibStatut(5);
2029 }*/
2030 $datas['ref'] = '<br><b>'.$langs->trans('Ref').':</b> '.($this->ref ? $this->ref : $this->id);
2031 if (!$nofetch) {
2032 $langs->load('companies');
2033 if (empty($this->thirdparty)) {
2034 $this->fetch_thirdparty();
2035 }
2036 $datas['customer'] = '<br><b>'.$langs->trans('Customer').':</b> '.$this->thirdparty->getNomUrl(1, '', 0, 1);
2037 }
2038 $datas['refcustomer'] = '<br><b>'.$langs->trans('RefCustomer').':</b> '. $this->ref_customer;
2039 if (!$nofetch) {
2040 $langs->load('project');
2041 if (is_null($this->project) || (is_object($this->project) && $this->project->isEmpty())) {
2042 $res = $this->fetch_project();
2043 if ($res > 0 && $this->project instanceof Project) {
2044 $datas['project'] = '<br><b>'.$langs->trans('Project').':</b> '.$this->project->getNomUrl(1, '', 0, 1);
2045 }
2046 }
2047 }
2048 $datas['refsupplier'] = '<br><b>'.$langs->trans('RefSupplier').':</b> '.$this->ref_supplier;
2049 if (!empty($this->total_ht)) {
2050 $datas['amountht'] = '<br><b>'.$langs->trans('AmountHT').':</b> '.price($this->total_ht, 0, $langs, 0, -1, -1, $conf->currency);
2051 }
2052 if (!empty($this->total_tva)) {
2053 $datas['vatamount'] = '<br><b>'.$langs->trans('VAT').':</b> '.price($this->total_tva, 0, $langs, 0, -1, -1, $conf->currency);
2054 }
2055 if (!empty($this->total_ttc)) {
2056 $datas['amounttc'] = '<br><b>'.$langs->trans('AmountTTC').':</b> '.price($this->total_ttc, 0, $langs, 0, -1, -1, $conf->currency);
2057 }
2058 }
2059 return $datas;
2060 }
2061
2071 public function getNomUrl($withpicto = 0, $maxlength = 0, $notooltip = 0, $save_lastsearch_value = -1)
2072 {
2073 global $conf, $langs, $user, $hookmanager;
2074
2075 $result = '';
2076
2077 $url = DOL_URL_ROOT.'/contrat/card.php?id='.$this->id;
2078
2079 //if ($option !== 'nolink')
2080 //{
2081 // Add param to save lastsearch_values or not
2082 $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
2083 if ($save_lastsearch_value == -1 && isset($_SERVER["PHP_SELF"]) && preg_match('/list\.php/', $_SERVER["PHP_SELF"])) {
2084 $add_save_lastsearch_values = 1;
2085 }
2086 if ($add_save_lastsearch_values) {
2087 $url .= '&save_lastsearch_values=1';
2088 }
2089 //}
2090 $params = [
2091 'id' => $this->id,
2092 'objecttype' => $this->element,
2093 'nofetch' => 1,
2094 ];
2095 $classfortooltip = 'classfortooltip';
2096 $dataparams = '';
2097 if (getDolGlobalInt('MAIN_ENABLE_AJAX_TOOLTIP')) {
2098 $classfortooltip = 'classforajaxtooltip';
2099 $dataparams = ' data-params="'.dol_escape_htmltag(json_encode($params)).'"';
2100 $label = '';
2101 } else {
2102 $label = implode($this->getTooltipContentArray($params));
2103 }
2104
2105 $linkclose = '';
2106 if (empty($notooltip) && $user->hasRight('contrat', 'lire')) {
2107 if (getDolGlobalString('MAIN_OPTIMIZEFORTEXTBROWSER')) {
2108 $label = $langs->trans("ShowContract");
2109 $linkclose .= ' alt="'.dol_escape_htmltag($label, 1).'"';
2110 }
2111 $linkclose .= ($label ? ' title="'.dol_escape_htmltag($label, 1).'"' : ' title="tocomplete"');
2112 $linkclose .= $dataparams.' class="'.$classfortooltip.'"';
2113 }
2114 $linkstart = '<a href="'.$url.'"';
2115 $linkstart .= $linkclose.'>';
2116 $linkend = '</a>';
2117
2118 $result .= $linkstart;
2119 if ($withpicto) {
2120 $result .= img_object(($notooltip ? '' : $label), ($this->picto ? $this->picto : 'generic'), (($withpicto != 2) ? 'class="paddingright"' : ''), 0, 0, $notooltip ? 0 : 1);
2121 }
2122 if ($withpicto != 2) {
2123 $result .= ($this->ref ? $this->ref : $this->id);
2124 }
2125 $result .= $linkend;
2126
2127 global $action;
2128 $hookmanager->initHooks(array('contractdao'));
2129 $parameters = array('id' => $this->id, 'getnomurl' => &$result);
2130 $reshook = $hookmanager->executeHooks('getNomUrl', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
2131 if ($reshook > 0) {
2132 $result = $hookmanager->resPrint;
2133 } else {
2134 $result .= $hookmanager->resPrint;
2135 }
2136
2137 return $result;
2138 }
2139
2146 public function info($id)
2147 {
2148 $sql = "SELECT c.rowid, c.ref, c.datec,";
2149 $sql .= " c.tms as date_modification,";
2150 $sql .= " fk_user_author";
2151 $sql .= " FROM ".MAIN_DB_PREFIX."contrat as c";
2152 $sql .= " WHERE c.rowid = ".((int) $id);
2153
2154 $result = $this->db->query($sql);
2155 if ($result) {
2156 if ($this->db->num_rows($result)) {
2157 $obj = $this->db->fetch_object($result);
2158
2159 $this->id = $obj->rowid;
2160 $this->ref = (!$obj->ref) ? $obj->rowid : $obj->ref;
2161
2162 $this->user_creation_id = $obj->fk_user_author;
2163 $this->date_creation = $this->db->jdate($obj->datec);
2164 $this->date_modification = $this->db->jdate($obj->date_modification);
2165 }
2166
2167 $this->db->free($result);
2168 } else {
2169 dol_print_error($this->db);
2170 }
2171 }
2172
2173 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2180 public function array_detail($status = -1)
2181 {
2182 // phpcs:enable
2183 $tab = array();
2184
2185 $sql = "SELECT cd.rowid";
2186 $sql .= " FROM ".MAIN_DB_PREFIX."contratdet as cd";
2187 $sql .= " WHERE fk_contrat =".((int) $this->id);
2188 if ($status >= 0) {
2189 $sql .= " AND statut = ".((int) $status);
2190 }
2191
2192 dol_syslog(get_class($this)."::array_detail()", LOG_DEBUG);
2193 $resql = $this->db->query($sql);
2194 if ($resql) {
2195 $num = $this->db->num_rows($resql);
2196 $i = 0;
2197 while ($i < $num) {
2198 $obj = $this->db->fetch_object($resql);
2199 $tab[$i] = $obj->rowid;
2200 $i++;
2201 }
2202 return $tab;
2203 } else {
2204 $this->error = $this->db->error();
2205 return -1;
2206 }
2207 }
2208
2218 public function getListOfContracts($option = 'all', $status = [], $product_categories = [], $line_status = [])
2219 {
2220 $tab = array();
2221
2222 $sql = "SELECT c.rowid";
2223 $sql .= " FROM ".MAIN_DB_PREFIX."contrat as c";
2224 if (!empty($product_categories)) {
2225 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."contratdet as cd ON cd.fk_contrat = c.rowid";
2226 $sql .= " INNER JOIN ".MAIN_DB_PREFIX."categorie_product as cp ON cp.fk_product = cd.fk_product AND cp.fk_categorie IN (".$this->db->sanitize(implode(', ', $product_categories)).")";
2227 }
2228 $sql .= " WHERE c.fk_soc =".((int) $this->socid);
2229 $sql .= ($option == 'others') ? " AND c.rowid <> ".((int) $this->id) : "";
2230 $sql .= (!empty($status)) ? " AND c.statut IN (".$this->db->sanitize(implode(', ', $status)).")" : "";
2231 $sql .= (!empty($line_status)) ? " AND cd.statut IN (".$this->db->sanitize(implode(', ', $line_status)).")" : "";
2232 $sql .= " GROUP BY c.rowid";
2233
2234 dol_syslog(get_class($this)."::getOtherContracts()", LOG_DEBUG);
2235 $resql = $this->db->query($sql);
2236 if ($resql) {
2237 $num = $this->db->num_rows($resql);
2238 $i = 0;
2239 while ($i < $num) {
2240 $obj = $this->db->fetch_object($resql);
2241 $contrat = new Contrat($this->db);
2242 $contrat->fetch($obj->rowid);
2243 $tab[$contrat->id] = $contrat;
2244 $i++;
2245 }
2246 return $tab;
2247 } else {
2248 $this->error = $this->db->lasterror();
2249 return -1;
2250 }
2251 }
2252
2253
2254 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2262 public function load_board($user, $mode)
2263 {
2264 // phpcs:enable
2265 global $conf, $langs;
2266
2267 $this->from = " FROM ".MAIN_DB_PREFIX."contrat as c";
2268 $this->from .= ", ".MAIN_DB_PREFIX."contratdet as cd";
2269 $this->from .= ", ".MAIN_DB_PREFIX."societe as s";
2270 if (!$user->hasRight('societe', 'client', 'voir')) {
2271 $this->from .= ", ".MAIN_DB_PREFIX."societe_commerciaux as sc";
2272 }
2273
2274 if ($mode == 'inactive') {
2275 $sql = "SELECT cd.rowid, cd.date_ouverture_prevue as datefin";
2276 $sql .= $this->from;
2277 $sql .= " WHERE c.statut = 1";
2278 $sql .= " AND c.rowid = cd.fk_contrat";
2279 $sql .= " AND cd.statut = 0";
2280 } elseif ($mode == 'expired') {
2281 $sql = "SELECT cd.rowid, cd.date_fin_validite as datefin";
2282 $sql .= $this->from;
2283 $sql .= " WHERE c.statut = 1";
2284 $sql .= " AND c.rowid = cd.fk_contrat";
2285 $sql .= " AND cd.statut = 4";
2286 $sql .= " AND cd.date_fin_validite < '".$this->db->idate(dol_now())."'";
2287 } elseif ($mode == 'active') {
2288 $sql = "SELECT cd.rowid, cd.date_fin_validite as datefin";
2289 $sql .= $this->from;
2290 $sql .= " WHERE c.statut = 1";
2291 $sql .= " AND c.rowid = cd.fk_contrat";
2292 $sql .= " AND cd.statut = 4";
2293 //$datetouse = dol_now();
2294 //$sql.= " AND cd.date_fin_validite < '".$this->db->idate($datetouse)."'";
2295 }
2296 $sql .= " AND c.fk_soc = s.rowid";
2297 $sql .= " AND c.entity = ".((int) $conf->entity);
2298 if ($user->socid) {
2299 $sql .= " AND c.fk_soc = ".((int) $user->socid);
2300 }
2301 if (!$user->hasRight('societe', 'client', 'voir')) {
2302 $sql .= " AND c.fk_soc = sc.fk_soc AND sc.fk_user = ".((int) $user->id);
2303 }
2304
2305 $resql = $this->db->query($sql);
2306 if ($resql) {
2307 $langs->load("contracts");
2308 $now = dol_now();
2309
2310 if ($mode == 'inactive') {
2311 $warning_delay = $conf->contrat->services->inactifs->warning_delay;
2312 $label = $langs->trans("BoardNotActivatedServices");
2313 $labelShort = $langs->trans("BoardNotActivatedServicesShort");
2314 $url = DOL_URL_ROOT.'/contrat/services_list.php?mainmenu=commercial&leftmenu=contracts&search_status=0&sortfield=cd.date_fin_validite&sortorder=asc';
2315 } elseif ($mode == 'expired') {
2316 $warning_delay = $conf->contrat->services->expires->warning_delay;
2317 $url = DOL_URL_ROOT.'/contrat/services_list.php?mainmenu=commercial&leftmenu=contracts&search_status=4&filter=expired&sortfield=cd.date_fin_validite&sortorder=asc';
2318 $label = $langs->trans("BoardExpiredServices");
2319 $labelShort = $langs->trans("BoardExpiredServicesShort");
2320 } else {
2321 $warning_delay = $conf->contrat->services->expires->warning_delay;
2322 $url = DOL_URL_ROOT.'/contrat/services_list.php?mainmenu=commercial&leftmenu=contracts&search_status=4&sortfield=cd.date_fin_validite&sortorder=asc';
2323 //$url.= '&op2day='.$arraydatetouse['mday'].'&op2month='.$arraydatetouse['mon'].'&op2year='.$arraydatetouse['year'];
2324 //if ($warning_delay >= 0) $url.='&amp;filter=expired';
2325 $label = $langs->trans("BoardRunningServices");
2326 $labelShort = $langs->trans("BoardRunningServicesShort");
2327 }
2328
2329 $response = new WorkboardResponse();
2330 $response->warning_delay = $warning_delay / 60 / 60 / 24;
2331 $response->label = $label;
2332 $response->labelShort = $labelShort;
2333 $response->url = $url;
2334 $response->img = img_object('', "contract");
2335
2336 while ($obj = $this->db->fetch_object($resql)) {
2337 $response->nbtodo++;
2338
2339 if ($obj->datefin && $this->db->jdate($obj->datefin) < ($now - $warning_delay)) {
2340 $response->nbtodolate++;
2341 }
2342 }
2343
2344 return $response;
2345 } else {
2346 dol_print_error($this->db);
2347 $this->error = $this->db->error();
2348 return -1;
2349 }
2350 }
2351
2357 public function loadStateBoard()
2358 {
2359 global $conf, $user;
2360
2361 $this->nb = array();
2362 $clause = "WHERE";
2363
2364 $sql = "SELECT count(c.rowid) as nb";
2365 $sql .= " FROM ".MAIN_DB_PREFIX."contrat as c";
2366 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON c.fk_soc = s.rowid";
2367 if (!$user->hasRight('societe', 'client', 'voir')) {
2368 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe_commerciaux as sc ON s.rowid = sc.fk_soc";
2369 $sql .= " WHERE sc.fk_user = ".((int) $user->id);
2370 $clause = "AND";
2371 }
2372 $sql .= " ".$clause." c.entity = ".$conf->entity;
2373
2374 $resql = $this->db->query($sql);
2375 if ($resql) {
2376 while ($obj = $this->db->fetch_object($resql)) {
2377 $this->nb["contracts"] = $obj->nb;
2378 }
2379 $this->db->free($resql);
2380 return 1;
2381 } else {
2382 dol_print_error($this->db);
2383 $this->error = $this->db->error();
2384 return -1;
2385 }
2386 }
2387
2388
2389 /* gestion des contacts d'un contrat */
2390
2396 public function getIdBillingContact()
2397 {
2398 return $this->getIdContact('external', 'BILLING');
2399 }
2400
2406 public function getIdServiceContact()
2407 {
2408 return $this->getIdContact('external', 'SERVICE');
2409 }
2410
2411
2419 public function initAsSpecimen()
2420 {
2421 global $user, $langs, $conf;
2422
2423 // Load array of products prodids
2424 $num_prods = 0;
2425 $prodids = array();
2426 $sql = "SELECT rowid";
2427 $sql .= " FROM ".MAIN_DB_PREFIX."product";
2428 $sql .= " WHERE entity IN (".getEntity('product').")";
2429 $sql .= " AND tosell = 1";
2430 $sql .= $this->db->plimit(100);
2431
2432 $resql = $this->db->query($sql);
2433 if ($resql) {
2434 $num_prods = $this->db->num_rows($resql);
2435 $i = 0;
2436 while ($i < $num_prods) {
2437 $i++;
2438 $row = $this->db->fetch_row($resql);
2439 $prodids[$i] = $row[0];
2440 }
2441 }
2442
2443 // Initialise parameters
2444 $this->id = 0;
2445 $this->specimen = 1;
2446
2447 $this->ref = 'SPECIMEN';
2448 $this->ref_customer = 'SPECIMENCUST';
2449 $this->ref_supplier = 'SPECIMENSUPP';
2450 $this->socid = 1;
2451 $this->status = 0;
2452 $this->date_creation = (dol_now() - 3600 * 24 * 7);
2453 $this->date_contrat = dol_now();
2454 $this->commercial_signature_id = 1;
2455 $this->commercial_suivi_id = 1;
2456 $this->note_private = 'This is a comment (private)';
2457 $this->note_public = 'This is a comment (public)';
2458 $this->fk_project = 0;
2459 // Lines
2460 $nbp = 5;
2461 $xnbp = 0;
2462 while ($xnbp < $nbp) {
2463 $line = new ContratLigne($this->db);
2464 $line->qty = 1;
2465 $line->subprice = 100;
2466 $line->tva_tx = 19.6;
2467 $line->remise_percent = 10;
2468 $line->total_ht = 90;
2469 $line->total_ttc = 107.64; // 90 * 1.196
2470 $line->total_tva = 17.64;
2471 $line->date_start = dol_now() - 500000;
2472 $line->date_start_real = dol_now() - 200000;
2473 $line->date_end = dol_now() + 500000;
2474 $line->date_end_real = dol_now() - 100000;
2475 if ($num_prods > 0) {
2476 $prodid = mt_rand(1, $num_prods);
2477 $line->fk_product = $prodids[$prodid];
2478 }
2479 $this->lines[$xnbp] = $line;
2480 $xnbp++;
2481 }
2482
2483 return 1;
2484 }
2485
2491 public function getLinesArray()
2492 {
2493 return $this->fetch_lines();
2494 }
2495
2501 public function getTicketsArray()
2502 {
2503 global $user;
2504
2505 $ticket = new Ticket($this->db);
2506 $nbTicket = $ticket->fetchAll($user, 'ASC', 't.datec', '', 0, '', array('t.fk_contract' => $this->id));
2507
2508 return ($nbTicket < 0 ? $nbTicket : $ticket->lines);
2509 }
2510
2511
2523 public function generateDocument($modele, $outputlangs, $hidedetails = 0, $hidedesc = 0, $hideref = 0, $moreparams = null)
2524 {
2525 global $conf, $langs;
2526
2527 if (!dol_strlen($modele)) {
2528 $modele = ''; // No doc template/generation by default
2529
2530 if (!empty($this->model_pdf)) {
2531 $modele = $this->model_pdf;
2532 } elseif (getDolGlobalString('CONTRACT_ADDON_PDF')) {
2533 $modele = getDolGlobalString('CONTRACT_ADDON_PDF');
2534 }
2535 }
2536
2537 if (empty($modele)) {
2538 return 0;
2539 } else {
2540 $langs->load("contracts");
2541 $outputlangs->load("products");
2542
2543 $modelpath = "core/modules/contract/doc/";
2544 return $this->commonGenerateDocument($modelpath, $modele, $outputlangs, $hidedetails, $hidedesc, $hideref, $moreparams);
2545 }
2546 }
2547
2556 public static function replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
2557 {
2558 $tables = array(
2559 'contrat'
2560 );
2561
2562 return CommonObject::commonReplaceThirdparty($dbs, $origin_id, $dest_id, $tables);
2563 }
2564
2573 public static function replaceProduct(DoliDB $db, $origin_id, $dest_id)
2574 {
2575 $tables = array(
2576 'contratdet'
2577 );
2578
2579 return CommonObject::commonReplaceProduct($db, $origin_id, $dest_id, $tables);
2580 }
2581
2590 public function createFromClone(User $user, $socid = 0, $notrigger = 0)
2591 {
2592 global $db, $langs, $conf, $hookmanager, $extrafields;
2593
2594 dol_include_once('/projet/class/project.class.php');
2595
2596 $error = 0;
2597
2598 $this->fetch($this->id);
2599
2600 // Load dest object
2601 $clonedObj = clone $this;
2602 $clonedObj->socid = $socid;
2603
2604 $this->db->begin();
2605
2606 $objsoc = new Societe($this->db);
2607
2608 $objsoc->fetch($clonedObj->socid);
2609
2610 // Clean data
2611 $clonedObj->statut = 0;
2612 // Clean extrafields
2613 if (is_array($clonedObj->array_options) && count($clonedObj->array_options) > 0) {
2614 $extrafields->fetch_name_optionals_label($this->table_element);
2615 foreach ($clonedObj->array_options as $key => $option) {
2616 $shortkey = preg_replace('/options_/', '', $key);
2617 //var_dump($shortkey); var_dump($extrafields->attributes[$this->element]['unique'][$shortkey]);
2618 if (!empty($extrafields->attributes[$this->element]['unique'][$shortkey])) {
2619 //var_dump($key); var_dump($clonedObj->array_options[$key]); exit;
2620 unset($clonedObj->array_options[$key]);
2621 }
2622 }
2623 }
2624
2625 if (!getDolGlobalString('CONTRACT_ADDON') || !is_readable(DOL_DOCUMENT_ROOT."/core/modules/contract/" . getDolGlobalString('CONTRACT_ADDON').".php")) {
2626 $this->error = 'ErrorSetupNotComplete';
2627 dol_syslog($this->error);
2628 return -1;
2629 }
2630
2631 // Set ref
2632 require_once DOL_DOCUMENT_ROOT."/core/modules/contract/" . getDolGlobalString('CONTRACT_ADDON').'.php';
2633 $obj = getDolGlobalString('CONTRACT_ADDON');
2634 $modContract = new $obj();
2635 '@phan-var-force CommonNumRefGenerator $modContrat';
2636 $clonedObj->ref = $modContract->getNextValue($objsoc, $clonedObj);
2637
2638 // get extrafields so they will be clone
2639 foreach ($this->lines as $line) {
2640 $line->fetch_optionals($line->id);
2641 }
2642
2643 // Create clone
2644 $clonedObj->context['createfromclone'] = 'createfromclone';
2645 $result = $clonedObj->create($user);
2646 if ($result < 0) {
2647 $error++;
2648 $this->error = $clonedObj->error;
2649 $this->errors[] = $clonedObj->error;
2650 } else {
2651 // copy external contacts if same company
2652 if ($this->socid == $clonedObj->socid) {
2653 if ($clonedObj->copy_linked_contact($this, 'external') < 0) {
2654 $error++;
2655 }
2656 }
2657 }
2658
2659 if (!$error) {
2660 foreach ($this->lines as $line) {
2661 $result = $clonedObj->addline($line->description, $line->subprice, $line->qty, $line->tva_tx, $line->localtax1_tx, $line->localtax2_tx, $line->fk_product, $line->remise_percent, $line->date_start, $line->date_cloture, 'HT', 0, $line->info_bits, $line->fk_fournprice, $line->pa_ht, $line->array_options, $line->fk_unit, $line->rang);
2662 if ($result < 0) {
2663 $error++;
2664 $this->error = $clonedObj->error;
2665 $this->errors[] = $clonedObj->error;
2666 }
2667 }
2668 }
2669
2670 if (!$error) {
2671 // Hook of thirdparty module
2672 if (is_object($hookmanager)) {
2673 $parameters = array(
2674 'objFrom' => $this,
2675 'clonedObj' => $clonedObj
2676 );
2677 $action = '';
2678 $reshook = $hookmanager->executeHooks('createFrom', $parameters, $clonedObj, $action); // Note that $action and $object may have been modified by some hooks
2679 if ($reshook < 0) {
2680 $this->setErrorsFromObject($hookmanager);
2681 $error++;
2682 }
2683 }
2684 }
2685
2686 unset($clonedObj->context['createfromclone']);
2687
2688 // End
2689 if (!$error) {
2690 $this->db->commit();
2691 return $clonedObj->id;
2692 } else {
2693 $this->db->rollback();
2694 return -1;
2695 }
2696 }
2697
2698
2708 public function doAutoRenewContracts($thirdparty_id = 0, $delayindaysshort = 0)
2709 {
2710 global $langs, $user;
2711
2712 $langs->load("agenda");
2713
2714 $now = dol_now();
2715
2716 $enddatetoscan = dol_time_plus_duree($now, -1 * abs($delayindaysshort), 'd');
2717
2718 $error = 0;
2719 $this->output = '';
2720 $this->error = '';
2721
2722 $contractlineprocessed = array();
2723 $contractignored = array();
2724 $contracterror = array();
2725
2726 dol_syslog(__METHOD__, LOG_DEBUG);
2727
2728 $sql = 'SELECT c.rowid, c.ref_customer, cd.rowid as lid, cd.date_fin_validite, p.duration';
2729 $sql .= ' FROM '.MAIN_DB_PREFIX.'contrat as c, '.MAIN_DB_PREFIX.'contratdet as cd';
2730 $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product as p ON p.rowid = cd.fk_product';
2731 $sql .= ' WHERE cd.fk_contrat = c.rowid';
2732 $sql .= " AND date_format(cd.date_fin_validite, '%Y-%m-%d') <= date_format('".$this->db->idate($enddatetoscan)."', '%Y-%m-%d')";
2733 $sql .= " AND cd.statut = 4";
2734 if ($thirdparty_id > 0) {
2735 $sql .= " AND c.fk_soc = ".((int) $thirdparty_id);
2736 }
2737 //print $sql;
2738
2739 $resql = $this->db->query($sql);
2740 if ($resql) {
2741 $num = $this->db->num_rows($resql);
2742
2743 include_once DOL_DOCUMENT_ROOT.'/core/class/html.formmail.class.php';
2744
2745 $i = 0;
2746 while ($i < $num) {
2747 $obj = $this->db->fetch_object($resql);
2748 if ($obj) {
2749 if (!empty($contractlineprocessed[$obj->lid]) || !empty($contractignored[$obj->rowid]) || !empty($contracterror[$obj->rowid])) {
2750 continue;
2751 }
2752
2753 // Load contract
2754 $object = new Contrat($this->db);
2755 $object->fetch($obj->rowid); // fetch also lines
2756 //$object->fetch_thirdparty();
2757
2758 if ($object->id <= 0) {
2759 $error++;
2760 $this->errors[] = 'Failed to load contract with id='.$obj->rowid;
2761 continue;
2762 }
2763
2764 dol_syslog("* Process contract line in doRenewalContracts for contract id=".$object->id." ref=".$object->ref." ref_customer=".$object->ref_customer." contract line id=".$obj->lid);
2765
2766 // Update expiration date of line
2767 $expirationdate = $this->db->jdate($obj->date_fin_validite);
2768 $duration_value = preg_replace('/[^0-9]/', '', $obj->duration);
2769 $duration_unit = preg_replace('/\d/', '', $obj->duration);
2770 //var_dump($expirationdate.' '.$enddatetoscan);
2771
2772 // Load linked ->linkedObjects (objects linked)
2773 // @TODO Comment this line and then make the search if there is n open invoice(s) by doing a dedicated SQL COUNT request to fill $contractcanceled.
2774 $object->fetchObjectLinked(null, '', null, '', 'OR', 1, 'sourcetype', 1);
2775
2776 // Test if there is at least 1 open invoice
2777 if (isset($object->linkedObjects['facture']) && is_array($object->linkedObjects['facture']) && count($object->linkedObjects['facture']) > 0) {
2778 // Sort array of linked invoices by ascending date
2779 usort($object->linkedObjects['facture'], array('Contrat', 'contractCmpDate'));
2780 //dol_sort_array($object->linkedObjects['facture'], 'date');
2781
2782 $someinvoicenotpaid = 0;
2783 foreach ($object->linkedObjects['facture'] as $idinvoice => $invoice) {
2784 if ($invoice->statut == Facture::STATUS_DRAFT) {
2785 continue;
2786 } // Draft invoice are not invoice not paid
2787
2788 if (empty($invoice->paye)) {
2789 $someinvoicenotpaid++;
2790 }
2791 }
2792 if ($someinvoicenotpaid) {
2793 $this->output .= 'Contract '.$object->ref.' is qualified for renewal but there is '.$someinvoicenotpaid.' invoice(s) unpayed so we cancel renewal'."\n";
2794 $contractignored[$object->id] = $object->ref;
2795 continue;
2796 }
2797 }
2798
2799 if ($expirationdate && $expirationdate < $enddatetoscan) {
2800 dol_syslog("Define the newdate of end of services from expirationdate=".$expirationdate);
2801 $newdate = $expirationdate;
2802 $protecti = 0; // $protecti is to avoid infinite loop
2803 while ($newdate < $enddatetoscan && $protecti < 1000) {
2804 $newdate = dol_time_plus_duree($newdate, (int) $duration_value, $duration_unit);
2805 $protecti++;
2806 }
2807
2808 if ($protecti < 1000) { // If not, there is a pb
2809 // We will update the end of date of contrat, so first we refresh contract data
2810 dol_syslog("We will update the end of date of contract with newdate = ".dol_print_date($newdate, 'dayhourrfc'));
2811
2812 $this->db->begin();
2813
2814 $errorforlocaltransaction = 0;
2815
2816 $label = 'Renewal of contrat '.$object->ref.' line '.$obj->lid;
2817 $comment = 'Renew date of contract '.$object->ref.' line '.$obj->lid.' by doAutoRenewContracts';
2818
2819 $sqlupdate = 'UPDATE '.MAIN_DB_PREFIX."contratdet SET date_fin_validite = '".$this->db->idate($newdate)."'";
2820 $sqlupdate .= ' WHERE rowid = '.((int) $obj->lid);
2821 $resqlupdate = $this->db->query($sqlupdate);
2822 if ($resqlupdate) {
2823 $contractlineprocessed[$obj->lid] = $object->ref;
2824
2825 $actioncode = 'RENEW_CONTRACT';
2826 $now = dol_now();
2827
2828 // Create an event
2829 $actioncomm = new ActionComm($this->db);
2830 $actioncomm->type_code = 'AC_OTH_AUTO'; // Type of event ('AC_OTH', 'AC_OTH_AUTO', 'AC_XXX'...)
2831 $actioncomm->code = 'AC_'.$actioncode;
2832 $actioncomm->label = $label;
2833 $actioncomm->datep = $now;
2834 $actioncomm->datef = $now;
2835 $actioncomm->percentage = -1; // Not applicable
2836 $actioncomm->socid = $object->socid;
2837 $actioncomm->authorid = $user->id; // User saving action
2838 $actioncomm->userownerid = $user->id; // Owner of action
2839 $actioncomm->fk_element = $object->id;
2840 $actioncomm->elementtype = 'contract';
2841 $actioncomm->note_private = $comment;
2842
2843 $ret = $actioncomm->create($user); // User creating action
2844 } else {
2845 $contracterror[$object->id] = $object->ref;
2846
2847 $error++;
2848 $errorforlocaltransaction++;
2849 $this->error = $this->db->lasterror();
2850 }
2851
2852 if (! $errorforlocaltransaction) {
2853 $this->db->commit();
2854 } else {
2855 $this->db->rollback();
2856 }
2857 } else {
2858 $error++;
2859 $this->error = "Bad value for newdate in doAutoRenewContracts - expirationdate=".$expirationdate." enddatetoscan=".$enddatetoscan." duration_value=".$duration_value." duration_unit=".$duration_value;
2860 dol_syslog($this->error, LOG_ERR);
2861 }
2862 }
2863 }
2864 $i++;
2865 }
2866 } else {
2867 $error++;
2868 $this->error = $this->db->lasterror();
2869 }
2870
2871 $this->output .= count($contractlineprocessed).' contract line(s) with end date before '.dol_print_date($enddatetoscan, 'day').' were renewed'.(count($contractlineprocessed) > 0 ? ' : '.implode(',', $contractlineprocessed) : '');
2872
2873 return ($error ? 1 : 0);
2874 }
2875
2883 public static function contractCmpDate($a, $b)
2884 {
2885 if ($a->date == $b->date) {
2886 return strcmp((string) $a->id, (string) $b->id);
2887 }
2888 return ($a->date < $b->date) ? -1 : 1;
2889 }
2890
2898 public function getKanbanView($option = '', $arraydata = null)
2899 {
2900 global $langs;
2901
2902 $selected = (empty($arraydata['selected']) ? 0 : $arraydata['selected']);
2903
2904 $return = '<div class="box-flex-item box-flex-grow-zero">';
2905 $return .= '<div class="info-box info-box-sm">';
2906 $return .= '<span class="info-box-icon bg-infobox-action">';
2907 $return .= img_picto('', $this->picto);
2908 $return .= '</span>';
2909 $return .= '<div class="info-box-content">';
2910 $return .= '<span class="info-box-ref inline-block tdoverflowmax150 valignmiddle">'.(method_exists($this, 'getNomUrl') ? $this->getNomUrl() : $this->ref).'</span>';
2911 if ($selected >= 0) {
2912 $return .= '<input id="cb'.$this->id.'" class="flat checkforselect fright" type="checkbox" name="toselect[]" value="'.$this->id.'"'.($selected ? ' checked="checked"' : '').'>';
2913 }
2914 if (!empty($arraydata['thirdparty'])) {
2915 $tmpthirdparty = $arraydata['thirdparty'];
2916 $return .= '<br><div class="info-box-label inline-block valignmiddle">'.$tmpthirdparty->getNomUrl(1).'</div>';
2917 }
2918 if (property_exists($this, 'date_contrat')) {
2919 $return .= '<br><span class="opacitymedium valignmiddle">'.$langs->trans("DateContract").' : </span><span class="info-box-label valignmiddle">'.dol_print_date($this->date_contrat, 'day').'</span>';
2920 }
2921 if (method_exists($this, 'getLibStatut')) {
2922 $return .= '<br><div class="info-box-status valignmiddle">'.$this->getLibStatut(7).'</div>';
2923 }
2924 $return .= '</div>';
2925 $return .= '</div>';
2926 $return .= '</div>';
2927
2928 return $return;
2929 }
2930
2931 // @Todo getLibSignedStatus, LibSignedStatus
2932
2942 public function setSignedStatus(User $user, int $status = 0, int $notrigger = 0, $triggercode = ''): int
2943 {
2944 return $this->setSignedStatusCommon($user, $status, $notrigger, $triggercode);
2945 }
2946}
2947
2948
2953{
2957 public $element = 'contratdet';
2958
2962 public $table_element = 'contratdet';
2963
2967 public $parent_element = 'contrat';
2968
2972 public $fk_parent_attribute = 'fk_contrat';
2973
2978 public $element_for_permission = 'contrat';
2979
2983 public $id;
2984
2988 public $ref;
2989
2993 public $fk_contrat;
2994
2998 public $fk_product;
2999
3000 public $statut; // 0 inactive, 4 active, 5 closed
3001 public $type; // 0 for product, 1 for service
3002
3007 public $label;
3008
3013 public $libelle;
3014
3018 public $description;
3019
3020 public $product_type; // 0 for product, 1 for service
3021 public $product_ref;
3022 public $product_label;
3023
3024 public $date_commande;
3025
3026 public $date_start; // date start planned
3027 public $date_start_real; // date start real
3028 public $date_end; // date end planned
3029 public $date_end_real; // date end real
3030
3031 public $tva_tx;
3032 public $vat_src_code;
3033 public $localtax1_tx;
3034 public $localtax2_tx;
3035 public $localtax1_type; // Local tax 1 type
3036 public $localtax2_type; // Local tax 2 type
3037 public $qty;
3038 public $remise_percent;
3039 public $remise;
3040
3044 public $fk_remise_except;
3045
3050 public $subprice;
3051
3057 public $price;
3058
3062 public $price_ht;
3063
3064 public $total_ht;
3065 public $total_tva;
3066 public $total_localtax1;
3067 public $total_localtax2;
3068 public $total_ttc;
3069
3073 public $fk_fournprice;
3074
3075 public $pa_ht;
3076
3080 public $info_bits;
3081
3085 public $fk_user_author;
3086
3090 public $fk_user_ouverture;
3091
3095 public $fk_user_cloture;
3096
3100 public $commentaire;
3101
3102
3106 public $rang = 0;
3107
3108
3109 const STATUS_INITIAL = 0;
3110 const STATUS_OPEN = 4;
3111 const STATUS_CLOSED = 5;
3112
3113
3114 // BEGIN MODULEBUILDER PROPERTIES
3118 public $fields = array(
3119 'rowid' => array('type' => 'integer', 'label' => 'TechnicalID', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 10),
3120 'entity' => array('type' => 'integer', 'label' => 'Entity', 'default' => '1', 'enabled' => 1, 'visible' => -2, 'notnull' => 1, 'position' => 30, 'index' => 1),
3121 'tms' => array('type' => 'timestamp', 'label' => 'DateModification', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 35),
3122 'qty' => array('type' => 'integer', 'label' => 'Quantity', 'enabled' => 1, 'visible' => 1, 'notnull' => 1, 'position' => 35, 'isameasure' => 1),
3123 'total_ht' => array('type' => 'integer', 'label' => 'AmountHT', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 36, 'isameasure' => 1),
3124 'total_tva' => array('type' => 'integer', 'label' => 'AmountVAT', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 37, 'isameasure' => 1),
3125 'total_ttc' => array('type' => 'integer', 'label' => 'AmountTTC', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 38, 'isameasure' => 1),
3126 //'datec' =>array('type'=>'datetime', 'label'=>'DateCreation', 'enabled'=>1, 'visible'=>-1, 'position'=>40),
3127 //'fk_soc' =>array('type'=>'integer:Societe:societe/class/societe.class.php', 'label'=>'ThirdParty', 'enabled'=>1, 'visible'=>-1, 'notnull'=>1, 'position'=>70),
3128 'fk_contrat' => array('type' => 'integer:Contrat:contrat/class/contrat.class.php', 'label' => 'Contract', 'enabled' => 1, 'visible' => -1, 'notnull' => 1, 'position' => 70),
3129 'fk_product' => array('type' => 'integer:Product:product/class/product.class.php:1', 'label' => 'Product', 'enabled' => 1, 'visible' => -1, 'position' => 75),
3130 //'fk_user_author' =>array('type'=>'integer:User:user/class/user.class.php', 'label'=>'Fk user author', 'enabled'=>1, 'visible'=>-1, 'notnull'=>1, 'position'=>90),
3131 'note_private' => array('type' => 'html', 'label' => 'NotePublic', 'enabled' => 1, 'visible' => 0, 'position' => 105),
3132 'note_public' => array('type' => 'html', 'label' => 'NotePrivate', 'enabled' => 1, 'visible' => 0, 'position' => 110),
3133 //'model_pdf' =>array('type'=>'varchar(255)', 'label'=>'Model pdf', 'enabled'=>1, 'visible'=>0, 'position'=>115),
3134 //'import_key' =>array('type'=>'varchar(14)', 'label'=>'ImportId', 'enabled'=>1, 'visible'=>-2, 'position'=>120),
3135 //'extraparams' =>array('type'=>'varchar(255)', 'label'=>'Extraparams', 'enabled'=>1, 'visible'=>-1, 'position'=>125),
3136 'fk_user_ouverture' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserStartingService', 'enabled' => 1, 'visible' => -2, 'notnull' => -1, 'position' => 135),
3137 'fk_user_cloture' => array('type' => 'integer:User:user/class/user.class.php', 'label' => 'UserClosingService', 'enabled' => 1, 'visible' => -2, 'notnull' => -1, 'position' => 135),
3138 'statut' => array('type' => 'smallint(6)', 'label' => 'Statut', 'enabled' => 1, 'visible' => -1, 'position' => 500, 'arrayofkeyval' => array(0 => 'Draft', 4 => 'Open', 5 => 'Closed')),
3139 'rang' => array('type' => 'integer', 'label' => 'Rank', 'enabled' => 1, 'visible' => 0, 'position' => 500, 'default' => '0')
3140 );
3141 // END MODULEBUILDER PROPERTIES
3142
3143
3149 public function __construct($db)
3150 {
3151 $this->db = $db;
3152 }
3153
3154
3161 public function getLibStatut($mode)
3162 {
3163 return $this->LibStatut($this->statut, $mode, ((!empty($this->date_end)) ? ($this->date_end < dol_now() ? 1 : 0) : -1));
3164 }
3165
3166 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
3177 public static function LibStatut($status, $mode, $expired = -1, $moreatt = '', $morelabel = '')
3178 {
3179 // phpcs:enable
3180 global $langs;
3181 $langs->load("contracts");
3182
3183 if ($status == self::STATUS_INITIAL) {
3184 $labelStatus = $langs->transnoentities("ServiceStatusInitial");
3185 $labelStatusShort = $langs->transnoentities("ServiceStatusInitial");
3186 } elseif ($status == self::STATUS_OPEN && $expired == -1) {
3187 $labelStatus = $langs->transnoentities("ServiceStatusRunning");
3188 $labelStatusShort = $langs->transnoentities("ServiceStatusRunning");
3189 } elseif ($status == self::STATUS_OPEN && $expired == 0) {
3190 $labelStatus = $langs->transnoentities("ServiceStatusNotLate");
3191 $labelStatusShort = $langs->transnoentities("ServiceStatusNotLateShort");
3192 } elseif ($status == self::STATUS_OPEN && $expired == 1) {
3193 $labelStatus = $langs->transnoentities("ServiceStatusLate");
3194 $labelStatusShort = $langs->transnoentities("ServiceStatusLateShort");
3195 } elseif ($status == self::STATUS_CLOSED) {
3196 $labelStatus = $langs->transnoentities("ServiceStatusClosed");
3197 $labelStatusShort = $langs->transnoentities("ServiceStatusClosed");
3198 }
3199
3200 $statusType = 'status'.$status;
3201 if ($status == self::STATUS_OPEN && $expired == 1) {
3202 $statusType = 'status1';
3203 }
3204 if ($status == self::STATUS_CLOSED) {
3205 $statusType = 'status6';
3206 }
3207
3208 $params = array();
3209 $reg = array();
3210 if (preg_match('/class="(.*)"/', $moreatt, $reg)) {
3211 $params = array('badgeParams' => array('css' => $reg[1]));
3212 }
3213 return dolGetStatus($labelStatus.($morelabel ? ' '.$morelabel : ''), $labelStatusShort.($morelabel ? ' '.$morelabel : ''), '', $statusType, $mode, '', $params);
3214 }
3215
3222 public function getTooltipContentArray($params)
3223 {
3224 global $conf, $langs, $user;
3225
3226 $datas = [];
3227 $datas['label'] = $langs->trans("ShowContractOfService").': '.$this->label;
3228 if (empty($datas['label'])) {
3229 $datas['label'] = $this->description;
3230 }
3231
3232 return $datas;
3233 }
3234
3242 public function getNomUrl($withpicto = 0, $maxlength = 0)
3243 {
3244 global $langs;
3245
3246 $result = '';
3247 $label = $langs->trans("ShowContractOfService").': '.$this->label;
3248 if (empty($label)) {
3249 $label = $this->description;
3250 }
3251 $classfortooltip = 'classfortooltip';
3252 $dataparams = '';
3253 if (getDolGlobalInt('MAIN_ENABLE_AJAX_TOOLTIP')) {
3254 $params = [
3255 'id' => $this->fk_contrat,
3256 'objecttype' => $this->element,
3257 ];
3258 $classfortooltip = 'classforajaxtooltip';
3259 $dataparams = ' data-params="'.dol_escape_htmltag(json_encode($params)).'"';
3260 $label = '';
3261 }
3262
3263 $link = '<a href="'.DOL_URL_ROOT.'/contrat/card.php?id='.$this->fk_contrat.'"';
3264 $link .= ($label ? ' title="'.dol_escape_htmltag($label, 1).'"' : ' title="tocomplete"');
3265 $link .= $dataparams.' class="'.$classfortooltip.'">';
3266 $linkend = '</a>';
3267
3268 $picto = 'service';
3269 if ($this->type == 0) {
3270 $picto = 'product';
3271 }
3272
3273 if ($withpicto) {
3274 $result .= ($link.img_object($label, $picto, $dataparams.' class="'.$classfortooltip.'"').$linkend);
3275 }
3276 if ($withpicto && $withpicto != 2) {
3277 $result .= ' ';
3278 }
3279 if ($withpicto != 2) {
3280 $result .= $link.($this->product_ref ? $this->product_ref.' ' : '').($this->label ? $this->label : $this->description).$linkend;
3281 }
3282 return $result;
3283 }
3284
3292 public function fetch($id, $ref = '')
3293 {
3294 // Check parameters
3295 if (empty($id) && empty($ref)) {
3296 return -1;
3297 }
3298
3299 $sql = "SELECT";
3300 $sql .= " t.rowid,";
3301 $sql .= " t.tms,";
3302 $sql .= " t.fk_contrat,";
3303 $sql .= " t.fk_product,";
3304 $sql .= " t.statut,";
3305 $sql .= " t.label,"; // This field is not used. Only label of product
3306 $sql .= " p.ref as product_ref,";
3307 $sql .= " p.label as product_label,";
3308 $sql .= " p.description as product_desc,";
3309 $sql .= " p.fk_product_type as product_type,";
3310 $sql .= " t.description,";
3311 $sql .= " t.date_commande,";
3312 $sql .= " t.date_ouverture_prevue as date_start,";
3313 $sql .= " t.date_ouverture as date_start_real,";
3314 $sql .= " t.date_fin_validite as date_end,";
3315 $sql .= " t.date_cloture as date_end_real,";
3316 $sql .= " t.tva_tx,";
3317 $sql .= " t.vat_src_code,";
3318 $sql .= " t.localtax1_tx,";
3319 $sql .= " t.localtax2_tx,";
3320 $sql .= " t.localtax1_type,";
3321 $sql .= " t.localtax2_type,";
3322 $sql .= " t.qty,";
3323 $sql .= " t.remise_percent,";
3324 $sql .= " t.remise,";
3325 $sql .= " t.fk_remise_except,";
3326 $sql .= " t.subprice,";
3327 $sql .= " t.price_ht,";
3328 $sql .= " t.total_ht,";
3329 $sql .= " t.total_tva,";
3330 $sql .= " t.total_localtax1,";
3331 $sql .= " t.total_localtax2,";
3332 $sql .= " t.total_ttc,";
3333 $sql .= " t.fk_product_fournisseur_price as fk_fournprice,";
3334 $sql .= " t.buy_price_ht as pa_ht,";
3335 $sql .= " t.info_bits,";
3336 $sql .= " t.fk_user_author,";
3337 $sql .= " t.fk_user_ouverture,";
3338 $sql .= " t.fk_user_cloture,";
3339 $sql .= " t.commentaire,";
3340 $sql .= " t.fk_unit,";
3341 $sql .= " t.rang";
3342 $sql .= " FROM ".MAIN_DB_PREFIX."contratdet as t LEFT JOIN ".MAIN_DB_PREFIX."product as p ON p.rowid = t.fk_product";
3343 if ($id) {
3344 $sql .= " WHERE t.rowid = ".((int) $id);
3345 }
3346 if ($ref) {
3347 $sql .= " WHERE t.rowid = '".$this->db->escape($ref)."'";
3348 }
3349
3350 dol_syslog(get_class($this)."::fetch", LOG_DEBUG);
3351 $resql = $this->db->query($sql);
3352 if ($resql) {
3353 if ($this->db->num_rows($resql)) {
3354 $obj = $this->db->fetch_object($resql);
3355
3356 $this->id = $obj->rowid;
3357 $this->ref = $obj->rowid;
3358
3359 $this->tms = $this->db->jdate($obj->tms);
3360 $this->fk_contrat = $obj->fk_contrat;
3361 $this->fk_product = $obj->fk_product;
3362 $this->statut = $obj->statut;
3363 $this->product_ref = $obj->product_ref;
3364 $this->product_label = $obj->product_label;
3365 $this->product_type = $obj->product_type;
3366 $this->label = $obj->label; // deprecated. We do not use this field. Only ref and label of product, and description of contract line
3367 $this->description = $obj->description;
3368 $this->date_commande = $this->db->jdate($obj->date_commande);
3369
3370 $this->date_start = $this->db->jdate($obj->date_start);
3371 $this->date_start_real = $this->db->jdate($obj->date_start_real);
3372 $this->date_end = $this->db->jdate($obj->date_end);
3373 $this->date_end_real = $this->db->jdate($obj->date_end_real);
3374 // For backward compatibility
3375 //$this->date_ouverture_prevue = $this->db->jdate($obj->date_ouverture_prevue);
3376 //$this->date_ouverture = $this->db->jdate($obj->date_ouverture);
3377 //$this->date_fin_validite = $this->db->jdate($obj->date_fin_validite);
3378 //$this->date_cloture = $this->db->jdate($obj->date_cloture);
3379
3380 $this->tva_tx = $obj->tva_tx;
3381 $this->vat_src_code = $obj->vat_src_code;
3382 $this->localtax1_tx = $obj->localtax1_tx;
3383 $this->localtax2_tx = $obj->localtax2_tx;
3384 $this->localtax1_type = $obj->localtax1_type;
3385 $this->localtax2_type = $obj->localtax2_type;
3386 $this->qty = $obj->qty;
3387 $this->remise_percent = $obj->remise_percent;
3388 $this->fk_remise_except = $obj->fk_remise_except;
3389 $this->subprice = $obj->subprice;
3390 $this->price_ht = $obj->price_ht;
3391 $this->total_ht = $obj->total_ht;
3392 $this->total_tva = $obj->total_tva;
3393 $this->total_localtax1 = $obj->total_localtax1;
3394 $this->total_localtax2 = $obj->total_localtax2;
3395 $this->total_ttc = $obj->total_ttc;
3396 $this->info_bits = $obj->info_bits;
3397 $this->fk_user_author = $obj->fk_user_author;
3398 $this->fk_user_ouverture = $obj->fk_user_ouverture;
3399 $this->fk_user_cloture = $obj->fk_user_cloture;
3400 $this->commentaire = $obj->commentaire;
3401 $this->fk_fournprice = $obj->fk_fournprice;
3402
3403 $marginInfos = getMarginInfos($obj->subprice, $obj->remise_percent, $obj->tva_tx, $obj->localtax1_tx, $obj->localtax2_tx, $this->fk_fournprice, $obj->pa_ht);
3404 $this->pa_ht = $marginInfos[0];
3405 $this->fk_unit = $obj->fk_unit;
3406
3407 $this->rang = $obj->rang;
3408
3409 $this->fetch_optionals();
3410 }
3411
3412 $this->db->free($resql);
3413
3414 return 1;
3415 } else {
3416 $this->error = "Error ".$this->db->lasterror();
3417 return -1;
3418 }
3419 }
3420
3421
3429 public function update($user, $notrigger = 0)
3430 {
3431 global $mysoc;
3432
3433 $error = 0;
3434
3435 // Clean parameters
3436 $this->fk_contrat = (int) $this->fk_contrat;
3437 $this->fk_product = (int) $this->fk_product;
3438 $this->statut = (int) $this->statut;
3439 $this->label = trim($this->label);
3440 $this->description = trim($this->description);
3441 $this->vat_src_code = trim($this->vat_src_code);
3442 $this->tva_tx = trim((string) $this->tva_tx);
3443 $this->localtax1_tx = trim($this->localtax1_tx);
3444 $this->localtax2_tx = trim($this->localtax2_tx);
3445 $this->qty = (float) $this->qty;
3446 $this->remise_percent = trim((string) $this->remise_percent);
3447 $this->fk_remise_except = (int) $this->fk_remise_except;
3448 $this->subprice = (float) price2num($this->subprice);
3449 $this->price_ht = (float) price2num($this->price_ht);
3450 $this->info_bits = (int) $this->info_bits;
3451 $this->fk_user_author = (int) $this->fk_user_author;
3452 $this->fk_user_ouverture = (int) $this->fk_user_ouverture;
3453 $this->fk_user_cloture = (int) $this->fk_user_cloture;
3454 $this->commentaire = trim($this->commentaire);
3455 $this->rang = (int) $this->rang;
3456 //if (empty($this->subprice)) $this->subprice = 0;
3457 if (empty($this->price_ht)) {
3458 $this->price_ht = 0;
3459 }
3460 if (empty($this->total_ht)) {
3461 $this->total_ht = 0;
3462 }
3463 if (empty($this->total_tva)) {
3464 $this->total_tva = 0;
3465 }
3466 if (empty($this->total_ttc)) {
3467 $this->total_ttc = 0;
3468 }
3469 if (empty($this->localtax1_tx)) {
3470 $this->localtax1_tx = 0;
3471 }
3472 if (empty($this->localtax2_tx)) {
3473 $this->localtax2_tx = 0;
3474 }
3475 if (empty($this->remise_percent)) {
3476 $this->remise_percent = 0;
3477 }
3478
3479 // Calcul du total TTC et de la TVA pour la ligne a partir de
3480 // qty, pu, remise_percent et txtva
3481 // TRES IMPORTANT: C'est au moment de l'insertion ligne qu'on doit stocker
3482 // la part ht, tva et ttc, et ce au niveau de la ligne qui a son propre taux tva.
3483 $localtaxes_type = getLocalTaxesFromRate($this->tva_tx, 0, $this->thirdparty, $mysoc);
3484
3485 $tabprice = calcul_price_total($this->qty, $this->price_ht, $this->remise_percent, $this->tva_tx, $this->localtax1_tx, $this->localtax2_tx, 0, 'HT', 0, 1, $mysoc, $localtaxes_type);
3486 $this->total_ht = $tabprice[0];
3487 $this->total_tva = $tabprice[1];
3488 $this->total_ttc = $tabprice[2];
3489 $this->total_localtax1 = $tabprice[9];
3490 $this->total_localtax2 = $tabprice[10];
3491
3492 if (empty($this->pa_ht)) {
3493 $this->pa_ht = 0;
3494 }
3495
3496 // if buy price not defined, define buyprice as configured in margin admin
3497 if ($this->pa_ht == 0) {
3498 $result = $this->defineBuyPrice($this->subprice, $this->remise_percent, $this->fk_product);
3499 if ($result < 0) {
3500 return -1;
3501 } else {
3502 $this->pa_ht = $result;
3503 }
3504 }
3505
3506 // $this->oldcopy should have been set by the caller of update (here properties were already modified)
3507 if (empty($this->oldcopy)) {
3508 dol_syslog("this->oldcopy should have been set by the caller of update (here properties were already modified)", LOG_WARNING);
3509 $this->oldcopy = dol_clone($this, 2);
3510 }
3511
3512 $this->db->begin();
3513
3514 // Update request
3515 $sql = "UPDATE ".MAIN_DB_PREFIX."contratdet SET";
3516 $sql .= " fk_contrat = ".((int) $this->fk_contrat).",";
3517 $sql .= " fk_product = ".($this->fk_product ? ((int) $this->fk_product) : 'null').",";
3518 $sql .= " statut = ".((int) $this->statut).",";
3519 $sql .= " label = '".$this->db->escape($this->label)."',";
3520 $sql .= " description = '".$this->db->escape($this->description)."',";
3521 $sql .= " date_commande = ".($this->date_commande != '' ? "'".$this->db->idate($this->date_commande)."'" : "null").",";
3522 $sql .= " date_ouverture_prevue = ".($this->date_start != '' ? "'".$this->db->idate($this->date_start)."'" : "null").",";
3523 $sql .= " date_ouverture = ".($this->date_start_real != '' ? "'".$this->db->idate($this->date_start_real)."'" : "null").",";
3524 $sql .= " date_fin_validite = ".($this->date_end != '' ? "'".$this->db->idate($this->date_end)."'" : "null").",";
3525 $sql .= " date_cloture = ".($this->date_end_real != '' ? "'".$this->db->idate($this->date_end_real)."'" : "null").",";
3526 $sql .= " vat_src_code = '".$this->db->escape($this->vat_src_code)."',";
3527 $sql .= " tva_tx = ".price2num($this->tva_tx).",";
3528 $sql .= " localtax1_tx = ".price2num($this->localtax1_tx).",";
3529 $sql .= " localtax2_tx = ".price2num($this->localtax2_tx).",";
3530 $sql .= " qty = ".price2num($this->qty).",";
3531 $sql .= " remise_percent = ".price2num($this->remise_percent).",";
3532 $sql .= " remise = ".($this->remise ? price2num($this->remise) : "null").",";
3533 $sql .= " fk_remise_except = ".($this->fk_remise_except > 0 ? $this->fk_remise_except : "null").",";
3534 $sql .= " subprice = ".($this->subprice != '' ? $this->subprice : "null").",";
3535 $sql .= " price_ht = ".($this->price_ht != '' ? $this->price_ht : "null").",";
3536 $sql .= " total_ht = ".$this->total_ht.",";
3537 $sql .= " total_tva = ".$this->total_tva.",";
3538 $sql .= " total_localtax1 = ".$this->total_localtax1.",";
3539 $sql .= " total_localtax2 = ".$this->total_localtax2.",";
3540 $sql .= " total_ttc = ".$this->total_ttc.",";
3541 $sql .= " fk_product_fournisseur_price = ".(!empty($this->fk_fournprice) ? $this->fk_fournprice : "NULL").",";
3542 $sql .= " buy_price_ht = '".price2num($this->pa_ht)."',";
3543 $sql .= " info_bits = '".$this->db->escape($this->info_bits)."',";
3544 $sql .= " fk_user_author = ".($this->fk_user_author >= 0 ? $this->fk_user_author : "NULL").",";
3545 $sql .= " fk_user_ouverture = ".($this->fk_user_ouverture > 0 ? $this->fk_user_ouverture : "NULL").",";
3546 $sql .= " fk_user_cloture = ".($this->fk_user_cloture > 0 ? $this->fk_user_cloture : "NULL").",";
3547 $sql .= " commentaire = '".$this->db->escape($this->commentaire)."',";
3548 $sql .= " fk_unit = ".(!$this->fk_unit ? 'NULL' : $this->fk_unit).",";
3549 $sql .= " rang = ".(empty($this->rang) ? '0' : (int) $this->rang);
3550 $sql .= " WHERE rowid = ".((int) $this->id);
3551
3552 dol_syslog(get_class($this)."::update", LOG_DEBUG);
3553 $resql = $this->db->query($sql);
3554 if (!$resql) {
3555 $this->error = "Error ".$this->db->lasterror();
3556 $error++;
3557 }
3558
3559 if (!$error) { // For avoid conflicts if trigger used
3560 $result = $this->insertExtraFields();
3561 if ($result < 0) {
3562 $error++;
3563 }
3564 }
3565
3566 // If we change a planned date (start or end) of one contract line, sync dates for all other services too
3567 if (!$error && getDolGlobalString('CONTRACT_SYNC_PLANNED_DATE_OF_SERVICES')) {
3568 dol_syslog(get_class($this)."::update CONTRACT_SYNC_PLANNED_DATE_OF_SERVICES is on so we update date for all lines", LOG_DEBUG);
3569
3570 if ($this->date_start != $this->oldcopy->date_start) {
3571 $sql = 'UPDATE '.MAIN_DB_PREFIX.'contratdet SET';
3572 $sql .= " date_ouverture_prevue = ".($this->date_start != '' ? "'".$this->db->idate($this->date_start)."'" : "null");
3573 $sql .= " WHERE fk_contrat = ".((int) $this->fk_contrat);
3574
3575 $resql = $this->db->query($sql);
3576 if (!$resql) {
3577 $error++;
3578 $this->error = "Error ".$this->db->lasterror();
3579 }
3580 }
3581 if ($this->date_end != $this->oldcopy->date_end) {
3582 $sql = 'UPDATE '.MAIN_DB_PREFIX.'contratdet SET';
3583 $sql .= " date_fin_validite = ".($this->date_end != '' ? "'".$this->db->idate($this->date_end)."'" : "null");
3584 $sql .= " WHERE fk_contrat = ".((int) $this->fk_contrat);
3585
3586 $resql = $this->db->query($sql);
3587 if (!$resql) {
3588 $error++;
3589 $this->error = "Error ".$this->db->lasterror();
3590 }
3591 }
3592 }
3593
3594 if (!$error && !$notrigger) {
3595 // Call trigger
3596 $result = $this->call_trigger('LINECONTRACT_MODIFY', $user);
3597 if ($result < 0) {
3598 $error++;
3599 $this->db->rollback();
3600 }
3601 // End call triggers
3602 }
3603
3604 if (!$error) {
3605 $this->db->commit();
3606 return 1;
3607 } else {
3608 $this->db->rollback();
3609 $this->errors[] = $this->error;
3610 return -1;
3611 }
3612 }
3613
3614
3615 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
3622 public function update_total()
3623 {
3624 // phpcs:enable
3625 $this->db->begin();
3626
3627 // Mise a jour ligne en base
3628 $sql = "UPDATE ".MAIN_DB_PREFIX."contratdet SET";
3629 $sql .= " total_ht=".price2num($this->total_ht, 'MT');
3630 $sql .= ",total_tva=".price2num($this->total_tva, 'MT');
3631 $sql .= ",total_localtax1=".price2num($this->total_localtax1, 'MT');
3632 $sql .= ",total_localtax2=".price2num($this->total_localtax2, 'MT');
3633 $sql .= ",total_ttc=".price2num($this->total_ttc, 'MT');
3634 $sql .= " WHERE rowid = ".((int) $this->id);
3635
3636 dol_syslog(get_class($this)."::update_total", LOG_DEBUG);
3637
3638 $resql = $this->db->query($sql);
3639 if ($resql) {
3640 $this->db->commit();
3641 return 1;
3642 } else {
3643 $this->error = $this->db->error();
3644 $this->db->rollback();
3645 return -2;
3646 }
3647 }
3648
3649
3656 public function insert($notrigger = 0)
3657 {
3658 global $user;
3659
3660 $error = 0;
3661
3662 // Insertion dans la base
3663 $sql = "INSERT INTO ".MAIN_DB_PREFIX."contratdet";
3664 $sql .= " (fk_contrat, label, description, fk_product, qty, vat_src_code, tva_tx,";
3665 $sql .= " localtax1_tx, localtax2_tx, localtax1_type, localtax2_type, remise_percent, subprice,";
3666 $sql .= " total_ht, total_tva, total_localtax1, total_localtax2, total_ttc,";
3667 $sql .= " info_bits,";
3668 $sql .= " rang,";
3669 $sql .= " price_ht, remise, fk_product_fournisseur_price, buy_price_ht";
3670 if ($this->date_start > 0) {
3671 $sql .= ",date_ouverture_prevue";
3672 }
3673 if ($this->date_end > 0) {
3674 $sql .= ",date_fin_validite";
3675 }
3676 $sql .= ") VALUES ($this->fk_contrat, '', '".$this->db->escape($this->description)."',";
3677 $sql .= ($this->fk_product > 0 ? $this->fk_product : "null").",";
3678 $sql .= " '".$this->db->escape($this->qty)."',";
3679 $sql .= " '".$this->db->escape($this->vat_src_code)."',";
3680 $sql .= " '".$this->db->escape($this->tva_tx)."',";
3681 $sql .= " '".$this->db->escape($this->localtax1_tx)."',";
3682 $sql .= " '".$this->db->escape($this->localtax2_tx)."',";
3683 $sql .= " '".$this->db->escape($this->localtax1_type)."',";
3684 $sql .= " '".$this->db->escape($this->localtax2_type)."',";
3685 $sql .= " ".price2num($this->remise_percent).",".price2num($this->subprice).",";
3686 $sql .= " ".price2num($this->total_ht).",".price2num($this->total_tva).",".price2num($this->total_localtax1).",".price2num($this->total_localtax2).",".price2num($this->total_ttc).",";
3687 $sql .= " '".$this->db->escape($this->info_bits)."',";
3688 $sql .= " ".(empty($this->rang) ? '0' : (int) $this->rang).",";
3689 $sql .= " ".price2num($this->price_ht).",".price2num($this->remise).",";
3690 if ($this->fk_fournprice > 0) {
3691 $sql .= ' '.((int) $this->fk_fournprice).',';
3692 } else {
3693 $sql .= ' null,';
3694 }
3695 if ($this->pa_ht > 0) {
3696 $sql .= ' '.((float) price2num($this->pa_ht));
3697 } else {
3698 $sql .= ' null';
3699 }
3700 if ($this->date_start > 0) {
3701 $sql .= ",'".$this->db->idate($this->date_start)."'";
3702 }
3703 if ($this->date_end > 0) {
3704 $sql .= ",'".$this->db->idate($this->date_end)."'";
3705 }
3706 $sql .= ")";
3707
3708 dol_syslog(get_class($this)."::insert", LOG_DEBUG);
3709
3710 $resql = $this->db->query($sql);
3711 if ($resql) {
3712 $this->id = $this->db->last_insert_id(MAIN_DB_PREFIX.'contratdet');
3713
3714 // Insert of extrafields
3715 if (!$error) {
3716 $result = $this->insertExtraFields();
3717 if ($result < 0) {
3718 $this->db->rollback();
3719 return -1;
3720 }
3721 }
3722
3723 if (!$notrigger) {
3724 // Call trigger
3725 $result = $this->call_trigger('LINECONTRACT_INSERT', $user);
3726 if ($result < 0) {
3727 $this->db->rollback();
3728 return -1;
3729 }
3730 // End call triggers
3731 }
3732
3733 $this->db->commit();
3734 return 1;
3735 } else {
3736 $this->db->rollback();
3737 $this->error = $this->db->error()." sql=".$sql;
3738 return -1;
3739 }
3740 }
3741
3742 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
3752 public function active_line($user, $date, $date_end = '', $comment = '')
3753 {
3754 // phpcs:enable
3755 $error = 0;
3756
3757 $this->db->begin();
3758
3759 $this->statut = ContratLigne::STATUS_OPEN;
3760 $this->date_start_real = $date;
3761 $this->date_end = $date_end;
3762 $this->fk_user_ouverture = $user->id;
3763 $this->date_end_real = null;
3764 $this->commentaire = $comment;
3765
3766 $sql = "UPDATE ".MAIN_DB_PREFIX."contratdet SET statut = ".((int) $this->statut).",";
3767 $sql .= " date_ouverture = ".(dol_strlen($this->date_start_real) != 0 ? "'".$this->db->idate($this->date_start_real)."'" : "null").",";
3768 if ($date_end >= 0) {
3769 $sql .= " date_fin_validite = ".(dol_strlen($this->date_end) != 0 ? "'".$this->db->idate($this->date_end)."'" : "null").",";
3770 }
3771 $sql .= " fk_user_ouverture = ".((int) $this->fk_user_ouverture).",";
3772 $sql .= " date_cloture = null,";
3773 $sql .= " commentaire = '".$this->db->escape($comment)."'";
3774 $sql .= " WHERE rowid = ".((int) $this->id)." AND (statut = ".ContratLigne::STATUS_INITIAL." OR statut = ".ContratLigne::STATUS_CLOSED.")";
3775
3776 dol_syslog(get_class($this)."::active_line", LOG_DEBUG);
3777 $resql = $this->db->query($sql);
3778 if ($resql) {
3779 // Call trigger
3780 $result = $this->call_trigger('LINECONTRACT_ACTIVATE', $user);
3781 if ($result < 0) {
3782 $error++;
3783 }
3784 // End call triggers
3785
3786 if (!$error) {
3787 $this->db->commit();
3788 return 1;
3789 } else {
3790 $this->db->rollback();
3791 return -1;
3792 }
3793 } else {
3794 $this->error = $this->db->lasterror();
3795 $this->db->rollback();
3796 return -1;
3797 }
3798 }
3799
3800 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
3810 public function close_line($user, $date_end_real, $comment = '', $notrigger = 0)
3811 {
3812 // phpcs:enable
3813 $this->date_cloture = $date_end_real;
3814 $this->date_end_real = $date_end_real;
3815 $this->user_closing_id = $user->id;
3816 $this->commentaire = $comment;
3817
3818 $error = 0;
3819
3820 // statut actif : 4
3821
3822 $this->db->begin();
3823
3824 $sql = "UPDATE ".MAIN_DB_PREFIX."contratdet SET statut = ".((int) ContratLigne::STATUS_CLOSED).",";
3825 $sql .= " date_cloture = '".$this->db->idate($date_end_real)."',";
3826 $sql .= " fk_user_cloture = ".((int) $user->id).",";
3827 $sql .= " commentaire = '".$this->db->escape($comment)."'";
3828 $sql .= " WHERE rowid = ".((int) $this->id)." AND statut = ".((int) ContratLigne::STATUS_OPEN);
3829
3830 $resql = $this->db->query($sql);
3831 if ($resql) {
3832 if (!$notrigger) {
3833 // Call trigger
3834 $result = $this->call_trigger('LINECONTRACT_CLOSE', $user);
3835 if ($result < 0) {
3836 $error++;
3837 $this->db->rollback();
3838 return -1;
3839 }
3840 // End call triggers
3841 }
3842
3843 $this->db->commit();
3844 return 1;
3845 } else {
3846 $this->error = $this->db->lasterror();
3847 $this->db->rollback();
3848 return -1;
3849 }
3850 }
3851}
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Definition card.php:58
print $langs trans("AuditedSecurityEvents").'</strong >< span class="opacitymedium"></span >< br > status
Or an array listing all the potential status of the object: array: int of the status => translated la...
Definition security.php:626
$object ref
Definition info.php:79
Class to manage agenda events (actions)
Parent class of all other business classes (invoices, contracts, proposals, orders,...
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...
add_object_linked($origin=null, $origin_id=null, $f_user=null, $notrigger=0)
Add an object link into llx_element_element.
defineBuyPrice($unitPrice=0.0, $discountPercent=0.0, $fk_product=0)
Get buy price to use for margin calculation.
commonGenerateDocument($modelspath, $modele, $outputlangs, $hidedetails, $hidedesc, $hideref, $moreparams=null)
Common function for all objects extending CommonObject for generating documents.
fetch_thirdparty($force_thirdparty_id=0)
Load the third party of object, from id $this->socid or $this->fk_soc, into this->thirdparty.
getIdContact($source, $code, $status=0)
Return id of contacts for a source and a contact code.
deleteObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $rowid=0, $f_user=null, $notrigger=0)
Delete all links between an object $this.
setErrorsFromObject($object)
setErrorsFromObject
fetch_project()
Load the project with id $this->fk_project into this->project.
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.
setSignedStatusCommon($user, $status, $notrigger=0, $triggercode='')
Set to a signed status.
static commonReplaceProduct(DoliDB $dbs, $origin_id, $dest_id, array $tables, $ignoreerrors=0)
Function used to replace a product id with another one.
insertExtraFields($trigger='', $userused=null)
Add/Update all extra fields values for the current object.
delete_linked_contact($source='', $code='')
Delete all links between an object $this and all its contacts in llx_element_contact.
call_trigger($triggerName, $user)
Call trigger based on this instance.
add_contact($fk_socpeople, $type_contact, $source='external', $notrigger=0)
Add a link between element $this->element and a contact.
Parent class for class inheritance lines of business objects This class is useless for the moment so ...
Class to manage contracts.
static contractCmpDate($a, $b)
Used to sort lines by date.
createFromClone(User $user, $socid=0, $notrigger=0)
Load an object from its id and create a new one in database.
fetch($id, $ref='', $ref_customer='', $ref_supplier='', $noextrafields=0, $nolines=0)
Load a contract from database.
loadStateBoard()
Load the indicators this->nb for state board.
update_statut($user)
Update statut of contract according to services.
updateline($rowid, $desc, $pu, $qty, $remise_percent, $date_start, $date_end, $tvatx, $localtax1tx=0.0, $localtax2tx=0.0, $date_start_real='', $date_end_real='', $price_base_type='HT', $info_bits=0, $fk_fournprice=null, $pa_ht=0, $array_options=array(), $fk_unit=null, $rang=0)
Mets a jour une ligne de contrat.
getTicketsArray()
Create an array of associated tickets.
setSignedStatus(User $user, int $status=0, int $notrigger=0, $triggercode='')
Set signed status.
generateDocument($modele, $outputlangs, $hidedetails=0, $hidedesc=0, $hideref=0, $moreparams=null)
Create a document onto disk according to template module.
active_line($user, $line_id, $date_start, $date_end='', $comment='')
Activate a contract line.
getTooltipContentArray($params)
getTooltipContentArray
getLinesArray()
Create an array of order lines.
validate(User $user, $force_number='', $notrigger=0)
Validate a contract.
getIdBillingContact()
Return id des contacts clients de facturation.
initAsSpecimen()
Initialise an instance with random values.
create($user)
Create a contract into database.
close_line($user, $line_id, $date_end, $comment='')
Close a contract line.
reopen($user, $notrigger=0)
Unvalidate a contract.
getKanbanView($option='', $arraydata=null)
Return clicable link of object (with eventually picto)
deleteLine($idline, User $user)
Delete a contract line.
array_detail($status=-1)
Return list of line rowid.
addline($desc, $pu_ht, $qty, $txtva, $txlocaltax1, $txlocaltax2, $fk_product, $remise_percent, $date_start, $date_end, $price_base_type='HT', $pu_ttc=0.0, $info_bits=0, $fk_fournprice=null, $pa_ht=0, $array_options=array(), $fk_unit=null, $rang=0)
Ajoute une ligne de contrat en base.
update($user, $notrigger=0)
Update object into database.
closeAll(User $user, $notrigger=0, $comment='')
Close all lines of a contract.
static replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
Function used to replace a thirdparty id with another one.
getListOfContracts($option='all', $status=[], $product_categories=[], $line_status=[])
Return list of other contracts for the same company than current contract.
load_board($user, $mode)
Load indicators for dashboard (this->nbtodo and this->nbtodolate)
LibStatut($status, $mode)
Return the label of a given contrat status.
activateAll($user, $date_start='', $notrigger=0, $comment='', $date_end='')
Open all lines of a contract.
__construct($db)
Constructor.
doAutoRenewContracts($thirdparty_id=0, $delayindaysshort=0)
Action executed by scheduler CAN BE A CRON TASK Loop on each contract lines and update the end of dat...
getNomUrl($withpicto=0, $maxlength=0, $notooltip=0, $save_lastsearch_value=-1)
Return clicable name (with picto eventually)
info($id)
Charge les information d'ordre info dans l'objet contrat.
getLibStatut($mode)
Return label of a contract status.
static replaceProduct(DoliDB $db, $origin_id, $dest_id)
Function used to replace a product id with another one.
fetch_lines($only_services=0, $loadalsotranslation=0, $noextrafields=0)
Load lines array into this->lines.
getIdServiceContact()
Return id des contacts clients de prestation.
getNextNumRef($soc)
Return next contract ref.
Class to manage lines of contracts.
active_line($user, $date, $date_end='', $comment='')
Activate a contract line.
getLibStatut($mode)
Return label of this contract line status.
insert($notrigger=0)
Inserts a contrat line into database.
fetch($id, $ref='')
Load object in memory from database.
close_line($user, $date_end_real, $comment='', $notrigger=0)
Close a contract line.
static LibStatut($status, $mode, $expired=-1, $moreatt='', $morelabel='')
Return label of a contract line status.
update($user, $notrigger=0)
Update database for contract line.
update_total()
Update in database the fields total_xxx of lines Used by migration process.
__construct($db)
Constructor.
getTooltipContentArray($params)
getTooltipContentArray
getNomUrl($withpicto=0, $maxlength=0)
Return clicable name (with picto eventually) for ContratLigne.
Class to manage Dolibarr database access.
Class to manage shipments.
const STATUS_DRAFT
Draft status.
Class to manage products or services.
Class to manage projects.
Class to manage third parties objects (customers, suppliers, prospects...)
Class to manage Dolibarr users.
dol_time_plus_duree($time, $duration_value, $duration_unit, $ruleforendofmonth=0)
Add a delay to a date.
Definition date.lib.php:124
print $script_file $mode $langs defaultlang(is_numeric($duration_value) ? " delay=". $duration_value :"").(is_numeric($duration_value2) ? " after cd cd cd description as description
Only used if Module[ID]Desc translation string is not found.
dol_delete_dir_recursive($dir, $count=0, $nophperrors=0, $onlysub=0, &$countdeleted=0, $indexdatabase=1, $nolog=0)
Remove a directory $dir and its subdirectories (or only files and subdirectories)
dol_dir_list($utf8_path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0, $nbsecondsold=0)
Scan a directory and return a list of files/directories.
Definition files.lib.php:63
img_object($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0)
Show a picto called object_picto (generic function)
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
dol_strlen($string, $stringencoding='UTF-8')
Make a strlen call.
price($amount, $form=0, $outlangs='', $trunc=1, $rounding=-1, $forcerounding=-1, $currency_code='')
Function to format a value into an amount for visual output Function used into PDF and HTML pages.
dol_now($mode='auto')
Return date for now.
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)
Output date in a string format according to outputlangs (or langs if not defined).
if(!function_exists( 'dol_getprefix')) dol_include_once($relpath, $classname='')
Make an include_once using default root and alternate root if it fails.
dol_clone($object, $native=0)
Create a clone of instance of object (new instance with same value for each properties) With native =...
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
dol_sanitizeFileName($str, $newstr='_', $unaccent=1)
Clean a string to use it as a file name.
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
getDolGlobalString($key, $default='')
Return dolibarr global constant string value.
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
getMarginInfos($pv_ht, $remise_percent, $tva_tx, $localtax1_tx, $localtax2_tx, $fk_pa, $pa_ht)
Return an array with margins information of a line.
Class to generate the form for creating a new ticket.
calcul_price_total($qty, $pu, $remise_percent_ligne, $txtva, $uselocaltax1_rate, $uselocaltax2_rate, $remise_percent_global, $price_base_type, $info_bits, $type, $seller='', $localtaxes_array=[], $progress=100, $multicurrency_tx=1, $pu_devise=0, $multicurrency_code='')
Calculate totals (net, vat, ...) of a line.
Definition price.lib.php:88
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition repair.php:139
publicphonebutton2 phonegreen basiclayout basiclayout TotalHT VATCode TotalVAT TotalLT1 TotalLT2 TotalTTC TotalHT clearboth nowraponall TAKEPOS_SHOW_SUBPRICE right right right takeposterminal SELECT e e e e e statut
Definition invoice.php:1929