dolibarr  19.0.0-dev
supplier_proposal.class.php
Go to the documentation of this file.
1 <?php
2 /* Copyright (C) 2002-2004 Rodolphe Quiedeville <rodolphe@quiedeville.org>
3  * Copyright (C) 2004 Eric Seigne <eric.seigne@ryxeo.com>
4  * Copyright (C) 2004-2011 Laurent Destailleur <eldy@users.sourceforge.net>
5  * Copyright (C) 2005 Marc Barilley <marc@ocebo.com>
6  * Copyright (C) 2005-2013 Regis Houssin <regis.houssin@inodbox.com>
7  * Copyright (C) 2006 Andre Cianfarani <acianfa@free.fr>
8  * Copyright (C) 2008 Raphael Bertrand <raphael.bertrand@resultic.fr>
9  * Copyright (C) 2010-2020 Juanjo Menent <jmenent@2byte.es>
10  * Copyright (C) 2010-2018 Philippe Grand <philippe.grand@atoo-net.com>
11  * Copyright (C) 2012-2014 Christophe Battarel <christophe.battarel@altairis.fr>
12  * Copyright (C) 2013 Florian Henry <florian.henry@open-concept.pro>
13  * Copyright (C) 2014 Marcos García <marcosgdf@gmail.com>
14  * Copyright (C) 2016 Ferran Marcet <fmarcet@2byte.es>
15  * Copyright (C) 2018 Nicolas ZABOURI <info@inovea-conseil.com>
16  * Copyright (C) 2019-2023 Frédéric France <frederic.france@netlogic.fr>
17  * Copyright (C) 2020 Tobias Sekan <tobias.sekan@startmail.com>
18  * Copyright (C) 2022 Gauthier VERDOL <gauthier.verdol@atm-consulting.fr>
19  *
20  * This program is free software; you can redistribute it and/or modify
21  * it under the terms of the GNU General Public License as published by
22  * the Free Software Foundation; either version 3 of the License, or
23  * (at your option) any later version.
24  *
25  * This program is distributed in the hope that it will be useful,
26  * but WITHOUT ANY WARRANTY; without even the implied warranty of
27  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
28  * GNU General Public License for more details.
29  *
30  * You should have received a copy of the GNU General Public License
31  * along with this program. If not, see <https://www.gnu.org/licenses/>.
32  */
33 
39 require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.product.class.php';
40 require_once DOL_DOCUMENT_ROOT.'/core/class/commonobject.class.php';
41 require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
42 require_once DOL_DOCUMENT_ROOT.'/contact/class/contact.class.php';
43 require_once DOL_DOCUMENT_ROOT.'/margin/lib/margins.lib.php';
44 require_once DOL_DOCUMENT_ROOT.'/multicurrency/class/multicurrency.class.php';
45 require_once DOL_DOCUMENT_ROOT.'/core/class/commonincoterm.class.php';
46 
51 {
52  use CommonIncoterm;
53 
57  public $element = 'supplier_proposal';
58 
62  public $table_element = 'supplier_proposal';
63 
67  public $table_element_line = 'supplier_proposaldet';
68 
72  public $fk_element = 'fk_supplier_proposal';
73 
77  public $picto = 'supplier_proposal';
78 
83  public $ismultientitymanaged = 1;
84 
89  public $restrictiononfksoc = 1;
90 
94  protected $table_ref_field = 'ref';
95 
96  public $socid; // Id client
97 
102  public $author;
103 
104  public $ref_fourn; //Reference saisie lors de l'ajout d'une ligne à la demande
105  public $ref_supplier; //Reference saisie lors de l'ajout d'une ligne à la demande
106  public $statut; // 0 (draft), 1 (validated), 2 (signed), 3 (not signed), 4 (processed/billed)
107 
111  public $date;
112 
117  public $date_livraison;
118 
122  public $delivery_date;
123 
128  public $datec;
129 
133  public $date_creation;
134 
139  public $datev;
140 
144  public $date_validation;
145 
146 
147  public $user_author_id;
148  public $user_valid_id;
149  public $user_close_id;
150 
155  public $price;
156 
161  public $tva;
162 
167  public $total;
168 
169  public $cond_reglement_code;
170  public $mode_reglement_code;
171  public $remise = 0;
172  public $remise_percent = 0;
173  public $remise_absolue = 0;
174 
175  public $extraparams = array();
176  public $lines = array();
177  public $line;
178 
179  public $labelStatus = array();
180  public $labelStatusShort = array();
181 
182  public $nbtodo;
183  public $nbtodolate;
184 
185  // Multicurrency
189  public $fk_multicurrency;
190 
191  public $multicurrency_code;
192  public $multicurrency_tx;
193  public $multicurrency_total_ht;
194  public $multicurrency_total_tva;
195  public $multicurrency_total_ttc;
196 
200  const STATUS_DRAFT = 0;
201 
205  const STATUS_VALIDATED = 1;
206 
210  const STATUS_SIGNED = 2;
211 
215  const STATUS_NOTSIGNED = 3;
216 
220  const STATUS_CLOSE = 4;
221 
222 
223 
231  public function __construct($db, $socid = "", $supplier_proposalid = 0)
232  {
233  global $conf, $langs;
234 
235  $this->db = $db;
236 
237  $this->socid = $socid;
238  $this->id = $supplier_proposalid;
239  }
240 
241 
242  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
253  public function add_product($idproduct, $qty, $remise_percent = 0)
254  {
255  // phpcs:enable
256  global $conf, $mysoc;
257 
258  if (!$qty) {
259  $qty = 1;
260  }
261 
262  dol_syslog(get_class($this)."::add_product $idproduct, $qty, $remise_percent");
263  if ($idproduct > 0) {
264  $prod = new Product($this->db);
265  $prod->fetch($idproduct);
266 
267  $productdesc = $prod->description;
268 
269  $tva_tx = get_default_tva($mysoc, $this->thirdparty, $prod->id);
270  $tva_npr = get_default_npr($mysoc, $this->thirdparty, $prod->id);
271  if (empty($tva_tx)) {
272  $tva_npr = 0;
273  }
274  $localtax1_tx = get_localtax($tva_tx, 1, $mysoc, $this->thirdparty, $tva_npr);
275  $localtax2_tx = get_localtax($tva_tx, 2, $mysoc, $this->thirdparty, $tva_npr);
276 
277  // multiprix
278  if ($conf->global->PRODUIT_MULTIPRICES && $this->thirdparty->price_level) {
279  $price = $prod->multiprices[$this->thirdparty->price_level];
280  } else {
281  $price = $prod->price;
282  }
283 
284  $line = new SupplierProposalLine($this->db);
285 
286  $line->fk_product = $idproduct;
287  $line->desc = $productdesc;
288  $line->qty = $qty;
289  $line->subprice = $price;
290  $line->remise_percent = $remise_percent;
291  $line->tva_tx = $tva_tx;
292 
293  $this->lines[] = $line;
294  return 1;
295  }
296  return -1;
297  }
298 
299  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
306  public function insert_discount($idremise)
307  {
308  // phpcs:enable
309  global $langs;
310 
311  include_once DOL_DOCUMENT_ROOT.'/core/lib/price.lib.php';
312  include_once DOL_DOCUMENT_ROOT.'/core/class/discount.class.php';
313 
314  $this->db->begin();
315 
316  $remise = new DiscountAbsolute($this->db);
317  $result = $remise->fetch($idremise);
318 
319  if ($result > 0) {
320  if ($remise->fk_facture) { // Protection against multiple submission
321  $this->error = $langs->trans("ErrorDiscountAlreadyUsed");
322  $this->db->rollback();
323  return -5;
324  }
325 
326  $supplier_proposalligne = new SupplierProposalLine($this->db);
327  $supplier_proposalligne->fk_supplier_proposal = $this->id;
328  $supplier_proposalligne->fk_remise_except = $remise->id;
329  $supplier_proposalligne->desc = $remise->description; // Description ligne
330  $supplier_proposalligne->tva_tx = $remise->tva_tx;
331  $supplier_proposalligne->subprice = -$remise->amount_ht;
332  $supplier_proposalligne->fk_product = 0; // Id produit predefini
333  $supplier_proposalligne->qty = 1;
334  $supplier_proposalligne->remise_percent = 0;
335  $supplier_proposalligne->rang = -1;
336  $supplier_proposalligne->info_bits = 2;
337 
338  $supplier_proposalligne->total_ht = -$remise->amount_ht;
339  $supplier_proposalligne->total_tva = -$remise->amount_tva;
340  $supplier_proposalligne->total_ttc = -$remise->amount_ttc;
341 
342  $result = $supplier_proposalligne->insert();
343  if ($result > 0) {
344  $result = $this->update_price(1);
345  if ($result > 0) {
346  $this->db->commit();
347  return 1;
348  } else {
349  $this->db->rollback();
350  return -1;
351  }
352  } else {
353  $this->error = $supplier_proposalligne->error;
354  $this->db->rollback();
355  return -2;
356  }
357  } else {
358  $this->db->rollback();
359  return -2;
360  }
361  }
362 
400  public function addline($desc, $pu_ht, $qty, $txtva, $txlocaltax1 = 0, $txlocaltax2 = 0, $fk_product = 0, $remise_percent = 0, $price_base_type = 'HT', $pu_ttc = 0, $info_bits = 0, $type = 0, $rang = -1, $special_code = 0, $fk_parent_line = 0, $fk_fournprice = 0, $pa_ht = 0, $label = '', $array_options = 0, $ref_supplier = '', $fk_unit = '', $origin = '', $origin_id = 0, $pu_ht_devise = 0, $date_start = 0, $date_end = 0)
401  {
402  global $mysoc, $conf, $langs;
403 
404  dol_syslog(get_class($this)."::addline supplier_proposalid=$this->id, desc=$desc, pu_ht=$pu_ht, qty=$qty, txtva=$txtva, fk_product=$fk_product, remise_except=$remise_percent, price_base_type=$price_base_type, pu_ttc=$pu_ttc, info_bits=$info_bits, type=$type");
405  include_once DOL_DOCUMENT_ROOT.'/core/lib/price.lib.php';
406 
407  // Clean parameters
408  if (empty($remise_percent)) {
409  $remise_percent = 0;
410  }
411  if (empty($qty)) {
412  $qty = 0;
413  }
414  if (empty($info_bits)) {
415  $info_bits = 0;
416  }
417  if (empty($rang)) {
418  $rang = 0;
419  }
420  if (empty($fk_parent_line) || $fk_parent_line < 0) {
421  $fk_parent_line = 0;
422  }
423  if (empty($pu_ht)) {
424  $pu_ht = 0;
425  }
426 
427  $remise_percent = price2num($remise_percent);
428  $qty = price2num($qty);
429  $pu_ht = price2num($pu_ht);
430  $pu_ttc = price2num($pu_ttc);
431  if (!preg_match('/\‍((.*)\‍)/', $txtva)) {
432  $txtva = price2num($txtva); // $txtva can have format '5.0(XXX)' or '5'
433  }
434  $txlocaltax1 = price2num($txlocaltax1);
435  $txlocaltax2 = price2num($txlocaltax2);
436  $pa_ht = price2num($pa_ht);
437  if ($price_base_type == 'HT') {
438  $pu = $pu_ht;
439  } else {
440  $pu = $pu_ttc;
441  }
442 
443  // Check parameters
444  if ($type < 0) {
445  return -1;
446  }
447 
448  if ($this->statut == self::STATUS_DRAFT) {
449  $this->db->begin();
450 
451  if ($fk_product > 0) {
452  if (!empty($conf->global->SUPPLIER_PROPOSAL_WITH_PREDEFINED_PRICES_ONLY)) {
453  // Check quantity is enough
454  dol_syslog(get_class($this)."::addline we check supplier prices fk_product=".$fk_product." fk_fournprice=".$fk_fournprice." qty=".$qty." ref_supplier=".$ref_supplier);
455  $productsupplier = new ProductFournisseur($this->db);
456  if ($productsupplier->fetch($fk_product) > 0) {
457  $product_type = $productsupplier->type;
458  $label = $productsupplier->label;
459  $fk_prod_fourn_price = $fk_fournprice;
460 
461  // We use 'none' instead of $ref_supplier, because fourn_ref may not exists anymore. So we will take the first supplier price ok.
462  // If we want a dedicated supplier price, we must provide $fk_prod_fourn_price.
463  $result = $productsupplier->get_buyprice($fk_prod_fourn_price, $qty, $fk_product, 'none', ($this->fk_soc ? $this->fk_soc : $this->socid)); // Search on couple $fk_prod_fourn_price/$qty first, then on triplet $qty/$fk_product/$ref_supplier/$this->fk_soc
464  if ($result > 0) {
465  $pu = $productsupplier->fourn_pu; // Unit price supplier price set by get_buyprice
466  $ref_supplier = $productsupplier->ref_supplier; // Ref supplier price set by get_buyprice
467  // is remise percent not keyed but present for the product we add it
468  if ($remise_percent == 0 && $productsupplier->remise_percent != 0) {
469  $remise_percent = $productsupplier->remise_percent;
470  }
471  }
472  if ($result == 0) { // If result == 0, we failed to found the supplier reference price
473  $langs->load("errors");
474  $this->error = "Ref ".$productsupplier->ref." ".$langs->trans("ErrorQtyTooLowForThisSupplier");
475  $this->db->rollback();
476  dol_syslog(get_class($this)."::addline we did not found supplier price, so we can't guess unit price");
477  //$pu = $productsupplier->fourn_pu; // We do not overwrite unit price
478  //$ref = $productsupplier_fourn; // We do not overwrite ref supplier price
479  return -1;
480  }
481  if ($result == -1) {
482  $langs->load("errors");
483  $this->error = "Ref ".$productsupplier->ref." ".$langs->trans("ErrorQtyTooLowForThisSupplier");
484  $this->db->rollback();
485  dol_syslog(get_class($this)."::addline result=".$result." - ".$this->error, LOG_DEBUG);
486  return -1;
487  }
488  if ($result < -1) {
489  $this->error = $productsupplier->error;
490  $this->errors = $productsupplier->errors;
491  $this->db->rollback();
492  dol_syslog(get_class($this)."::addline result=".$result." - ".$this->error, LOG_ERR);
493  return -1;
494  }
495  } else {
496  $this->error = $productsupplier->error;
497  $this->errors = $productsupplier->errors;
498  $this->db->rollback();
499  return -1;
500  }
501  }
502  } else {
503  $product_type = $type;
504  }
505 
506  // Calcul du total TTC et de la TVA pour la ligne a partir de
507  // qty, pu, remise_percent et txtva
508  // TRES IMPORTANT: C'est au moment de l'insertion ligne qu'on doit stocker
509  // la part ht, tva et ttc, et ce au niveau de la ligne qui a son propre taux tva.
510 
511  $localtaxes_type = getLocalTaxesFromRate($txtva, 0, $this->thirdparty, $mysoc);
512 
513  // Clean vat code
514  $reg = array();
515  $vat_src_code = '';
516  if (preg_match('/\‍((.*)\‍)/', $txtva, $reg)) {
517  $vat_src_code = $reg[1];
518  $txtva = preg_replace('/\s*\‍(.*\‍)/', '', $txtva); // Remove code into vatrate.
519  }
520 
521  if (isModEnabled("multicurrency") && $pu_ht_devise > 0) {
522  $pu = 0;
523  }
524 
525  $tabprice = calcul_price_total($qty, $pu, $remise_percent, $txtva, $txlocaltax1, $txlocaltax2, 0, $price_base_type, $info_bits, $type, $this->thirdparty, $localtaxes_type, 100, $this->multicurrency_tx, $pu_ht_devise);
526  $total_ht = $tabprice[0];
527  $total_tva = $tabprice[1];
528  $total_ttc = $tabprice[2];
529  $total_localtax1 = $tabprice[9];
530  $total_localtax2 = $tabprice[10];
531  $pu = $pu_ht = $tabprice[3];
532 
533  // MultiCurrency
534  $multicurrency_total_ht = $tabprice[16];
535  $multicurrency_total_tva = $tabprice[17];
536  $multicurrency_total_ttc = $tabprice[18];
537  $pu_ht_devise = $tabprice[19];
538 
539  // Rang to use
540  $ranktouse = $rang;
541  if ($ranktouse == -1) {
542  $rangmax = $this->line_max($fk_parent_line);
543  $ranktouse = $rangmax + 1;
544  }
545 
546  // TODO A virer
547  // Anciens indicateurs: $price, $remise (a ne plus utiliser)
548  $price = $pu;
549  $remise = 0;
550  if ($remise_percent > 0) {
551  $remise = round(($pu * $remise_percent / 100), 2);
552  $price = $pu - $remise;
553  }
554 
555  // Insert line
556  $this->line = new SupplierProposalLine($this->db);
557 
558  $this->line->fk_supplier_proposal = $this->id;
559  $this->line->label = $label;
560  $this->line->desc = $desc;
561  $this->line->qty = $qty;
562 
563  $this->line->vat_src_code = $vat_src_code;
564  $this->line->tva_tx = $txtva;
565  $this->line->localtax1_tx = ($total_localtax1 ? $localtaxes_type[1] : 0);
566  $this->line->localtax2_tx = ($total_localtax2 ? $localtaxes_type[3] : 0);
567  $this->line->localtax1_type = empty($localtaxes_type[0]) ? '' : $localtaxes_type[0];
568  $this->line->localtax2_type = empty($localtaxes_type[2]) ? '' : $localtaxes_type[2];
569  $this->line->fk_product = $fk_product;
570  $this->line->remise_percent = $remise_percent;
571  $this->line->subprice = $pu_ht;
572  $this->line->rang = $ranktouse;
573  $this->line->info_bits = $info_bits;
574  $this->line->total_ht = $total_ht;
575  $this->line->total_tva = $total_tva;
576  $this->line->total_localtax1 = $total_localtax1;
577  $this->line->total_localtax2 = $total_localtax2;
578  $this->line->total_ttc = $total_ttc;
579  $this->line->product_type = $type;
580  $this->line->special_code = $special_code;
581  $this->line->fk_parent_line = $fk_parent_line;
582  $this->line->fk_unit = $fk_unit;
583  $this->line->origin = $origin;
584  $this->line->origin_id = $origin_id;
585  $this->line->ref_fourn = $this->db->escape($ref_supplier);
586  $this->line->date_start = $date_start;
587  $this->line->date_end = $date_end;
588 
589  // infos marge
590  if (!empty($fk_product) && $fk_product > 0 && empty($fk_fournprice) && empty($pa_ht)) {
591  // When fk_fournprice is 0, we take the lowest buying price
592  include_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.product.class.php';
593  $productFournisseur = new ProductFournisseur($this->db);
594  $productFournisseur->find_min_price_product_fournisseur($fk_product);
595  $this->line->fk_fournprice = $productFournisseur->product_fourn_price_id;
596  } else {
597  $this->line->fk_fournprice = ($fk_fournprice > 0 ? $fk_fournprice : 0); // If fk_fournprice is -1, we will not use fk_fournprice
598  }
599  $this->line->pa_ht = $pa_ht;
600  //var_dump($this->line->fk_fournprice);exit;
601 
602  // Multicurrency
603  $this->line->fk_multicurrency = $this->fk_multicurrency;
604  $this->line->multicurrency_code = $this->multicurrency_code;
605  $this->line->multicurrency_subprice = $pu_ht_devise;
606  $this->line->multicurrency_total_ht = $multicurrency_total_ht;
607  $this->line->multicurrency_total_tva = $multicurrency_total_tva;
608  $this->line->multicurrency_total_ttc = $multicurrency_total_ttc;
609 
610  // Mise en option de la ligne
611  if (empty($qty) && empty($special_code)) {
612  $this->line->special_code = 3;
613  }
614 
615  if (is_array($array_options) && count($array_options) > 0) {
616  $this->line->array_options = $array_options;
617  }
618 
619  $result = $this->line->insert();
620  if ($result > 0) {
621  // Reorder if child line
622  if (!empty($fk_parent_line)) {
623  $this->line_order(true, 'DESC');
624  } elseif ($ranktouse > 0 && $ranktouse <= count($this->lines)) { // Update all rank of all other lines
625  $linecount = count($this->lines);
626  for ($ii = $ranktouse; $ii <= $linecount; $ii++) {
627  $this->updateRangOfLine($this->lines[$ii - 1]->id, $ii + 1);
628  }
629  }
630 
631  // Mise a jour informations denormalisees au niveau de la propale meme
632  $result = $this->update_price(1, 'auto', 0, $this->thirdparty); // This method is designed to add line from user input so total calculation must be done using 'auto' mode.
633  if ($result > 0) {
634  $this->db->commit();
635  return $this->line->id;
636  } else {
637  $this->error = $this->error();
638  $this->errors = $this->errors();
639  $this->db->rollback();
640  return -1;
641  }
642  } else {
643  $this->error = $this->line->error;
644  $this->errors = $this->line->errors;
645  $this->db->rollback();
646  return -2;
647  }
648  } else {
649  $this->error = 'BadStatusOfObjectToAddLine';
650  return -5;
651  }
652  }
653 
654 
681  public function updateline($rowid, $pu, $qty, $remise_percent, $txtva, $txlocaltax1 = 0, $txlocaltax2 = 0, $desc = '', $price_base_type = 'HT', $info_bits = 0, $special_code = 0, $fk_parent_line = 0, $skip_update_total = 0, $fk_fournprice = 0, $pa_ht = 0, $label = '', $type = 0, $array_options = 0, $ref_supplier = '', $fk_unit = '', $pu_ht_devise = 0)
682  {
683  global $conf, $user, $langs, $mysoc;
684 
685  dol_syslog(get_class($this)."::updateLine $rowid, $pu, $qty, $remise_percent, $txtva, $desc, $price_base_type, $info_bits");
686  include_once DOL_DOCUMENT_ROOT.'/core/lib/price.lib.php';
687 
688  // Clean parameters
689  $remise_percent = price2num($remise_percent);
690  $qty = price2num($qty);
691  $pu = price2num($pu);
692  if (!preg_match('/\‍((.*)\‍)/', $txtva)) {
693  $txtva = price2num($txtva); // $txtva can have format '5.0(XXX)' or '5'
694  }
695  $txlocaltax1 = price2num($txlocaltax1);
696  $txlocaltax2 = price2num($txlocaltax2);
697  $pa_ht = price2num($pa_ht);
698  if (empty($qty) && empty($special_code)) {
699  $special_code = 3; // Set option tag
700  }
701  if (!empty($qty) && $special_code == 3) {
702  $special_code = 0; // Remove option tag
703  }
704 
705  if ($this->statut == 0) {
706  $this->db->begin();
707 
708  // Calcul du total TTC et de la TVA pour la ligne a partir de
709  // qty, pu, remise_percent et txtva
710  // TRES IMPORTANT: C'est au moment de l'insertion ligne qu'on doit stocker
711  // la part ht, tva et ttc, et ce au niveau de la ligne qui a son propre taux tva.
712 
713  $localtaxes_type = getLocalTaxesFromRate($txtva, 0, $mysoc, $this->thirdparty);
714 
715  // Clean vat code
716  $reg = array();
717  $vat_src_code = '';
718  if (preg_match('/\‍((.*)\‍)/', $txtva, $reg)) {
719  $vat_src_code = $reg[1];
720  $txtva = preg_replace('/\s*\‍(.*\‍)/', '', $txtva); // Remove code into vatrate.
721  }
722 
723  if (isModEnabled("multicurrency") && $pu_ht_devise > 0) {
724  $pu = 0;
725  }
726 
727  $tabprice = calcul_price_total($qty, $pu, $remise_percent, $txtva, $txlocaltax1, $txlocaltax2, 0, $price_base_type, $info_bits, $type, $this->thirdparty, $localtaxes_type, 100, $this->multicurrency_tx, $pu_ht_devise);
728  $total_ht = $tabprice[0];
729  $total_tva = $tabprice[1];
730  $total_ttc = $tabprice[2];
731  $total_localtax1 = $tabprice[9];
732  $total_localtax2 = $tabprice[10];
733  $pu_ht = $tabprice[3];
734  $pu_tva = $tabprice[4];
735  $pu_ttc = $tabprice[5];
736 
737  // MultiCurrency
738  $multicurrency_total_ht = $tabprice[16];
739  $multicurrency_total_tva = $tabprice[17];
740  $multicurrency_total_ttc = $tabprice[18];
741  $pu_ht_devise = $tabprice[19];
742 
743  $pu = $pu_ht;
744  if ($price_base_type == 'TTC') {
745  $pu = $pu_ttc;
746  }
747 
748  //Fetch current line from the database and then clone the object and set it in $oldline property
749  $line = new SupplierProposalLine($this->db);
750  $line->fetch($rowid);
751  $line->fetch_optionals();
752 
753  // Stock previous line records
754  $staticline = clone $line;
755 
756  $line->oldline = $staticline;
757  $this->line = $line;
758  $this->line->context = $this->context;
759 
760  // Reorder if fk_parent_line change
761  if (!empty($fk_parent_line) && !empty($staticline->fk_parent_line) && $fk_parent_line != $staticline->fk_parent_line) {
762  $rangmax = $this->line_max($fk_parent_line);
763  $this->line->rang = $rangmax + 1;
764  }
765 
766  $this->line->id = $rowid;
767  $this->line->label = $label;
768  $this->line->desc = $desc;
769  $this->line->qty = $qty;
770  $this->line->product_type = $type;
771 
772  $this->line->vat_src_code = $vat_src_code;
773  $this->line->tva_tx = $txtva;
774  $this->line->localtax1_tx = $txlocaltax1;
775  $this->line->localtax2_tx = $txlocaltax2;
776  $this->line->localtax1_type = empty($localtaxes_type[0]) ? '' : $localtaxes_type[0];
777  $this->line->localtax2_type = empty($localtaxes_type[2]) ? '' : $localtaxes_type[2];
778  $this->line->remise_percent = $remise_percent;
779  $this->line->subprice = $pu;
780  $this->line->info_bits = $info_bits;
781  $this->line->total_ht = $total_ht;
782  $this->line->total_tva = $total_tva;
783  $this->line->total_localtax1 = $total_localtax1;
784  $this->line->total_localtax2 = $total_localtax2;
785  $this->line->total_ttc = $total_ttc;
786  $this->line->special_code = $special_code;
787  $this->line->fk_parent_line = $fk_parent_line;
788  $this->line->skip_update_total = $skip_update_total;
789  $this->line->ref_fourn = $ref_supplier;
790  $this->line->fk_unit = $fk_unit;
791 
792  // infos marge
793  if (!empty($fk_product) && $fk_product > 0 && empty($fk_fournprice) && empty($pa_ht)) {
794  // by external module, take lowest buying price
795  include_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.product.class.php';
796  $productFournisseur = new ProductFournisseur($this->db);
797  $productFournisseur->find_min_price_product_fournisseur($fk_product);
798  $this->line->fk_fournprice = $productFournisseur->product_fourn_price_id;
799  } else {
800  $this->line->fk_fournprice = $fk_fournprice;
801  }
802  $this->line->pa_ht = $pa_ht;
803 
804  if (is_array($array_options) && count($array_options) > 0) {
805  // We replace values in this->line->array_options only for entries defined into $array_options
806  foreach ($array_options as $key => $value) {
807  $this->line->array_options[$key] = $array_options[$key];
808  }
809  }
810 
811  // Multicurrency
812  $this->line->multicurrency_subprice = $pu_ht_devise;
813  $this->line->multicurrency_total_ht = $multicurrency_total_ht;
814  $this->line->multicurrency_total_tva = $multicurrency_total_tva;
815  $this->line->multicurrency_total_ttc = $multicurrency_total_ttc;
816 
817  $result = $this->line->update();
818  if ($result > 0) {
819  // Reorder if child line
820  if (!empty($fk_parent_line)) {
821  $this->line_order(true, 'DESC');
822  }
823 
824  $this->update_price(1);
825 
826  $this->fk_supplier_proposal = $this->id;
827 
828  $this->db->commit();
829  return $result;
830  } else {
831  $this->error = $this->db->error();
832  $this->db->rollback();
833  return -1;
834  }
835  } else {
836  dol_syslog(get_class($this)."::updateline Erreur -2 SupplierProposal en mode incompatible pour cette action");
837  return -2;
838  }
839  }
840 
841 
848  public function deleteline($lineid)
849  {
850  if ($this->statut == 0) {
851  $line = new SupplierProposalLine($this->db);
852 
853  // For triggers
854  $line->fetch($lineid);
855 
856  if ($line->delete() > 0) {
857  $this->update_price(1);
858 
859  return 1;
860  } else {
861  return -1;
862  }
863  } else {
864  return -2;
865  }
866  }
867 
868 
877  public function create($user, $notrigger = 0)
878  {
879  global $langs, $conf, $mysoc, $hookmanager;
880  $error = 0;
881 
882  $now = dol_now();
883 
884  dol_syslog(get_class($this)."::create");
885 
886  // Check parameters
887  $result = $this->fetch_thirdparty();
888  if ($result < 0) {
889  $this->error = "Failed to fetch company";
890  dol_syslog(get_class($this)."::create ".$this->error, LOG_ERR);
891  return -3;
892  }
893  if (!empty($this->ref)) { // We check that ref is not already used
894  $result = self::isExistingObject($this->element, 0, $this->ref); // Check ref is not yet used
895  if ($result > 0) {
896  $this->error = 'ErrorRefAlreadyExists';
897  dol_syslog(get_class($this)."::create ".$this->error, LOG_WARNING);
898  $this->db->rollback();
899  return -1;
900  }
901  }
902 
903  // Set tmp vars
904  $delivery_date = empty($this->delivery_date) ? $this->date_livraison : $this->delivery_date;
905 
906  // Multicurrency
907  if (!empty($this->multicurrency_code)) {
908  list($this->fk_multicurrency, $this->multicurrency_tx) = MultiCurrency::getIdAndTxFromCode($this->db, $this->multicurrency_code, $now);
909  }
910  if (empty($this->fk_multicurrency)) {
911  $this->multicurrency_code = $conf->currency;
912  $this->fk_multicurrency = 0;
913  $this->multicurrency_tx = 1;
914  }
915 
916  $this->db->begin();
917 
918  // Insert into database
919  $sql = "INSERT INTO ".MAIN_DB_PREFIX."supplier_proposal (";
920  $sql .= "fk_soc";
921  $sql .= ", price";
922  $sql .= ", remise";
923  $sql .= ", remise_percent";
924  $sql .= ", remise_absolue";
925  $sql .= ", total_tva";
926  $sql .= ", total_ttc";
927  $sql .= ", datec";
928  $sql .= ", ref";
929  $sql .= ", fk_user_author";
930  $sql .= ", note_private";
931  $sql .= ", note_public";
932  $sql .= ", model_pdf";
933  $sql .= ", fk_cond_reglement";
934  $sql .= ", fk_mode_reglement";
935  $sql .= ", fk_account";
936  $sql .= ", date_livraison";
937  $sql .= ", fk_shipping_method";
938  $sql .= ", fk_projet";
939  $sql .= ", entity";
940  $sql .= ", fk_multicurrency";
941  $sql .= ", multicurrency_code";
942  $sql .= ", multicurrency_tx";
943  $sql .= ") ";
944  $sql .= " VALUES (";
945  $sql .= ((int) $this->socid);
946  $sql .= ", 0";
947  $sql .= ", ".((double) $this->remise);
948  $sql .= ", ".($this->remise_percent ? ((double) $this->remise_percent) : 'null');
949  $sql .= ", ".($this->remise_absolue ? ((double) $this->remise_absolue) : 'null');
950  $sql .= ", 0";
951  $sql .= ", 0";
952  $sql .= ", '".$this->db->idate($now)."'";
953  $sql .= ", '(PROV)'";
954  $sql .= ", ".($user->id > 0 ? ((int) $user->id) : "null");
955  $sql .= ", '".$this->db->escape($this->note_private)."'";
956  $sql .= ", '".$this->db->escape($this->note_public)."'";
957  $sql .= ", '".$this->db->escape($this->model_pdf)."'";
958  $sql .= ", ".($this->cond_reglement_id > 0 ? ((int) $this->cond_reglement_id) : 'NULL');
959  $sql .= ", ".($this->mode_reglement_id > 0 ? ((int) $this->mode_reglement_id) : 'NULL');
960  $sql .= ", ".($this->fk_account > 0 ? ((int) $this->fk_account) : 'NULL');
961  $sql .= ", ".($delivery_date ? "'".$this->db->idate($delivery_date)."'" : "null");
962  $sql .= ", ".($this->shipping_method_id > 0 ? ((int) $this->shipping_method_id) : 'NULL');
963  $sql .= ", ".($this->fk_project > 0 ? ((int) $this->fk_project) : "null");
964  $sql .= ", ".((int) $conf->entity);
965  $sql .= ", ".((int) $this->fk_multicurrency);
966  $sql .= ", '".$this->db->escape($this->multicurrency_code)."'";
967  $sql .= ", ".((double) $this->multicurrency_tx);
968  $sql .= ")";
969 
970  dol_syslog(get_class($this)."::create", LOG_DEBUG);
971  $resql = $this->db->query($sql);
972  if ($resql) {
973  $this->id = $this->db->last_insert_id(MAIN_DB_PREFIX."supplier_proposal");
974 
975  if ($this->id) {
976  $this->ref = '(PROV'.$this->id.')';
977  $sql = 'UPDATE '.MAIN_DB_PREFIX."supplier_proposal SET ref='".$this->db->escape($this->ref)."' WHERE rowid=".((int) $this->id);
978 
979  dol_syslog(get_class($this)."::create", LOG_DEBUG);
980  $resql = $this->db->query($sql);
981  if (!$resql) {
982  $error++;
983  }
984 
985  if (!empty($this->linkedObjectsIds) && empty($this->linked_objects)) { // To use new linkedObjectsIds instead of old linked_objects
986  $this->linked_objects = $this->linkedObjectsIds; // TODO Replace linked_objects with linkedObjectsIds
987  }
988 
989  // Add object linked
990  if (!$error && $this->id && !empty($this->linked_objects) && is_array($this->linked_objects)) {
991  foreach ($this->linked_objects as $origin => $tmp_origin_id) {
992  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, ...))
993  foreach ($tmp_origin_id as $origin_id) {
994  $ret = $this->add_object_linked($origin, $origin_id);
995  if (!$ret) {
996  dol_print_error($this->db);
997  $error++;
998  }
999  }
1000  }
1001  }
1002  }
1003 
1004  /*
1005  * Insertion du detail des produits dans la base
1006  */
1007  if (!$error) {
1008  $fk_parent_line = 0;
1009  $num = count($this->lines);
1010 
1011  for ($i = 0; $i < $num; $i++) {
1012  // Reset fk_parent_line for no child products and special product
1013  if (($this->lines[$i]->product_type != 9 && empty($this->lines[$i]->fk_parent_line)) || $this->lines[$i]->product_type == 9) {
1014  $fk_parent_line = 0;
1015  }
1016 
1017  $result = $this->addline(
1018  $this->lines[$i]->desc,
1019  $this->lines[$i]->subprice,
1020  $this->lines[$i]->qty,
1021  $this->lines[$i]->tva_tx,
1022  $this->lines[$i]->localtax1_tx,
1023  $this->lines[$i]->localtax2_tx,
1024  $this->lines[$i]->fk_product,
1025  $this->lines[$i]->remise_percent,
1026  'HT',
1027  0,
1028  0,
1029  $this->lines[$i]->product_type,
1030  $this->lines[$i]->rang,
1031  $this->lines[$i]->special_code,
1032  $fk_parent_line,
1033  $this->lines[$i]->fk_fournprice,
1034  $this->lines[$i]->pa_ht,
1035  empty($this->lines[$i]->label) ? '' : $this->lines[$i]->label, // deprecated
1036  $this->lines[$i]->array_options,
1037  $this->lines[$i]->ref_fourn,
1038  $this->lines[$i]->fk_unit,
1039  'supplier_proposal',
1040  $this->lines[$i]->rowid
1041  );
1042 
1043  if ($result < 0) {
1044  $error++;
1045  $this->error = $this->db->error;
1046  dol_print_error($this->db);
1047  break;
1048  }
1049  // Defined the new fk_parent_line
1050  if ($result > 0 && $this->lines[$i]->product_type == 9) {
1051  $fk_parent_line = $result;
1052  }
1053  }
1054  }
1055 
1056  if (!$error) {
1057  // Mise a jour infos denormalisees
1058  $resql = $this->update_price(1);
1059  if ($resql) {
1060  $action = 'update';
1061 
1062  // Actions on extra fields
1063  if (!$error) {
1064  $result = $this->insertExtraFields();
1065  if ($result < 0) {
1066  $error++;
1067  }
1068  }
1069 
1070  if (!$error && !$notrigger) {
1071  // Call trigger
1072  $result = $this->call_trigger('PROPOSAL_SUPPLIER_CREATE', $user);
1073  if ($result < 0) {
1074  $error++;
1075  }
1076  // End call triggers
1077  }
1078  } else {
1079  $this->error = $this->db->lasterror();
1080  $error++;
1081  }
1082  }
1083  } else {
1084  $this->error = $this->db->lasterror();
1085  $error++;
1086  }
1087 
1088  if (!$error) {
1089  $this->db->commit();
1090  dol_syslog(get_class($this)."::create done id=".$this->id);
1091  return $this->id;
1092  } else {
1093  $this->db->rollback();
1094  return -2;
1095  }
1096  } else {
1097  $this->error = $this->db->lasterror();
1098  $this->db->rollback();
1099  return -1;
1100  }
1101  }
1102 
1110  public function createFromClone(User $user, $fromid = 0)
1111  {
1112  global $conf, $hookmanager;
1113 
1114  $error = 0;
1115  $now = dol_now();
1116 
1117  $this->db->begin();
1118 
1119  // get extrafields so they will be clone
1120  foreach ($this->lines as $line) {
1121  $line->fetch_optionals();
1122  }
1123 
1124  // Load source object
1125  $objFrom = clone $this;
1126 
1127  $objsoc = new Societe($this->db);
1128 
1129  // Change socid if needed
1130  if (!empty($fromid) && $fromid != $this->socid) {
1131  if ($objsoc->fetch($fromid) > 0) {
1132  $this->socid = $objsoc->id;
1133  $this->cond_reglement_id = (!empty($objsoc->cond_reglement_id) ? $objsoc->cond_reglement_id : 0);
1134  $this->mode_reglement_id = (!empty($objsoc->mode_reglement_id) ? $objsoc->mode_reglement_id : 0);
1135  $this->fk_project = '';
1136  }
1137 
1138  // TODO Change product price if multi-prices
1139  } else {
1140  $objsoc->fetch($this->socid);
1141  }
1142 
1143  $this->id = 0;
1144  $this->statut = 0;
1145 
1146  if (empty($conf->global->SUPPLIER_PROPOSAL_ADDON) || !is_readable(DOL_DOCUMENT_ROOT."/core/modules/supplier_proposal/".$conf->global->SUPPLIER_PROPOSAL_ADDON.".php")) {
1147  $this->error = 'ErrorSetupNotComplete';
1148  return -1;
1149  }
1150 
1151  // Clear fields
1152  $this->user_author = $user->id; // deprecated
1153  $this->user_author_id = $user->id;
1154  $this->user_valid = 0; // deprecated
1155  $this->user_valid_id = 0;
1156  $this->date = $now;
1157 
1158  // Set ref
1159  require_once DOL_DOCUMENT_ROOT."/core/modules/supplier_proposal/".$conf->global->SUPPLIER_PROPOSAL_ADDON.'.php';
1160  $obj = $conf->global->SUPPLIER_PROPOSAL_ADDON;
1161  $modSupplierProposal = new $obj;
1162  $this->ref = $modSupplierProposal->getNextValue($objsoc, $this);
1163 
1164  // Create clone
1165  $this->context['createfromclone'] = 'createfromclone';
1166  $result = $this->create($user);
1167  if ($result < 0) {
1168  $error++;
1169  }
1170 
1171  if (!$error) {
1172  // Hook of thirdparty module
1173  if (is_object($hookmanager)) {
1174  $parameters = array('objFrom'=>$objFrom);
1175  $action = '';
1176  $reshook = $hookmanager->executeHooks('createFrom', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
1177  if ($reshook < 0) {
1178  $this->setErrorsFromObject($hookmanager);
1179  $error++;
1180  }
1181  }
1182  }
1183 
1184  unset($this->context['createfromclone']);
1185 
1186  // End
1187  if (!$error) {
1188  $this->db->commit();
1189  return $this->id;
1190  } else {
1191  $this->db->rollback();
1192  return -1;
1193  }
1194  }
1195 
1203  public function fetch($rowid, $ref = '')
1204  {
1205  global $conf;
1206 
1207  $sql = "SELECT p.rowid, p.entity, p.ref, p.remise, p.remise_percent, p.remise_absolue, p.fk_soc";
1208  $sql .= ", p.total_ttc, p.total_tva, p.localtax1, p.localtax2, p.total_ht";
1209  $sql .= ", p.datec";
1210  $sql .= ", p.date_valid as datev";
1211  $sql .= ", p.date_livraison as delivery_date";
1212  $sql .= ", p.model_pdf, p.extraparams";
1213  $sql .= ", p.note_private, p.note_public";
1214  $sql .= ", p.fk_projet as fk_project, p.fk_statut";
1215  $sql .= ", p.fk_user_author, p.fk_user_valid, p.fk_user_cloture";
1216  $sql .= ", p.fk_cond_reglement";
1217  $sql .= ", p.fk_mode_reglement";
1218  $sql .= ', p.fk_account';
1219  $sql .= ", p.fk_shipping_method";
1220  $sql .= ", p.fk_multicurrency, p.multicurrency_code, p.multicurrency_tx, p.multicurrency_total_ht, p.multicurrency_total_tva, p.multicurrency_total_ttc";
1221  $sql .= ", c.label as statut_label";
1222  $sql .= ", cr.code as cond_reglement_code, cr.libelle as cond_reglement, cr.libelle_facture as cond_reglement_libelle_doc";
1223  $sql .= ", cp.code as mode_reglement_code, cp.libelle as mode_reglement";
1224  $sql .= " FROM ".MAIN_DB_PREFIX."c_propalst as c, ".MAIN_DB_PREFIX."supplier_proposal as p";
1225  $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_paiement as cp ON p.fk_mode_reglement = cp.id';
1226  $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'c_payment_term as cr ON p.fk_cond_reglement = cr.rowid';
1227  $sql .= " WHERE p.fk_statut = c.id";
1228  $sql .= " AND p.entity IN (".getEntity('supplier_proposal').")";
1229  if ($ref) {
1230  $sql .= " AND p.ref = '".$this->db->escape($ref)."'";
1231  } else {
1232  $sql .= " AND p.rowid = ".((int) $rowid);
1233  }
1234 
1235  dol_syslog(get_class($this)."::fetch", LOG_DEBUG);
1236  $resql = $this->db->query($sql);
1237  if ($resql) {
1238  if ($this->db->num_rows($resql)) {
1239  $obj = $this->db->fetch_object($resql);
1240 
1241  $this->id = $obj->rowid;
1242  $this->entity = $obj->entity;
1243 
1244  $this->ref = $obj->ref;
1245  $this->remise = $obj->remise;
1246  $this->remise_percent = $obj->remise_percent;
1247  $this->remise_absolue = $obj->remise_absolue;
1248  $this->total_ht = $obj->total_ht;
1249  $this->total_tva = $obj->total_tva;
1250  $this->total_localtax1 = $obj->localtax1;
1251  $this->total_localtax2 = $obj->localtax2;
1252  $this->total_ttc = $obj->total_ttc;
1253  $this->socid = $obj->fk_soc;
1254  $this->fk_project = $obj->fk_project;
1255  $this->model_pdf = $obj->model_pdf;
1256  $this->modelpdf = $obj->model_pdf; // deprecated
1257  $this->note = $obj->note_private; // TODO deprecated
1258  $this->note_private = $obj->note_private;
1259  $this->note_public = $obj->note_public;
1260  $this->statut = (int) $obj->fk_statut;
1261  $this->status = (int) $obj->fk_statut;
1262  $this->datec = $this->db->jdate($obj->datec); // TODO deprecated
1263  $this->datev = $this->db->jdate($obj->datev); // TODO deprecated
1264  $this->date_creation = $this->db->jdate($obj->datec); // Creation date
1265  $this->date = $this->date_creation;
1266  $this->date_validation = $this->db->jdate($obj->datev); // Validation date
1267  $this->date_livraison = $this->db->jdate($obj->delivery_date); // deprecated
1268  $this->delivery_date = $this->db->jdate($obj->delivery_date);
1269  $this->shipping_method_id = ($obj->fk_shipping_method > 0) ? $obj->fk_shipping_method : null;
1270 
1271  $this->mode_reglement_id = $obj->fk_mode_reglement;
1272  $this->mode_reglement_code = $obj->mode_reglement_code;
1273  $this->mode_reglement = $obj->mode_reglement;
1274  $this->fk_account = ($obj->fk_account > 0) ? $obj->fk_account : null;
1275  $this->cond_reglement_id = $obj->fk_cond_reglement;
1276  $this->cond_reglement_code = $obj->cond_reglement_code;
1277  $this->cond_reglement = $obj->cond_reglement;
1278  $this->cond_reglement_doc = $obj->cond_reglement_libelle_doc;
1279 
1280  $this->extraparams = (array) json_decode($obj->extraparams, true);
1281 
1282  $this->user_author_id = $obj->fk_user_author;
1283  $this->user_valid_id = $obj->fk_user_valid;
1284  $this->user_close_id = $obj->fk_user_cloture;
1285 
1286  // Multicurrency
1287  $this->fk_multicurrency = $obj->fk_multicurrency;
1288  $this->multicurrency_code = $obj->multicurrency_code;
1289  $this->multicurrency_tx = $obj->multicurrency_tx;
1290  $this->multicurrency_total_ht = $obj->multicurrency_total_ht;
1291  $this->multicurrency_total_tva = $obj->multicurrency_total_tva;
1292  $this->multicurrency_total_ttc = $obj->multicurrency_total_ttc;
1293 
1294  if ($obj->fk_statut == 0) {
1295  $this->brouillon = 1;
1296  }
1297 
1298  // Retrieve all extrafield
1299  // fetch optionals attributes and labels
1300  $this->fetch_optionals();
1301 
1302  $this->db->free($resql);
1303 
1304  $this->lines = array();
1305 
1306  // Lines of supplier proposals
1307  $sql = "SELECT d.rowid, d.fk_supplier_proposal, d.fk_parent_line, d.label as custom_label, d.description, d.price, d.tva_tx, d.localtax1_tx, d.localtax2_tx, d.qty, d.fk_remise_except, d.remise_percent, d.subprice, d.fk_product,";
1308  $sql .= " d.info_bits, d.total_ht, d.total_tva, d.total_localtax1, d.total_localtax2, d.total_ttc, d.fk_product_fournisseur_price as fk_fournprice, d.buy_price_ht as pa_ht, d.special_code, d.rang, d.product_type,";
1309  $sql .= ' p.ref as product_ref, p.description as product_desc, p.fk_product_type, p.label as product_label,';
1310  $sql .= ' d.ref_fourn as ref_produit_fourn,';
1311  $sql .= ' d.fk_multicurrency, d.multicurrency_code, d.multicurrency_subprice, d.multicurrency_total_ht, d.multicurrency_total_tva, d.multicurrency_total_ttc, d.fk_unit';
1312  $sql .= " FROM ".MAIN_DB_PREFIX."supplier_proposaldet as d";
1313  $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."product as p ON d.fk_product = p.rowid";
1314  $sql .= " WHERE d.fk_supplier_proposal = ".((int) $this->id);
1315  $sql .= " ORDER by d.rang";
1316 
1317  $result = $this->db->query($sql);
1318  if ($result) {
1319  $num = $this->db->num_rows($result);
1320  $i = 0;
1321 
1322  while ($i < $num) {
1323  $objp = $this->db->fetch_object($result);
1324 
1325  $line = new SupplierProposalLine($this->db);
1326 
1327  $line->rowid = $objp->rowid; // deprecated
1328  $line->id = $objp->rowid;
1329  $line->fk_supplier_proposal = $objp->fk_supplier_proposal;
1330  $line->fk_parent_line = $objp->fk_parent_line;
1331  $line->product_type = $objp->product_type;
1332  $line->label = $objp->custom_label;
1333  $line->desc = $objp->description; // Description ligne
1334  $line->qty = $objp->qty;
1335  $line->tva_tx = $objp->tva_tx;
1336  $line->localtax1_tx = $objp->localtax1_tx;
1337  $line->localtax2_tx = $objp->localtax2_tx;
1338  $line->subprice = $objp->subprice;
1339  $line->fk_remise_except = $objp->fk_remise_except;
1340  $line->remise_percent = $objp->remise_percent;
1341 
1342  $line->info_bits = $objp->info_bits;
1343  $line->total_ht = $objp->total_ht;
1344  $line->total_tva = $objp->total_tva;
1345  $line->total_localtax1 = $objp->total_localtax1;
1346  $line->total_localtax2 = $objp->total_localtax2;
1347  $line->total_ttc = $objp->total_ttc;
1348  $line->fk_fournprice = $objp->fk_fournprice;
1349  $marginInfos = getMarginInfos($objp->subprice, $objp->remise_percent, $objp->tva_tx, $objp->localtax1_tx, $objp->localtax2_tx, $line->fk_fournprice, $objp->pa_ht);
1350  $line->pa_ht = $marginInfos[0];
1351  $line->marge_tx = $marginInfos[1];
1352  $line->marque_tx = $marginInfos[2];
1353  $line->special_code = $objp->special_code;
1354  $line->rang = $objp->rang;
1355 
1356  $line->fk_product = $objp->fk_product;
1357 
1358  $line->ref = $objp->product_ref; // deprecated
1359  $line->product_ref = $objp->product_ref;
1360  $line->libelle = $objp->product_label; // deprecated
1361  $line->product_label = $objp->product_label;
1362  $line->product_desc = $objp->product_desc; // Description produit
1363  $line->fk_product_type = $objp->fk_product_type;
1364 
1365  $line->ref_fourn = $objp->ref_produit_fourn;
1366 
1367  // Multicurrency
1368  $line->fk_multicurrency = $objp->fk_multicurrency;
1369  $line->multicurrency_code = $objp->multicurrency_code;
1370  $line->multicurrency_subprice = $objp->multicurrency_subprice;
1371  $line->multicurrency_total_ht = $objp->multicurrency_total_ht;
1372  $line->multicurrency_total_tva = $objp->multicurrency_total_tva;
1373  $line->multicurrency_total_ttc = $objp->multicurrency_total_ttc;
1374  $line->fk_unit = $objp->fk_unit;
1375 
1376  $this->lines[$i] = $line;
1377 
1378  $i++;
1379  }
1380  $this->db->free($result);
1381  } else {
1382  $this->error = $this->db->error();
1383  return -1;
1384  }
1385 
1386  // Retrieve all extrafield
1387  // fetch optionals attributes and labels
1388  $this->fetch_optionals();
1389 
1390  return 1;
1391  }
1392 
1393  $this->error = "Record Not Found";
1394  return 0;
1395  } else {
1396  $this->error = $this->db->error();
1397  return -1;
1398  }
1399  }
1400 
1408  public function valid($user, $notrigger = 0)
1409  {
1410  require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
1411 
1412  global $conf, $langs;
1413 
1414  $error = 0;
1415  $now = dol_now();
1416 
1417  if ((empty($conf->global->MAIN_USE_ADVANCED_PERMS) && !empty($user->rights->supplier_proposal->creer))
1418  || (!empty($conf->global->MAIN_USE_ADVANCED_PERMS) && !empty($user->rights->supplier_proposal->validate_advance))) {
1419  $this->db->begin();
1420 
1421  // Numbering module definition
1422  $soc = new Societe($this->db);
1423  $result = $soc->fetch($this->socid);
1424 
1425  if ($result < 0) {
1426  return -1;
1427  }
1428 
1429  // Define new ref
1430  if (preg_match('/^[\‍(]?PROV/i', $this->ref) || empty($this->ref)) { // empty should not happened, but when it occurs, the test save life
1431  $num = $this->getNextNumRef($soc);
1432  } else {
1433  $num = $this->ref;
1434  }
1435  $this->newref = dol_sanitizeFileName($num);
1436 
1437  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal";
1438  $sql .= " SET ref = '".$this->db->escape($num)."',";
1439  $sql .= " fk_statut = 1, date_valid='".$this->db->idate($now)."', fk_user_valid=".((int) $user->id);
1440  $sql .= " WHERE rowid = ".((int) $this->id)." AND fk_statut = 0";
1441 
1442  dol_syslog(get_class($this)."::valid", LOG_DEBUG);
1443  $resql = $this->db->query($sql);
1444  if (!$resql) {
1445  dol_print_error($this->db);
1446  $error++;
1447  }
1448 
1449  // Trigger calls
1450  if (!$error && !$notrigger) {
1451  // Call trigger
1452  $result = $this->call_trigger('PROPOSAL_SUPPLIER_VALIDATE', $user);
1453  if ($result < 0) {
1454  $error++;
1455  }
1456  // End call triggers
1457  }
1458 
1459  if (!$error) {
1460  $this->oldref = $this->ref;
1461 
1462  // Rename directory if dir was a temporary ref
1463  if (preg_match('/^[\‍(]?PROV/i', $this->ref)) {
1464  // Now we rename also files into index
1465  $sql = 'UPDATE '.MAIN_DB_PREFIX."ecm_files set filename = CONCAT('".$this->db->escape($this->newref)."', SUBSTR(filename, ".(strlen($this->ref) + 1).")), filepath = 'supplier_proposal/".$this->db->escape($this->newref)."'";
1466  $sql .= " WHERE filename LIKE '".$this->db->escape($this->ref)."%' AND filepath = 'supplier_proposal/".$this->db->escape($this->ref)."' and entity = ".$conf->entity;
1467  $resql = $this->db->query($sql);
1468  if (!$resql) {
1469  $error++; $this->error = $this->db->lasterror();
1470  }
1471 
1472  // We rename directory ($this->ref = old ref, $num = new ref) in order not to lose the attachments
1473  $oldref = dol_sanitizeFileName($this->ref);
1474  $newref = dol_sanitizeFileName($num);
1475  $dirsource = $conf->supplier_proposal->dir_output.'/'.$oldref;
1476  $dirdest = $conf->supplier_proposal->dir_output.'/'.$newref;
1477  if (!$error && file_exists($dirsource)) {
1478  dol_syslog(get_class($this)."::valid rename dir ".$dirsource." into ".$dirdest);
1479  if (@rename($dirsource, $dirdest)) {
1480  dol_syslog("Rename ok");
1481  // Rename docs starting with $oldref with $newref
1482  $listoffiles = dol_dir_list($conf->supplier_proposal->dir_output.'/'.$newref, 'files', 1, '^'.preg_quote($oldref, '/'));
1483  foreach ($listoffiles as $fileentry) {
1484  $dirsource = $fileentry['name'];
1485  $dirdest = preg_replace('/^'.preg_quote($oldref, '/').'/', $newref, $dirsource);
1486  $dirsource = $fileentry['path'].'/'.$dirsource;
1487  $dirdest = $fileentry['path'].'/'.$dirdest;
1488  @rename($dirsource, $dirdest);
1489  }
1490  }
1491  }
1492  }
1493 
1494  $this->ref = $num;
1495  $this->brouillon = 0;
1496  $this->statut = 1;
1497  $this->user_valid_id = $user->id;
1498  $this->datev = $now;
1499 
1500  $this->db->commit();
1501  return 1;
1502  } else {
1503  $this->db->rollback();
1504  return -1;
1505  }
1506  } else {
1507  dol_syslog("You don't have permission to validate supplier proposal", LOG_WARNING);
1508  return -2;
1509  }
1510  }
1511 
1512  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1521  public function set_date_livraison($user, $delivery_date)
1522  {
1523  // phpcs:enable
1524  return $this->setDeliveryDate($user, $delivery_date);
1525  }
1526 
1534  public function setDeliveryDate($user, $delivery_date)
1535  {
1536  if (!empty($user->rights->supplier_proposal->creer)) {
1537  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal ";
1538  $sql .= " SET date_livraison = ".($delivery_date != '' ? "'".$this->db->idate($delivery_date)."'" : 'null');
1539  $sql .= " WHERE rowid = ".((int) $this->id);
1540 
1541  if ($this->db->query($sql)) {
1542  $this->date_livraison = $delivery_date;
1543  $this->delivery_date = $delivery_date;
1544  return 1;
1545  } else {
1546  $this->error = $this->db->error();
1547  dol_syslog(get_class($this)."::setDeliveryDate Erreur SQL");
1548  return -1;
1549  }
1550  }
1551  return 0;
1552  }
1553 
1554  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1562  public function set_remise_percent($user, $remise)
1563  {
1564  // phpcs:enable
1565  $remise = trim($remise) ?trim($remise) : 0;
1566 
1567  if (!empty($user->rights->supplier_proposal->creer)) {
1568  $remise = price2num($remise, 2);
1569 
1570  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal SET remise_percent = ".((float) $remise);
1571  $sql .= " WHERE rowid = ".((int) $this->id)." AND fk_statut = 0";
1572 
1573  if ($this->db->query($sql)) {
1574  $this->remise_percent = ((float) $remise);
1575  $this->update_price(1);
1576  return 1;
1577  } else {
1578  $this->error = $this->db->error();
1579  return -1;
1580  }
1581  }
1582  return 0;
1583  }
1584 
1585 
1586  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1594  public function set_remise_absolue($user, $remise)
1595  {
1596  // phpcs:enable
1597  if (empty($remise)) {
1598  $remise = 0;
1599  }
1600 
1601  $remise = price2num($remise);
1602 
1603  if (!empty($user->rights->supplier_proposal->creer)) {
1604  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal ";
1605  $sql .= " SET remise_absolue = ".((float) $remise);
1606  $sql .= " WHERE rowid = ".((int) $this->id)." AND fk_statut = 0";
1607 
1608  if ($this->db->query($sql)) {
1609  $this->remise_absolue = $remise;
1610  $this->update_price(1);
1611  return 1;
1612  } else {
1613  $this->error = $this->db->error();
1614  return -1;
1615  }
1616  }
1617  return 0;
1618  }
1619 
1620 
1621 
1631  public function reopen($user, $statut, $note = '', $notrigger = 0)
1632  {
1633  global $langs, $conf;
1634 
1635  $this->statut = $statut;
1636  $error = 0;
1637 
1638  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal";
1639  $sql .= " SET fk_statut = ".((int) $this->statut).",";
1640  if (!empty($note)) {
1641  $sql .= " note_private = '".$this->db->escape($note)."',";
1642  }
1643  $sql .= " date_cloture=NULL, fk_user_cloture=NULL";
1644  $sql .= " WHERE rowid = ".((int) $this->id);
1645 
1646  $this->db->begin();
1647 
1648  dol_syslog(get_class($this)."::reopen", LOG_DEBUG);
1649  $resql = $this->db->query($sql);
1650  if (!$resql) {
1651  $error++; $this->errors[] = "Error ".$this->db->lasterror();
1652  }
1653  if (!$error) {
1654  if (!$notrigger) {
1655  // Call trigger
1656  $result = $this->call_trigger('PROPOSAL_SUPPLIER_REOPEN', $user);
1657  if ($result < 0) {
1658  $error++;
1659  }
1660  // End call triggers
1661  }
1662  }
1663 
1664  // Commit or rollback
1665  if ($error) {
1666  if (!empty($this->errors)) {
1667  foreach ($this->errors as $errmsg) {
1668  dol_syslog(get_class($this)."::update ".$errmsg, LOG_ERR);
1669  $this->error .= ($this->error ? ', '.$errmsg : $errmsg);
1670  }
1671  }
1672  $this->db->rollback();
1673  return -1 * $error;
1674  } else {
1675  $this->db->commit();
1676  return 1;
1677  }
1678  }
1679 
1680 
1689  public function cloture($user, $status, $note)
1690  {
1691  global $langs, $conf;
1692  $hidedetails = 0;
1693  $hidedesc = 0;
1694  $hideref = 0;
1695  $this->statut = $status;
1696  $error = 0;
1697  $now = dol_now();
1698 
1699  $this->db->begin();
1700 
1701  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal";
1702  $sql .= " SET fk_statut = ".((int) $status).", note_private = '".$this->db->escape($note)."', date_cloture='".$this->db->idate($now)."', fk_user_cloture=".$user->id;
1703  $sql .= " WHERE rowid = ".((int) $this->id);
1704 
1705  $resql = $this->db->query($sql);
1706  if ($resql) {
1707  $modelpdf = $conf->global->SUPPLIER_PROPOSAL_ADDON_PDF_ODT_CLOSED ? $conf->global->SUPPLIER_PROPOSAL_ADDON_PDF_ODT_CLOSED : (empty($this->model_pdf) ? '' : $this->model_pdf);
1708  $triggerName = 'PROPOSAL_SUPPLIER_CLOSE_REFUSED';
1709 
1710  if ($status == 2) {
1711  $triggerName = 'PROPOSAL_SUPPLIER_CLOSE_SIGNED';
1712  $modelpdf = $conf->global->SUPPLIER_PROPOSAL_ADDON_PDF_ODT_TOBILL ? $conf->global->SUPPLIER_PROPOSAL_ADDON_PDF_ODT_TOBILL : (empty($this->model_pdf) ? '' : $this->model_pdf);
1713 
1714  if (!empty($conf->global->SUPPLIER_PROPOSAL_UPDATE_PRICE_ON_SUPPlIER_PROPOSAL)) { // TODO This option was not tested correctly. Error if product ref does not exists
1715  $result = $this->updateOrCreatePriceFournisseur($user);
1716  }
1717  }
1718  if ($status == 4) {
1719  $triggerName = 'PROPOSAL_SUPPLIER_CLASSIFY_BILLED';
1720  }
1721 
1722  if (empty($conf->global->MAIN_DISABLE_PDF_AUTOUPDATE)) {
1723  // Define output language
1724  $outputlangs = $langs;
1725  if (getDolGlobalInt('MAIN_MULTILANGS')) {
1726  $outputlangs = new Translate("", $conf);
1727  $newlang = (GETPOST('lang_id', 'aZ09') ? GETPOST('lang_id', 'aZ09') : $this->thirdparty->default_lang);
1728  $outputlangs->setDefaultLang($newlang);
1729  }
1730  //$ret=$object->fetch($id); // Reload to get new records
1731  $this->generateDocument($modelpdf, $outputlangs, $hidedetails, $hidedesc, $hideref);
1732  }
1733 
1734  // Call trigger
1735  $result = $this->call_trigger($triggerName, $user);
1736  if ($result < 0) {
1737  $error++;
1738  }
1739  // End call triggers
1740 
1741  if (!$error) {
1742  $this->db->commit();
1743  return 1;
1744  } else {
1745  $this->db->rollback();
1746  return -1;
1747  }
1748  } else {
1749  $this->error = $this->db->lasterror();
1750  $this->errors[] = $this->db->lasterror();
1751  $this->db->rollback();
1752  return -1;
1753  }
1754  }
1755 
1762  public function updateOrCreatePriceFournisseur($user)
1763  {
1764  global $conf;
1765 
1766  dol_syslog(get_class($this)."::updateOrCreatePriceFournisseur", LOG_DEBUG);
1767  foreach ($this->lines as $product) {
1768  if ($product->subprice <= 0) {
1769  continue;
1770  }
1771  $productsupplier = new ProductFournisseur($this->db);
1772 
1773  $multicurrency_tx = 1;
1774  $fk_multicurrency = 0;
1775 
1776  if (empty($this->thirdparty)) {
1777  $this->fetch_thirdparty();
1778  }
1779 
1780  $ref_fourn = $product->ref_fourn;
1781  if (empty($ref_fourn)) {
1782  $ref_fourn = $product->ref_supplier;
1783  }
1784  if (isModEnabled("multicurrency") && !empty($product->multicurrency_code)) {
1785  list($fk_multicurrency, $multicurrency_tx) = MultiCurrency::getIdAndTxFromCode($this->db, $product->multicurrency_code);
1786  }
1787  $productsupplier->id = $product->fk_product;
1788 
1789  $productsupplier->update_buyprice($product->qty, $product->total_ht, $user, 'HT', $this->thirdparty, '', $ref_fourn, $product->tva_tx, 0, 0, 0, $product->info_bits, '', '', array(), '', $product->multicurrency_total_ht, 'HT', $multicurrency_tx, $product->multicurrency_code, '', '', '');
1790  }
1791 
1792  return 1;
1793  }
1794 
1803  public function updatePriceFournisseur($idProductFournPrice, $product, $user)
1804  {
1805  $price = price2num($product->subprice * $product->qty, 'MU');
1806  $unitPrice = price2num($product->subprice, 'MU');
1807 
1808  $sql = 'UPDATE '.MAIN_DB_PREFIX.'product_fournisseur_price SET '.(!empty($product->ref_fourn) ? 'ref_fourn = "'.$this->db->escape($product->ref_fourn).'", ' : '').' price ='.((float) $price).', unitprice ='.((float) $unitPrice).' WHERE rowid = '.((int) $idProductFournPrice);
1809 
1810  $resql = $this->db->query($sql);
1811  if (!$resql) {
1812  $this->error = $this->db->error();
1813  $this->db->rollback();
1814  return -1;
1815  }
1816  return 1;
1817  }
1818 
1826  public function createPriceFournisseur($product, $user)
1827  {
1828  global $conf;
1829 
1830  $price = price2num($product->subprice * $product->qty, 'MU');
1831  $qty = price2num($product->qty);
1832  $unitPrice = price2num($product->subprice, 'MU');
1833 
1834  $now = dol_now();
1835 
1836  $values = array(
1837  "'".$this->db->idate($now)."'",
1838  $product->fk_product,
1839  $this->thirdparty->id,
1840  "'".$product->ref_fourn."'",
1841  $price,
1842  $qty,
1843  $unitPrice,
1844  $product->tva_tx,
1845  $user->id
1846  );
1847  if (isModEnabled("multicurrency")) {
1848  if (!empty($product->multicurrency_code)) {
1849  include_once DOL_DOCUMENT_ROOT.'/multicurrency/class/multicurrency.class.php';
1850  $multicurrency = new MultiCurrency($this->db); //need to fetch because empty fk_multicurrency and rate
1851  $multicurrency->fetch(0, $product->multicurrency_code);
1852  if (!empty($multicurrency->id)) {
1853  $values[] = $multicurrency->id;
1854  $values[] = "'".$product->multicurrency_code."'";
1855  $values[] = $product->multicurrency_subprice;
1856  $values[] = $product->multicurrency_total_ht;
1857  $values[] = $multicurrency->rate->rate;
1858  } else {
1859  for ($i = 0; $i < 5; $i++) {
1860  $values[] = 'NULL';
1861  }
1862  }
1863  }
1864  }
1865 
1866  $sql = 'INSERT INTO '.MAIN_DB_PREFIX.'product_fournisseur_price ';
1867  $sql .= '(datec, fk_product, fk_soc, ref_fourn, price, quantity, unitprice, tva_tx, fk_user';
1868  if (isModEnabled("multicurrency") && !empty($product->multicurrency_code)) {
1869  $sql .= ',fk_multicurrency, multicurrency_code, multicurrency_unitprice, multicurrency_price, multicurrency_tx';
1870  }
1871  $sql .= ') VALUES ('.implode(',', $values).')';
1872 
1873  $resql = $this->db->query($sql);
1874  if (!$resql) {
1875  $this->error = $this->db->error();
1876  $this->db->rollback();
1877  return -1;
1878  }
1879  return 1;
1880  }
1881 
1882  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1889  public function setDraft($user)
1890  {
1891  // phpcs:enable
1892  global $conf, $langs;
1893 
1894  $error = 0;
1895 
1896  if ($this->statut == self::STATUS_DRAFT) {
1897  dol_syslog(get_class($this)."::setDraft already draft status", LOG_WARNING);
1898  return 0;
1899  }
1900 
1901  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposal";
1902  $sql .= " SET fk_statut = ".self::STATUS_DRAFT;
1903  $sql .= " WHERE rowid = ".((int) $this->id);
1904 
1905  if ($this->db->query($sql)) {
1906  if (!$error) {
1907  $this->oldcopy = clone $this;
1908  }
1909 
1910  if (!$error) {
1911  // Call trigger
1912  $result = $this->call_trigger('PROPOSAL_SUPPLIER_UNVALIDATE', $user);
1913  if ($result < 0) {
1914  $error++;
1915  }
1916  }
1917 
1918  if (!$error) {
1919  $this->statut = self::STATUS_DRAFT;
1920  $this->brouillon = 1;
1921  $this->db->commit();
1922  return 1;
1923  } else {
1924  $this->db->rollback();
1925  return -1;
1926  }
1927  } else {
1928  return -1;
1929  }
1930  }
1931 
1932 
1933  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
1947  public function liste_array($shortlist = 0, $draft = 0, $notcurrentuser = 0, $socid = 0, $limit = 0, $offset = 0, $sortfield = 'p.datec', $sortorder = 'DESC')
1948  {
1949  // phpcs:enable
1950  global $conf, $user;
1951 
1952  $ga = array();
1953 
1954  $sql = "SELECT s.rowid, s.nom as name, s.client,";
1955  $sql .= " p.rowid as supplier_proposalid, p.fk_statut, p.total_ht, p.ref, p.remise, ";
1956  $sql .= " p.datep as dp, p.fin_validite as datelimite";
1957  if (empty($user->rights->societe->client->voir) && !$socid) {
1958  $sql .= ", sc.fk_soc, sc.fk_user";
1959  }
1960  $sql .= " FROM ".MAIN_DB_PREFIX."societe as s, ".MAIN_DB_PREFIX."supplier_proposal as p, ".MAIN_DB_PREFIX."c_propalst as c";
1961  if (empty($user->rights->societe->client->voir) && !$socid) {
1962  $sql .= ", ".MAIN_DB_PREFIX."societe_commerciaux as sc";
1963  }
1964  $sql .= " WHERE p.entity IN (".getEntity('supplier_proposal').")";
1965  $sql .= " AND p.fk_soc = s.rowid";
1966  $sql .= " AND p.fk_statut = c.id";
1967  if (empty($user->rights->societe->client->voir) && !$socid) { //restriction
1968  $sql .= " AND s.rowid = sc.fk_soc AND sc.fk_user = ".((int) $user->id);
1969  }
1970  if ($socid) {
1971  $sql .= " AND s.rowid = ".((int) $socid);
1972  }
1973  if ($draft) {
1974  $sql .= " AND p.fk_statut = 0";
1975  }
1976  if ($notcurrentuser > 0) {
1977  $sql .= " AND p.fk_user_author <> ".$user->id;
1978  }
1979  $sql .= $this->db->order($sortfield, $sortorder);
1980  $sql .= $this->db->plimit($limit, $offset);
1981 
1982  $result = $this->db->query($sql);
1983  if ($result) {
1984  $num = $this->db->num_rows($result);
1985  if ($num) {
1986  $i = 0;
1987  while ($i < $num) {
1988  $obj = $this->db->fetch_object($result);
1989 
1990  if ($shortlist == 1) {
1991  $ga[$obj->supplier_proposalid] = $obj->ref;
1992  } elseif ($shortlist == 2) {
1993  $ga[$obj->supplier_proposalid] = $obj->ref.' ('.$obj->name.')';
1994  } else {
1995  $ga[$i]['id'] = $obj->supplier_proposalid;
1996  $ga[$i]['ref'] = $obj->ref;
1997  $ga[$i]['name'] = $obj->name;
1998  }
1999 
2000  $i++;
2001  }
2002  }
2003  return $ga;
2004  } else {
2005  dol_print_error($this->db);
2006  return -1;
2007  }
2008  }
2009 
2017  public function delete($user, $notrigger = 0)
2018  {
2019  global $conf, $langs;
2020  require_once DOL_DOCUMENT_ROOT.'/core/lib/files.lib.php';
2021 
2022  $error = 0;
2023 
2024  $this->db->begin();
2025 
2026  if (!$notrigger) {
2027  // Call trigger
2028  $result = $this->call_trigger('PROPOSAL_SUPPLIER_DELETE', $user);
2029  if ($result < 0) {
2030  $error++;
2031  }
2032  // End call triggers
2033  }
2034 
2035  if (!$error) {
2036  $main = MAIN_DB_PREFIX.'supplier_proposaldet';
2037  $ef = $main."_extrafields";
2038  $sqlef = "DELETE FROM $ef WHERE fk_object IN (SELECT rowid FROM $main WHERE fk_supplier_proposal = ".((int) $this->id).")";
2039  $sql = "DELETE FROM ".MAIN_DB_PREFIX."supplier_proposaldet WHERE fk_supplier_proposal = ".((int) $this->id);
2040  if ($this->db->query($sql)) {
2041  $sql = "DELETE FROM ".MAIN_DB_PREFIX."supplier_proposal WHERE rowid = ".((int) $this->id);
2042  if ($this->db->query($sqlef) && $this->db->query($sql)) {
2043  // Delete linked object
2044  $res = $this->deleteObjectLinked();
2045  if ($res < 0) {
2046  $error++;
2047  }
2048 
2049  if (!$error) {
2050  // Delete record into ECM index (Note that delete is also done when deleting files with the dol_delete_dir_recursive
2051  $this->deleteEcmFiles();
2052 
2053  // We remove directory
2054  $ref = dol_sanitizeFileName($this->ref);
2055  if ($conf->supplier_proposal->dir_output && !empty($this->ref)) {
2056  $dir = $conf->supplier_proposal->dir_output."/".$ref;
2057  $file = $dir."/".$ref.".pdf";
2058  if (file_exists($file)) {
2059  dol_delete_preview($this);
2060 
2061  if (!dol_delete_file($file, 0, 0, 0, $this)) { // For triggers
2062  $this->error = 'ErrorFailToDeleteFile';
2063  $this->errors = array('ErrorFailToDeleteFile');
2064  $this->db->rollback();
2065  return 0;
2066  }
2067  }
2068  if (file_exists($dir)) {
2069  $res = @dol_delete_dir_recursive($dir);
2070  if (!$res) {
2071  $this->error = 'ErrorFailToDeleteDir';
2072  $this->errors = array('ErrorFailToDeleteDir');
2073  $this->db->rollback();
2074  return 0;
2075  }
2076  }
2077  }
2078  }
2079 
2080  // Removed extrafields
2081  if (!$error) {
2082  $result = $this->deleteExtraFields();
2083  if ($result < 0) {
2084  $error++;
2085  $errorflag = -4;
2086  dol_syslog(get_class($this)."::delete erreur ".$errorflag." ".$this->error, LOG_ERR);
2087  }
2088  }
2089 
2090  if (!$error) {
2091  dol_syslog(get_class($this)."::delete ".$this->id." by ".$user->id, LOG_DEBUG);
2092  $this->db->commit();
2093  return 1;
2094  } else {
2095  $this->error = $this->db->lasterror();
2096  $this->db->rollback();
2097  return 0;
2098  }
2099  } else {
2100  $this->error = $this->db->lasterror();
2101  $this->db->rollback();
2102  return -3;
2103  }
2104  } else {
2105  $this->error = $this->db->lasterror();
2106  $this->db->rollback();
2107  return -2;
2108  }
2109  } else {
2110  $this->db->rollback();
2111  return -1;
2112  }
2113  }
2114 
2121  public function info($id)
2122  {
2123  $sql = "SELECT c.rowid, ";
2124  $sql .= " c.datec, c.date_valid as datev, c.date_cloture as dateo,";
2125  $sql .= " c.fk_user_author, c.fk_user_valid, c.fk_user_cloture";
2126  $sql .= " FROM ".MAIN_DB_PREFIX."supplier_proposal as c";
2127  $sql .= " WHERE c.rowid = ".((int) $id);
2128 
2129  $result = $this->db->query($sql);
2130 
2131  if ($result) {
2132  if ($this->db->num_rows($result)) {
2133  $obj = $this->db->fetch_object($result);
2134 
2135  $this->id = $obj->rowid;
2136 
2137  $this->date_creation = $this->db->jdate($obj->datec);
2138  $this->date_validation = $this->db->jdate($obj->datev);
2139  $this->date_cloture = $this->db->jdate($obj->dateo);
2140 
2141  $cuser = new User($this->db);
2142  $cuser->fetch($obj->fk_user_author);
2143  $this->user_creation = $cuser;
2144 
2145  if ($obj->fk_user_valid) {
2146  $vuser = new User($this->db);
2147  $vuser->fetch($obj->fk_user_valid);
2148  $this->user_validation = $vuser;
2149  }
2150 
2151  if ($obj->fk_user_cloture) {
2152  $cluser = new User($this->db);
2153  $cluser->fetch($obj->fk_user_cloture);
2154  $this->user_cloture = $cluser;
2155  }
2156  }
2157  $this->db->free($result);
2158  } else {
2159  dol_print_error($this->db);
2160  }
2161  }
2162 
2163 
2170  public function getLibStatut($mode = 0)
2171  {
2172  return $this->LibStatut((isset($this->statut) ? $this->statut : $this->status), $mode);
2173  }
2174 
2175  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2183  public function LibStatut($status, $mode = 1)
2184  {
2185  // phpcs:enable
2186 
2187  // Init/load array of translation of status
2188  if (empty($this->labelStatus) || empty($this->labelStatusShort)) {
2189  global $langs;
2190  $langs->load("supplier_proposal");
2191  $this->labelStatus[self::STATUS_DRAFT] = $langs->transnoentitiesnoconv("SupplierProposalStatusDraft");
2192  $this->labelStatus[self::STATUS_VALIDATED] = $langs->transnoentitiesnoconv("SupplierProposalStatusValidated");
2193  $this->labelStatus[self::STATUS_SIGNED] = $langs->transnoentitiesnoconv("SupplierProposalStatusSigned");
2194  $this->labelStatus[self::STATUS_NOTSIGNED] = $langs->transnoentitiesnoconv("SupplierProposalStatusNotSigned");
2195  $this->labelStatus[self::STATUS_CLOSE] = $langs->transnoentitiesnoconv("SupplierProposalStatusClosed");
2196  $this->labelStatusShort[self::STATUS_DRAFT] = $langs->transnoentitiesnoconv("SupplierProposalStatusDraftShort");
2197  $this->labelStatusShort[self::STATUS_VALIDATED] = $langs->transnoentitiesnoconv("SupplierProposalStatusValidatedShort");
2198  $this->labelStatusShort[self::STATUS_SIGNED] = $langs->transnoentitiesnoconv("SupplierProposalStatusSignedShort");
2199  $this->labelStatusShort[self::STATUS_NOTSIGNED] = $langs->transnoentitiesnoconv("SupplierProposalStatusNotSignedShort");
2200  $this->labelStatusShort[self::STATUS_CLOSE] = $langs->transnoentitiesnoconv("SupplierProposalStatusClosedShort");
2201  }
2202 
2203  $statusnew = '';
2204  if ($status == self::STATUS_DRAFT) {
2205  $statusnew = 'status0';
2206  } elseif ($status == self::STATUS_VALIDATED) {
2207  $statusnew = 'status1';
2208  } elseif ($status == self::STATUS_SIGNED) {
2209  $statusnew = 'status4';
2210  } elseif ($status == self::STATUS_NOTSIGNED) {
2211  $statusnew = 'status9';
2212  } elseif ($status == self::STATUS_CLOSE) {
2213  $statusnew = 'status6';
2214  }
2215 
2216  return dolGetStatus($this->labelStatus[$status], $this->labelStatusShort[$status], '', $statusnew, $mode);
2217  }
2218 
2219 
2220  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2228  public function load_board($user, $mode)
2229  {
2230  // phpcs:enable
2231  global $conf, $user, $langs;
2232 
2233  $now = dol_now();
2234 
2235  $clause = " WHERE";
2236 
2237  $sql = "SELECT p.rowid, p.ref, p.datec as datec, p.date_cloture as datefin";
2238  $sql .= " FROM ".MAIN_DB_PREFIX."supplier_proposal as p";
2239  if (empty($user->rights->societe->client->voir) && !$user->socid) {
2240  $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe_commerciaux as sc ON p.fk_soc = sc.fk_soc";
2241  $sql .= " WHERE sc.fk_user = ".((int) $user->id);
2242  $clause = " AND";
2243  }
2244  $sql .= $clause." p.entity IN (".getEntity('supplier_proposal').")";
2245  if ($mode == 'opened') {
2246  $sql .= " AND p.fk_statut = 1";
2247  }
2248  if ($mode == 'signed') {
2249  $sql .= " AND p.fk_statut = 2";
2250  }
2251  if ($user->socid) {
2252  $sql .= " AND p.fk_soc = ".((int) $user->socid);
2253  }
2254 
2255  $resql = $this->db->query($sql);
2256  if ($resql) {
2257  $label = $labelShort = '';
2258  $status = '';
2259  if ($mode == 'opened') {
2260  $delay_warning = !empty($conf->supplier_proposal->cloture->warning_delay) ? $conf->supplier_proposal->cloture->warning_delay : 0;
2261  $status = self::STATUS_VALIDATED;
2262  $label = $langs->trans("SupplierProposalsToClose");
2263  $labelShort = $langs->trans("ToAcceptRefuse");
2264  }
2265  if ($mode == 'signed') {
2266  $delay_warning = !empty($conf->supplier_proposal->facturation->warning_delay) ? $conf->supplier_proposal->facturation->warning_delay : 0;
2267  $status = self::STATUS_SIGNED;
2268  $label = $langs->trans("SupplierProposalsToProcess"); // May be billed or ordered
2269  $labelShort = $langs->trans("ToClose");
2270  }
2271 
2272  $response = new WorkboardResponse();
2273  $response->warning_delay = $delay_warning / 60 / 60 / 24;
2274  $response->label = $label;
2275  $response->labelShort = $labelShort;
2276  $response->url = DOL_URL_ROOT.'/supplier_proposal/list.php?search_status='.$status;
2277  $response->img = img_object('', "propal");
2278 
2279  // This assignment in condition is not a bug. It allows walking the results.
2280  while ($obj = $this->db->fetch_object($resql)) {
2281  $response->nbtodo++;
2282  if ($mode == 'opened') {
2283  $datelimit = $this->db->jdate($obj->datefin);
2284  if ($datelimit < ($now - $delay_warning)) {
2285  $response->nbtodolate++;
2286  }
2287  }
2288  // TODO Definir regle des propales a facturer en retard
2289  // if ($mode == 'signed' && ! count($this->FactureListeArray($obj->rowid))) $this->nbtodolate++;
2290  }
2291  return $response;
2292  } else {
2293  $this->error = $this->db->lasterror();
2294  return -1;
2295  }
2296  }
2297 
2298 
2306  public function initAsSpecimen()
2307  {
2308  global $user, $langs, $conf;
2309 
2310  // Load array of products prodids
2311  $num_prods = 0;
2312  $prodids = array();
2313  $sql = "SELECT rowid";
2314  $sql .= " FROM ".MAIN_DB_PREFIX."product";
2315  $sql .= " WHERE entity IN (".getEntity('product').")";
2316  $sql .= $this->db->plimit(100);
2317 
2318  $resql = $this->db->query($sql);
2319  if ($resql) {
2320  $num_prods = $this->db->num_rows($resql);
2321  $i = 0;
2322  while ($i < $num_prods) {
2323  $i++;
2324  $row = $this->db->fetch_row($resql);
2325  $prodids[$i] = $row[0];
2326  }
2327  }
2328 
2329  // Initialise parametres
2330  $this->id = 0;
2331  $this->ref = 'SPECIMEN';
2332  $this->specimen = 1;
2333  $this->socid = 1;
2334  $this->date = time();
2335  $this->cond_reglement_id = 1;
2336  $this->cond_reglement_code = 'RECEP';
2337  $this->mode_reglement_id = 7;
2338  $this->mode_reglement_code = 'CHQ';
2339  $this->note_public = 'This is a comment (public)';
2340  $this->note_private = 'This is a comment (private)';
2341  // Lines
2342  $nbp = 5;
2343  $xnbp = 0;
2344  while ($xnbp < $nbp) {
2345  $line = new SupplierProposalLine($this->db);
2346  $line->desc = $langs->trans("Description")." ".$xnbp;
2347  $line->qty = 1;
2348  $line->subprice = 100;
2349  $line->tva_tx = 19.6;
2350  $line->localtax1_tx = 0;
2351  $line->localtax2_tx = 0;
2352  if ($xnbp == 2) {
2353  $line->total_ht = 50;
2354  $line->total_ttc = 59.8;
2355  $line->total_tva = 9.8;
2356  $line->remise_percent = 50;
2357  } else {
2358  $line->total_ht = 100;
2359  $line->total_ttc = 119.6;
2360  $line->total_tva = 19.6;
2361  $line->remise_percent = 00;
2362  }
2363 
2364  if ($num_prods > 0) {
2365  $prodid = mt_rand(1, $num_prods);
2366  $line->fk_product = $prodids[$prodid];
2367  }
2368 
2369  $this->lines[$xnbp] = $line;
2370 
2371  $this->total_ht += $line->total_ht;
2372  $this->total_tva += $line->total_tva;
2373  $this->total_ttc += $line->total_ttc;
2374 
2375  $xnbp++;
2376  }
2377  }
2378 
2379  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
2385  public function load_state_board()
2386  {
2387  // phpcs:enable
2388  global $conf, $user;
2389 
2390  $this->nb = array();
2391  $clause = "WHERE";
2392 
2393  $sql = "SELECT count(p.rowid) as nb";
2394  $sql .= " FROM ".MAIN_DB_PREFIX."supplier_proposal as p";
2395  $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON p.fk_soc = s.rowid";
2396  if (empty($user->rights->societe->client->voir) && !$user->socid) {
2397  $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe_commerciaux as sc ON s.rowid = sc.fk_soc";
2398  $sql .= " WHERE sc.fk_user = ".((int) $user->id);
2399  $clause = "AND";
2400  }
2401  $sql .= " ".$clause." p.entity IN (".getEntity('supplier_proposal').")";
2402 
2403  $resql = $this->db->query($sql);
2404  if ($resql) {
2405  // This assignment in condition is not a bug. It allows walking the results.
2406  while ($obj = $this->db->fetch_object($resql)) {
2407  $this->nb["supplier_proposals"] = $obj->nb;
2408  }
2409  $this->db->free($resql);
2410  return 1;
2411  } else {
2412  dol_print_error($this->db);
2413  $this->error = $this->db->lasterror();
2414  return -1;
2415  }
2416  }
2417 
2418 
2426  public function getNextNumRef($soc)
2427  {
2428  global $conf, $db, $langs;
2429  $langs->load("supplier_proposal");
2430 
2431  if (!empty($conf->global->SUPPLIER_PROPOSAL_ADDON)) {
2432  $mybool = false;
2433 
2434  $file = $conf->global->SUPPLIER_PROPOSAL_ADDON.".php";
2435  $classname = $conf->global->SUPPLIER_PROPOSAL_ADDON;
2436 
2437  // Include file with class
2438  $dirmodels = array_merge(array('/'), (array) $conf->modules_parts['models']);
2439  foreach ($dirmodels as $reldir) {
2440  $dir = dol_buildpath($reldir."core/modules/supplier_proposal/");
2441 
2442  // Load file with numbering class (if found)
2443  $mybool |= @include_once $dir.$file;
2444  }
2445 
2446  if (!$mybool) {
2447  dol_print_error('', "Failed to include file ".$file);
2448  return '';
2449  }
2450 
2451  $obj = new $classname();
2452  $numref = "";
2453  $numref = $obj->getNextValue($soc, $this);
2454 
2455  if ($numref != "") {
2456  return $numref;
2457  } else {
2458  $this->error = $obj->error;
2459  return "";
2460  }
2461  } else {
2462  $langs->load("errors");
2463  print $langs->trans("Error")." ".$langs->trans("ErrorModuleSetupNotComplete", $langs->transnoentitiesnoconv("SupplierProposal"));
2464  return "";
2465  }
2466  }
2467 
2475  public function getTooltipContentArray($params)
2476  {
2477  global $conf, $langs, $menumanager;
2478 
2479  $langs->load('supplier_proposal');
2480 
2481  if (!empty($conf->global->MAIN_OPTIMIZEFORTEXTBROWSER)) {
2482  return ['optimize' => $langs->trans("ShowSupplierProposal")];
2483  }
2484 
2485  $option = $params['option'] ?? '';
2486  $datas = [];
2487 
2488  $datas['picto'] = img_picto('', $this->picto).' <u class="paddingrightonly">'.$langs->trans("SupplierProposal").'</u>';
2489  if (isset($this->status)) {
2490  $datas['picto'] .= ' '.$this->getLibStatut(5);
2491  }
2492  if (!empty($this->ref)) {
2493  $datas['ref'] = '<br><b>'.$langs->trans('Ref').':</b> '.$this->ref;
2494  }
2495  if (!empty($this->ref_fourn)) {
2496  $datas['ref_supplier'] = '<br><b>'.$langs->trans('RefSupplier').':</b> '.$this->ref_fourn;
2497  }
2498  if (!empty($this->total_ht)) {
2499  $datas['amount_ht'] = '<br><b>'.$langs->trans('AmountHT').':</b> '.price($this->total_ht, 0, $langs, 0, -1, -1, $conf->currency);
2500  }
2501  if (!empty($this->total_tva)) {
2502  $datas['amount_vat'] = '<br><b>'.$langs->trans('VAT').':</b> '.price($this->total_tva, 0, $langs, 0, -1, -1, $conf->currency);
2503  }
2504  if (!empty($this->total_ttc)) {
2505  $datas['amount_ttc'] = '<br><b>'.$langs->trans('AmountTTC').':</b> '.price($this->total_ttc, 0, $langs, 0, -1, -1, $conf->currency);
2506  }
2507 
2508  return $datas;
2509  }
2510 
2522  public function getNomUrl($withpicto = 0, $option = '', $get_params = '', $notooltip = 0, $save_lastsearch_value = -1, $addlinktonotes = 0)
2523  {
2524  global $langs, $conf, $user, $hookmanager;
2525 
2526  if (!empty($conf->dol_no_mouse_hover)) {
2527  $notooltip = 1; // Force disable tooltips
2528  }
2529 
2530  $url = '';
2531  $result = '';
2532  $params = [
2533  'id' => $this->id,
2534  'objecttype' => $this->element,
2535  'option' => $option,
2536  ];
2537  $classfortooltip = 'classfortooltip';
2538  $dataparams = '';
2539  if (getDolGlobalInt('MAIN_ENABLE_AJAX_TOOLTIP')) {
2540  $classfortooltip = 'classforajaxtooltip';
2541  $dataparams = ' data-params="'.dol_escape_htmltag(json_encode($params)).'"';
2542  $label = '';
2543  } else {
2544  $label = implode($this->getTooltipContentArray($params));
2545  }
2546 
2547  if ($option == '') {
2548  $url = DOL_URL_ROOT.'/supplier_proposal/card.php?id='.$this->id.$get_params;
2549  }
2550  if ($option == 'document') {
2551  $url = DOL_URL_ROOT.'/supplier_proposal/document.php?id='.$this->id.$get_params;
2552  }
2553 
2554  if ($option !== 'nolink') {
2555  // Add param to save lastsearch_values or not
2556  $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
2557  if ($save_lastsearch_value == -1 && preg_match('/list\.php/', $_SERVER["PHP_SELF"])) {
2558  $add_save_lastsearch_values = 1;
2559  }
2560  if ($add_save_lastsearch_values) {
2561  $url .= '&save_lastsearch_values=1';
2562  }
2563  }
2564 
2565  $linkclose = '';
2566  if (empty($notooltip) && $user->hasRight('propal', 'lire')) {
2567  if (!empty($conf->global->MAIN_OPTIMIZEFORTEXTBROWSER)) {
2568  $label = $langs->trans("ShowSupplierProposal");
2569  $linkclose .= ' alt="'.dol_escape_htmltag($label, 1).'"';
2570  }
2571  $linkclose .= ($label ? ' title="'.dol_escape_htmltag($label, 1).'"' : ' title="tocomplete"');
2572  $linkclose .= $dataparams.' class="'.$classfortooltip.'"';
2573  }
2574 
2575  $linkstart = '<a href="'.$url.'"';
2576  $linkstart .= $linkclose.'>';
2577  $linkend = '</a>';
2578 
2579  $result .= $linkstart;
2580  if ($withpicto) {
2581  $result .= img_object(($notooltip ? '' : $label), $this->picto, ($notooltip ? (($withpicto != 2) ? 'class="paddingright"' : '') : $dataparams.' class="'.(($withpicto != 2) ? 'paddingright ' : '').$classfortooltip.'"'), 0, 0, $notooltip ? 0 : 1);
2582  }
2583  if ($withpicto != 2) {
2584  $result .= $this->ref;
2585  }
2586  $result .= $linkend;
2587 
2588  if ($addlinktonotes) {
2589  $txttoshow = ($user->socid > 0 ? $this->note_public : $this->note_private);
2590  if ($txttoshow) {
2591  $notetoshow = $langs->trans("ViewPrivateNote").':<br>'.dol_string_nohtmltag($txttoshow, 1);
2592  $result .= ' <span class="note inline-block">';
2593  $result .= '<a href="'.DOL_URL_ROOT.'/supplier_proposal/note.php?id='.$this->id.'" class="classfortooltip" title="'.dol_escape_htmltag($notetoshow).'">';
2594  $result .= img_picto('', 'note');
2595  $result .= '</a>';
2596  //$result.=img_picto($langs->trans("ViewNote"),'object_generic');
2597  //$result.='</a>';
2598  $result .= '</span>';
2599  }
2600  }
2601  global $action;
2602  $hookmanager->initHooks(array($this->element . 'dao'));
2603  $parameters = array('id'=>$this->id, 'getnomurl' => &$result);
2604  $reshook = $hookmanager->executeHooks('getNomUrl', $parameters, $this, $action); // Note that $action and $object may have been modified by some hooks
2605  if ($reshook > 0) {
2606  $result = $hookmanager->resPrint;
2607  } else {
2608  $result .= $hookmanager->resPrint;
2609  }
2610  return $result;
2611  }
2612 
2618  public function getLinesArray()
2619  {
2620  // For other object, here we call fetch_lines. But fetch_lines does not exists on supplier proposal
2621 
2622  $sql = 'SELECT pt.rowid, pt.label as custom_label, pt.description, pt.fk_product, pt.fk_remise_except,';
2623  $sql .= ' pt.qty, pt.tva_tx, pt.vat_src_code, pt.remise_percent, pt.subprice, pt.info_bits,';
2624  $sql .= ' pt.total_ht, pt.total_tva, pt.total_ttc, pt.fk_product_fournisseur_price as fk_fournprice, pt.buy_price_ht as pa_ht, pt.special_code, pt.localtax1_tx, pt.localtax2_tx,';
2625  $sql .= ' pt.product_type, pt.rang, pt.fk_parent_line,';
2626  $sql .= ' p.label as product_label, p.ref, p.fk_product_type, p.rowid as prodid,';
2627  $sql .= ' p.description as product_desc, pt.ref_fourn as ref_supplier,';
2628  $sql .= ' pt.fk_multicurrency, pt.multicurrency_code, pt.multicurrency_subprice, pt.multicurrency_total_ht, pt.multicurrency_total_tva, pt.multicurrency_total_ttc, pt.fk_unit';
2629  $sql .= ' FROM '.MAIN_DB_PREFIX.'supplier_proposaldet as pt';
2630  $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product as p ON pt.fk_product=p.rowid';
2631  $sql .= ' WHERE pt.fk_supplier_proposal = '.((int) $this->id);
2632  $sql .= ' ORDER BY pt.rang ASC, pt.rowid';
2633 
2634  dol_syslog(get_class($this).'::getLinesArray', LOG_DEBUG);
2635  $resql = $this->db->query($sql);
2636  if ($resql) {
2637  $num = $this->db->num_rows($resql);
2638  $i = 0;
2639 
2640  while ($i < $num) {
2641  $obj = $this->db->fetch_object($resql);
2642 
2643  $this->lines[$i] = new SupplierProposalLine($this->db);
2644  $this->lines[$i]->id = $obj->rowid; // for backward compatibility
2645  $this->lines[$i]->rowid = $obj->rowid;
2646  $this->lines[$i]->label = $obj->custom_label;
2647  $this->lines[$i]->description = $obj->description;
2648  $this->lines[$i]->fk_product = $obj->fk_product;
2649  $this->lines[$i]->ref = $obj->ref;
2650  $this->lines[$i]->product_label = $obj->product_label;
2651  $this->lines[$i]->product_desc = $obj->product_desc;
2652  $this->lines[$i]->fk_product_type = $obj->fk_product_type; // deprecated
2653  $this->lines[$i]->product_type = $obj->product_type;
2654  $this->lines[$i]->qty = $obj->qty;
2655  $this->lines[$i]->subprice = $obj->subprice;
2656  $this->lines[$i]->fk_remise_except = $obj->fk_remise_except;
2657  $this->lines[$i]->remise_percent = $obj->remise_percent;
2658  $this->lines[$i]->tva_tx = $obj->tva_tx;
2659  $this->lines[$i]->vat_src_code = $obj->vat_src_code;
2660  $this->lines[$i]->info_bits = $obj->info_bits;
2661  $this->lines[$i]->total_ht = $obj->total_ht;
2662  $this->lines[$i]->total_tva = $obj->total_tva;
2663  $this->lines[$i]->total_ttc = $obj->total_ttc;
2664  $this->lines[$i]->fk_fournprice = $obj->fk_fournprice;
2665  $marginInfos = getMarginInfos($obj->subprice, $obj->remise_percent, $obj->tva_tx, $obj->localtax1_tx, $obj->localtax2_tx, $this->lines[$i]->fk_fournprice, $obj->pa_ht);
2666  $this->lines[$i]->pa_ht = $marginInfos[0];
2667  $this->lines[$i]->marge_tx = $marginInfos[1];
2668  $this->lines[$i]->marque_tx = $marginInfos[2];
2669  $this->lines[$i]->fk_parent_line = $obj->fk_parent_line;
2670  $this->lines[$i]->special_code = $obj->special_code;
2671  $this->lines[$i]->rang = $obj->rang;
2672 
2673  $this->lines[$i]->ref_fourn = $obj->ref_supplier; // deprecated
2674  $this->lines[$i]->ref_supplier = $obj->ref_supplier;
2675 
2676  // Multicurrency
2677  $this->lines[$i]->fk_multicurrency = $obj->fk_multicurrency;
2678  $this->lines[$i]->multicurrency_code = $obj->multicurrency_code;
2679  $this->lines[$i]->multicurrency_subprice = $obj->multicurrency_subprice;
2680  $this->lines[$i]->multicurrency_total_ht = $obj->multicurrency_total_ht;
2681  $this->lines[$i]->multicurrency_total_tva = $obj->multicurrency_total_tva;
2682  $this->lines[$i]->multicurrency_total_ttc = $obj->multicurrency_total_ttc;
2683  $this->lines[$i]->fk_unit = $obj->fk_unit;
2684 
2685  $i++;
2686  }
2687  $this->db->free($resql);
2688 
2689  return 1;
2690  } else {
2691  $this->error = $this->db->error();
2692  return -1;
2693  }
2694  }
2695 
2707  public function generateDocument($modele, $outputlangs, $hidedetails = 0, $hidedesc = 0, $hideref = 0, $moreparams = null)
2708  {
2709  global $conf, $langs;
2710 
2711  $langs->load("supplier_proposal");
2712  $outputlangs->load("products");
2713 
2714  if (!dol_strlen($modele)) {
2715  $modele = 'aurore';
2716 
2717  if ($this->model_pdf) {
2718  $modele = $this->model_pdf;
2719  } elseif (!empty($conf->global->SUPPLIER_PROPOSAL_ADDON_PDF)) {
2720  $modele = $conf->global->SUPPLIER_PROPOSAL_ADDON_PDF;
2721  }
2722  }
2723 
2724  $modelpath = "core/modules/supplier_proposal/doc/";
2725 
2726  return $this->commonGenerateDocument($modelpath, $modele, $outputlangs, $hidedetails, $hidedesc, $hideref, $moreparams);
2727  }
2728 
2729 
2738  public static function replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
2739  {
2740  $tables = array(
2741  'supplier_proposal'
2742  );
2743 
2744  return CommonObject::commonReplaceThirdparty($dbs, $origin_id, $dest_id, $tables);
2745  }
2746 
2755  public static function replaceProduct(DoliDB $db, $origin_id, $dest_id)
2756  {
2757  $tables = array(
2758  'supplier_proposaldet'
2759  );
2760 
2761  return CommonObject::commonReplaceProduct($db, $origin_id, $dest_id, $tables);
2762  }
2763 
2764 
2772  public function getKanbanView($option = '', $arraydata = null)
2773  {
2774  global $langs;
2775 
2776  $selected = (empty($arraydata['selected']) ? 0 : $arraydata['selected']);
2777 
2778  $return = '<div class="box-flex-item box-flex-grow-zero">';
2779  $return .= '<div class="info-box info-box-sm">';
2780  $return .= '<span class="info-box-icon bg-infobox-action">';
2781  $return .= img_picto('', $this->picto);
2782  //$return .= '<i class="fa fa-dol-action"></i>'; // Can be image
2783  $return .= '</span>';
2784  $return .= '<div class="info-box-content">';
2785  $return .= '<span class="info-box-ref inline-block tdoverflowmax150 valignmiddle">'.(method_exists($this, 'getNomUrl') ? $this->getNomUrl() : $this->ref).'</span>';
2786  $return .= '<input id="cb'.$this->id.'" class="flat checkforselect fright" type="checkbox" name="toselect[]" value="'.$this->id.'"'.($selected ? ' checked="checked"' : '').'>';
2787  if (property_exists($this, 'socid')) {
2788  $return .= '<span class="info-box-ref"> | '.$this->socid.'</span>';
2789  }
2790  if (property_exists($this, 'delivery_date')) {
2791  $return .= '<br><span class="opacitymedium">'.$langs->trans("DateEnd").'</span> : <span class="info-box-label">'.dol_print_date($this->delivery_date).'</span>';
2792  }
2793  if (property_exists($this, 'total_ttc')) {
2794  $return .='<br><span class="opacitymedium" >'.$langs->trans("AmountHT").' : </span><span class="info-box-label amount">'.price($this->total_ttc).'</span>';
2795  }
2796  if (method_exists($this, 'getLibStatut')) {
2797  $return .= '<br><div class="info-box-status margintoponly">'.$this->getLibStatut(3).'</div>';
2798  }
2799  $return .= '</div>';
2800  $return .= '</div>';
2801  $return .= '</div>';
2802  return $return;
2803  }
2804 }
2805 
2806 
2811 {
2815  public $db;
2816 
2820  public $error = '';
2821 
2825  public $element = 'supplier_proposaldet';
2826 
2830  public $table_element = 'supplier_proposaldet';
2831 
2832  public $oldline;
2833 
2837  public $id;
2838 
2842  public $fk_supplier_proposal;
2843 
2847  public $fk_parent_line;
2848 
2849  public $desc; // Description ligne
2850 
2854  public $fk_product; // Id produit predefini
2855 
2866  public $product_type = Product::TYPE_PRODUCT;
2867 
2868  public $qty;
2869  public $tva_tx;
2870  public $vat_src_code;
2871 
2872  public $subprice;
2873  public $remise_percent;
2874 
2878  public $fk_remise_except;
2879 
2880  public $rang = 0;
2881 
2885  public $fk_fournprice;
2886 
2887  public $pa_ht;
2888  public $marge_tx;
2889  public $marque_tx;
2890 
2891  public $special_code; // Tag for special lines (exlusive tags)
2892  // 1: frais de port
2893  // 2: ecotaxe
2894  // 3: option line (when qty = 0)
2895 
2896  public $info_bits = 0; // Liste d'options cumulables:
2897  // Bit 0: 0 si TVA normal - 1 si TVA NPR
2898  // Bit 1: 0 ligne normale - 1 si ligne de remise fixe
2899 
2900  public $total_ht; // Total HT de la ligne toute quantite et incluant la remise ligne
2901  public $total_tva; // Total TVA de la ligne toute quantite et incluant la remise ligne
2902  public $total_ttc; // Total TTC de la ligne toute quantite et incluant la remise ligne
2903 
2904  public $date_start;
2905  public $date_end;
2906 
2907  // From llx_product
2912  public $ref;
2913 
2918  public $product_ref;
2919 
2924  public $libelle;
2925 
2930  public $product_label;
2931 
2936  public $product_desc;
2937 
2938  public $localtax1_tx; // Local tax 1
2939  public $localtax2_tx; // Local tax 2
2940  public $localtax1_type; // Local tax 1 type
2941  public $localtax2_type; // Local tax 2 type
2942  public $total_localtax1; // Line total local tax 1
2943  public $total_localtax2; // Line total local tax 2
2944 
2945  public $skip_update_total; // Skip update price total for special lines
2946 
2947  public $ref_fourn;
2948  public $ref_supplier;
2949 
2950  // Multicurrency
2954  public $fk_multicurrency;
2955 
2956  public $multicurrency_code;
2957  public $multicurrency_subprice;
2958  public $multicurrency_total_ht;
2959  public $multicurrency_total_tva;
2960  public $multicurrency_total_ttc;
2961 
2967  public function __construct($db)
2968  {
2969  $this->db = $db;
2970  }
2971 
2978  public function fetch($rowid)
2979  {
2980  $sql = 'SELECT pd.rowid, pd.fk_supplier_proposal, pd.fk_parent_line, pd.fk_product, pd.label as custom_label, pd.description, pd.price, pd.qty, pd.tva_tx,';
2981  $sql .= ' pd.date_start, pd.date_end,';
2982  $sql .= ' pd.remise, pd.remise_percent, pd.fk_remise_except, pd.subprice,';
2983  $sql .= ' pd.info_bits, pd.total_ht, pd.total_tva, pd.total_ttc, pd.fk_product_fournisseur_price as fk_fournprice, pd.buy_price_ht as pa_ht, pd.special_code, pd.rang,';
2984  $sql .= ' pd.localtax1_tx, pd.localtax2_tx, pd.total_localtax1, pd.total_localtax2,';
2985  $sql .= ' p.ref as product_ref, p.label as product_label, p.description as product_desc,';
2986  $sql .= ' pd.product_type, pd.ref_fourn as ref_produit_fourn,';
2987  $sql .= ' pd.fk_multicurrency, pd.multicurrency_code, pd.multicurrency_subprice, pd.multicurrency_total_ht, pd.multicurrency_total_tva, pd.multicurrency_total_ttc, pd.fk_unit';
2988  $sql .= ' FROM '.MAIN_DB_PREFIX.'supplier_proposaldet as pd';
2989  $sql .= ' LEFT JOIN '.MAIN_DB_PREFIX.'product as p ON pd.fk_product = p.rowid';
2990  $sql .= ' WHERE pd.rowid = '.((int) $rowid);
2991 
2992  $result = $this->db->query($sql);
2993  if ($result) {
2994  $objp = $this->db->fetch_object($result);
2995 
2996  $this->id = $objp->rowid;
2997  $this->fk_supplier_proposal = $objp->fk_supplier_proposal;
2998  $this->fk_parent_line = $objp->fk_parent_line;
2999  $this->label = $objp->custom_label;
3000  $this->desc = $objp->description;
3001  $this->qty = $objp->qty;
3002  $this->subprice = $objp->subprice;
3003  $this->tva_tx = $objp->tva_tx;
3004  $this->remise_percent = $objp->remise_percent;
3005  $this->fk_remise_except = $objp->fk_remise_except;
3006  $this->fk_product = $objp->fk_product;
3007  $this->info_bits = $objp->info_bits;
3008  $this->date_start = $this->db->jdate($objp->date_start);
3009  $this->date_end = $this->db->jdate($objp->date_end);
3010 
3011  $this->total_ht = $objp->total_ht;
3012  $this->total_tva = $objp->total_tva;
3013  $this->total_ttc = $objp->total_ttc;
3014 
3015  $this->fk_fournprice = $objp->fk_fournprice;
3016 
3017  $marginInfos = getMarginInfos($objp->subprice, $objp->remise_percent, $objp->tva_tx, $objp->localtax1_tx, $objp->localtax2_tx, $this->fk_fournprice, $objp->pa_ht);
3018  $this->pa_ht = $marginInfos[0];
3019  $this->marge_tx = $marginInfos[1];
3020  $this->marque_tx = $marginInfos[2];
3021 
3022  $this->special_code = $objp->special_code;
3023  $this->product_type = $objp->product_type;
3024  $this->rang = $objp->rang;
3025 
3026  $this->ref = $objp->product_ref; // deprecated
3027  $this->product_ref = $objp->product_ref;
3028  $this->libelle = $objp->product_label; // deprecated
3029  $this->product_label = $objp->product_label;
3030  $this->product_desc = $objp->product_desc;
3031 
3032  $this->ref_fourn = $objp->ref_produit_fourn;
3033 
3034  // Multicurrency
3035  $this->fk_multicurrency = $objp->fk_multicurrency;
3036  $this->multicurrency_code = $objp->multicurrency_code;
3037  $this->multicurrency_subprice = $objp->multicurrency_subprice;
3038  $this->multicurrency_total_ht = $objp->multicurrency_total_ht;
3039  $this->multicurrency_total_tva = $objp->multicurrency_total_tva;
3040  $this->multicurrency_total_ttc = $objp->multicurrency_total_ttc;
3041  $this->fk_unit = $objp->fk_unit;
3042 
3043  $this->db->free($result);
3044  return 1;
3045  } else {
3046  dol_print_error($this->db);
3047  return -1;
3048  }
3049  }
3050 
3057  public function insert($notrigger = 0)
3058  {
3059  global $conf, $langs, $user;
3060 
3061  $error = 0;
3062 
3063  dol_syslog(get_class($this)."::insert rang=".$this->rang);
3064 
3065  // Clean parameters
3066  if (empty($this->tva_tx)) {
3067  $this->tva_tx = 0;
3068  }
3069  if (empty($this->vat_src_code)) {
3070  $this->vat_src_code = '';
3071  }
3072  if (empty($this->localtax1_tx)) {
3073  $this->localtax1_tx = 0;
3074  }
3075  if (empty($this->localtax2_tx)) {
3076  $this->localtax2_tx = 0;
3077  }
3078  if (empty($this->localtax1_type)) {
3079  $this->localtax1_type = 0;
3080  }
3081  if (empty($this->localtax2_type)) {
3082  $this->localtax2_type = 0;
3083  }
3084  if (empty($this->total_localtax1)) {
3085  $this->total_localtax1 = 0;
3086  }
3087  if (empty($this->total_localtax2)) {
3088  $this->total_localtax2 = 0;
3089  }
3090  if (empty($this->rang)) {
3091  $this->rang = 0;
3092  }
3093  if (empty($this->remise_percent)) {
3094  $this->remise_percent = 0;
3095  }
3096  if (empty($this->info_bits)) {
3097  $this->info_bits = 0;
3098  }
3099  if (empty($this->special_code)) {
3100  $this->special_code = 0;
3101  }
3102  if (empty($this->fk_parent_line)) {
3103  $this->fk_parent_line = 0;
3104  }
3105  if (empty($this->fk_fournprice)) {
3106  $this->fk_fournprice = 0;
3107  }
3108  if (empty($this->fk_unit)) {
3109  $this->fk_unit = 0;
3110  }
3111  if (empty($this->subprice)) {
3112  $this->subprice = 0;
3113  }
3114 
3115  if (empty($this->pa_ht)) {
3116  $this->pa_ht = 0;
3117  }
3118 
3119  // if buy price not defined, define buyprice as configured in margin admin
3120  if ($this->pa_ht == 0) {
3121  $result = $this->defineBuyPrice($this->subprice, $this->remise_percent, $this->fk_product);
3122  if ($result < 0) {
3123  return $result;
3124  } else {
3125  $this->pa_ht = $result;
3126  }
3127  }
3128 
3129  // Check parameters
3130  if ($this->product_type < 0) {
3131  return -1;
3132  }
3133 
3134  $this->db->begin();
3135 
3136  // Insert line into database
3137  $sql = 'INSERT INTO '.MAIN_DB_PREFIX.'supplier_proposaldet';
3138  $sql .= ' (fk_supplier_proposal, fk_parent_line, label, description, fk_product, product_type,';
3139  $sql .= ' date_start, date_end,';
3140  $sql .= ' fk_remise_except, qty, tva_tx, vat_src_code, localtax1_tx, localtax2_tx, localtax1_type, localtax2_type,';
3141  $sql .= ' subprice, remise_percent, ';
3142  $sql .= ' info_bits, ';
3143  $sql .= ' total_ht, total_tva, total_localtax1, total_localtax2, total_ttc, fk_product_fournisseur_price, buy_price_ht, special_code, rang,';
3144  $sql .= ' ref_fourn,';
3145  $sql .= ' fk_multicurrency, multicurrency_code, multicurrency_subprice, multicurrency_total_ht, multicurrency_total_tva, multicurrency_total_ttc, fk_unit)';
3146  $sql .= " VALUES (".$this->fk_supplier_proposal.",";
3147  $sql .= " ".($this->fk_parent_line > 0 ? ((int) $this->fk_parent_line) : "null").",";
3148  $sql .= " ".(!empty($this->label) ? "'".$this->db->escape($this->label)."'" : "null").",";
3149  $sql .= " '".$this->db->escape($this->desc)."',";
3150  $sql .= " ".($this->fk_product ? ((int) $this->fk_product) : "null").",";
3151  $sql .= " '".$this->db->escape($this->product_type)."',";
3152  $sql .= " ".($this->date_start ? "'".$this->db->idate($this->date_start)."'" : "null").",";
3153  $sql .= " ".($this->date_end ? "'".$this->db->idate($this->date_end)."'" : "null").",";
3154  $sql .= " ".($this->fk_remise_except ? ((int) $this->fk_remise_except) : "null").",";
3155  $sql .= " ".price2num($this->qty, 'MS').",";
3156  $sql .= " ".price2num($this->tva_tx).",";
3157  $sql .= " '".$this->db->escape($this->vat_src_code)."',";
3158  $sql .= " ".price2num($this->localtax1_tx).",";
3159  $sql .= " ".price2num($this->localtax2_tx).",";
3160  $sql .= " '".$this->db->escape($this->localtax1_type)."',";
3161  $sql .= " '".$this->db->escape($this->localtax2_type)."',";
3162  $sql .= " ".price2num($this->subprice, 'MU') .",";
3163  $sql .= " ".((float) $this->remise_percent).",";
3164  $sql .= " ".(isset($this->info_bits) ? ((int) $this->info_bits) : "null").",";
3165  $sql .= " ".price2num($this->total_ht, 'MT').",";
3166  $sql .= " ".price2num($this->total_tva, 'MT').",";
3167  $sql .= " ".price2num($this->total_localtax1, 'MT').",";
3168  $sql .= " ".price2num($this->total_localtax2, 'MT').",";
3169  $sql .= " ".price2num($this->total_ttc, 'MT').",";
3170  $sql .= " ".(!empty($this->fk_fournprice) ? ((int) $this->fk_fournprice) : "null").",";
3171  $sql .= " ".(isset($this->pa_ht) ? price2num($this->pa_ht, 'MU') : "null").",";
3172  $sql .= ' '.((int) $this->special_code).',';
3173  $sql .= ' '.((int) $this->rang).',';
3174  $sql .= " '".$this->db->escape($this->ref_fourn)."'";
3175  $sql .= ", ".($this->fk_multicurrency > 0 ? ((int) $this->fk_multicurrency) : 'null');
3176  $sql .= ", '".$this->db->escape($this->multicurrency_code)."'";
3177  $sql .= ", ".price2num($this->multicurrency_subprice, 'CU');
3178  $sql .= ", ".price2num($this->multicurrency_total_ht, 'CT');
3179  $sql .= ", ".price2num($this->multicurrency_total_tva, 'CT');
3180  $sql .= ", ".price2num($this->multicurrency_total_ttc, 'CT');
3181  $sql .= ", ".($this->fk_unit ? ((int) $this->fk_unit) : 'null');
3182  $sql .= ')';
3183 
3184  dol_syslog(get_class($this).'::insert', LOG_DEBUG);
3185  $resql = $this->db->query($sql);
3186  if ($resql) {
3187  $this->id = $this->db->last_insert_id(MAIN_DB_PREFIX.'supplier_proposaldet');
3188 
3189  if (!$error) {
3190  $result = $this->insertExtraFields();
3191  if ($result < 0) {
3192  $error++;
3193  }
3194  }
3195 
3196  if (!$error && !$notrigger) {
3197  // Call trigger
3198  $result = $this->call_trigger('LINESUPPLIER_PROPOSAL_INSERT', $user);
3199  if ($result < 0) {
3200  $this->db->rollback();
3201  return -1;
3202  }
3203  // End call triggers
3204  }
3205 
3206  $this->db->commit();
3207  return 1;
3208  } else {
3209  $this->error = $this->db->error()." sql=".$sql;
3210  $this->db->rollback();
3211  return -1;
3212  }
3213  }
3214 
3220  public function delete()
3221  {
3222  global $conf, $langs, $user;
3223 
3224  $error = 0;
3225  $this->db->begin();
3226 
3227  $sql = "DELETE FROM ".MAIN_DB_PREFIX."supplier_proposaldet WHERE rowid = ".((int) $this->id);
3228  dol_syslog("SupplierProposalLine::delete", LOG_DEBUG);
3229  if ($this->db->query($sql)) {
3230  // Remove extrafields
3231  if (!$error) {
3232  $result = $this->deleteExtraFields();
3233  if ($result < 0) {
3234  $error++;
3235  dol_syslog(get_class($this)."::delete error -4 ".$this->error, LOG_ERR);
3236  }
3237  }
3238 
3239  // Call trigger
3240  $result = $this->call_trigger('LINESUPPLIER_PROPOSAL_DELETE', $user);
3241  if ($result < 0) {
3242  $this->db->rollback();
3243  return -1;
3244  }
3245  // End call triggers
3246 
3247  $this->db->commit();
3248 
3249  return 1;
3250  } else {
3251  $this->error = $this->db->error()." sql=".$sql;
3252  $this->db->rollback();
3253  return -1;
3254  }
3255  }
3256 
3263  public function update($notrigger = 0)
3264  {
3265  global $conf, $langs, $user;
3266 
3267  $error = 0;
3268 
3269  // Clean parameters
3270  if (empty($this->tva_tx)) {
3271  $this->tva_tx = 0;
3272  }
3273  if (empty($this->localtax1_tx)) {
3274  $this->localtax1_tx = 0;
3275  }
3276  if (empty($this->localtax2_tx)) {
3277  $this->localtax2_tx = 0;
3278  }
3279  if (empty($this->total_localtax1)) {
3280  $this->total_localtax1 = 0;
3281  }
3282  if (empty($this->total_localtax2)) {
3283  $this->total_localtax2 = 0;
3284  }
3285  if (empty($this->localtax1_type)) {
3286  $this->localtax1_type = 0;
3287  }
3288  if (empty($this->localtax2_type)) {
3289  $this->localtax2_type = 0;
3290  }
3291  if (empty($this->marque_tx)) {
3292  $this->marque_tx = 0;
3293  }
3294  if (empty($this->marge_tx)) {
3295  $this->marge_tx = 0;
3296  }
3297  if (empty($this->remise_percent)) {
3298  $this->remise_percent = 0;
3299  }
3300  if (empty($this->info_bits)) {
3301  $this->info_bits = 0;
3302  }
3303  if (empty($this->special_code)) {
3304  $this->special_code = 0;
3305  }
3306  if (empty($this->fk_parent_line)) {
3307  $this->fk_parent_line = 0;
3308  }
3309  if (empty($this->fk_fournprice)) {
3310  $this->fk_fournprice = 0;
3311  }
3312  if (empty($this->fk_unit)) {
3313  $this->fk_unit = 0;
3314  }
3315  if (empty($this->subprice)) {
3316  $this->subprice = 0;
3317  }
3318 
3319  if (empty($this->pa_ht)) {
3320  $this->pa_ht = 0;
3321  }
3322 
3323  // if buy price not defined, define buyprice as configured in margin admin
3324  if ($this->pa_ht == 0) {
3325  $result = $this->defineBuyPrice($this->subprice, $this->remise_percent, $this->fk_product);
3326  if ($result < 0) {
3327  return $result;
3328  } else {
3329  $this->pa_ht = $result;
3330  }
3331  }
3332 
3333  $this->db->begin();
3334 
3335  // Mise a jour ligne en base
3336  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposaldet SET";
3337  $sql .= " description='".$this->db->escape($this->desc)."'";
3338  $sql .= " , label=".(!empty($this->label) ? "'".$this->db->escape($this->label)."'" : "null");
3339  $sql .= " , product_type=".((int) $this->product_type);
3340  $sql .= " , date_start=".($this->date_start ? "'".$this->db->idate($this->date_start)."'" : "null");
3341  $sql .= " , date_end=".($this->date_end ? "'".$this->db->idate($this->date_end)."'" : "null");
3342  $sql .= " , tva_tx='".price2num($this->tva_tx)."'";
3343  $sql .= " , localtax1_tx=".price2num($this->localtax1_tx);
3344  $sql .= " , localtax2_tx=".price2num($this->localtax2_tx);
3345  $sql .= " , localtax1_type='".$this->db->escape($this->localtax1_type)."'";
3346  $sql .= " , localtax2_type='".$this->db->escape($this->localtax2_type)."'";
3347  $sql .= " , qty='".price2num($this->qty)."'";
3348  $sql .= " , subprice=".price2num($this->subprice);
3349  $sql .= " , remise_percent=".price2num($this->remise_percent);
3350  $sql .= " , info_bits='".$this->db->escape($this->info_bits)."'";
3351  if (empty($this->skip_update_total)) {
3352  $sql .= " , total_ht=".price2num($this->total_ht);
3353  $sql .= " , total_tva=".price2num($this->total_tva);
3354  $sql .= " , total_ttc=".price2num($this->total_ttc);
3355  $sql .= " , total_localtax1=".price2num($this->total_localtax1);
3356  $sql .= " , total_localtax2=".price2num($this->total_localtax2);
3357  }
3358  $sql .= " , fk_product_fournisseur_price=".(!empty($this->fk_fournprice) ? "'".$this->db->escape($this->fk_fournprice)."'" : "null");
3359  $sql .= " , buy_price_ht=".price2num($this->pa_ht);
3360  if (strlen($this->special_code)) {
3361  $sql .= " , special_code=".((int) $this->special_code);
3362  }
3363  $sql .= " , fk_parent_line=".($this->fk_parent_line > 0 ? $this->fk_parent_line : "null");
3364  if (!empty($this->rang)) {
3365  $sql .= ", rang=".((int) $this->rang);
3366  }
3367  $sql .= " , ref_fourn=".(!empty($this->ref_fourn) ? "'".$this->db->escape($this->ref_fourn)."'" : "null");
3368  $sql .= " , fk_unit=".($this->fk_unit ? $this->fk_unit : 'null');
3369 
3370  // Multicurrency
3371  $sql .= " , multicurrency_subprice=".price2num($this->multicurrency_subprice);
3372  $sql .= " , multicurrency_total_ht=".price2num($this->multicurrency_total_ht);
3373  $sql .= " , multicurrency_total_tva=".price2num($this->multicurrency_total_tva);
3374  $sql .= " , multicurrency_total_ttc=".price2num($this->multicurrency_total_ttc);
3375 
3376  $sql .= " WHERE rowid = ".((int) $this->id);
3377 
3378  dol_syslog(get_class($this)."::update", LOG_DEBUG);
3379  $resql = $this->db->query($sql);
3380  if ($resql) {
3381  if (!$error) {
3382  $result = $this->insertExtraFields();
3383  if ($result < 0) {
3384  $error++;
3385  }
3386  }
3387 
3388  if (!$error && !$notrigger) {
3389  // Call trigger
3390  $result = $this->call_trigger('LINESUPPLIER_PROPOSAL_MODIFY', $user);
3391  if ($result < 0) {
3392  $this->db->rollback();
3393  return -1;
3394  }
3395  // End call triggers
3396  }
3397 
3398  $this->db->commit();
3399  return 1;
3400  } else {
3401  $this->error = $this->db->error();
3402  $this->db->rollback();
3403  return -2;
3404  }
3405  }
3406 
3407  // phpcs:disable PEAR.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
3414  public function update_total()
3415  {
3416  // phpcs:enable
3417  $this->db->begin();
3418 
3419  // Mise a jour ligne en base
3420  $sql = "UPDATE ".MAIN_DB_PREFIX."supplier_proposaldet SET";
3421  $sql .= " total_ht=".price2num($this->total_ht, 'MT');
3422  $sql .= ",total_tva=".price2num($this->total_tva, 'MT');
3423  $sql .= ",total_ttc=".price2num($this->total_ttc, 'MT');
3424  $sql .= " WHERE rowid = ".((int) $this->id);
3425 
3426  dol_syslog("SupplierProposalLine::update_total", LOG_DEBUG);
3427 
3428  $resql = $this->db->query($sql);
3429  if ($resql) {
3430  $this->db->commit();
3431  return 1;
3432  } else {
3433  $this->error = $this->db->error();
3434  $this->db->rollback();
3435  return -2;
3436  }
3437  }
3438 }
$object ref
Definition: info.php:78
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...
line_order($renum=false, $rowidorder='ASC', $fk_parent_line=true)
Save a new position (field rang) for details lines.
deleteEcmFiles($mode=0)
Delete related files of object in database.
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.
setErrorsFromObject($object)
setErrorsFromObject
static isExistingObject($element, $id, $ref='', $ref_ext='')
Check an object id/ref exists If you don't need/want to instantiate object and just need to know if o...
updateRangOfLine($rowid, $rang)
Update position of line (rang)
deleteObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $rowid='', $f_user=null, $notrigger=0)
Delete all links between an object $this.
update_price($exclspec=0, $roundingadjust='none', $nodatabaseupdate=0, $seller=null)
Update total_ht, total_ttc, total_vat, total_localtax1, total_localtax2 for an object (sum of lines).
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.
static commonReplaceProduct(DoliDB $dbs, $origin_id, $dest_id, array $tables, $ignoreerrors=0)
Function used to replace a product id with another one.
line_max($fk_parent_line=0)
Get max value used for position of line (rang)
insertExtraFields($trigger='', $userused=null)
Add/Update all extra fields values for the current object.
call_trigger($triggerName, $user)
Call trigger based on this instance.
Parent class for class inheritance lines of business objects This class is useless for the moment so ...
Class to manage absolute discounts.
Class to manage Dolibarr database access.
Class Currency.
static getIdAndTxFromCode($dbs, $code, $date_document='')
Get id and rate of currency from code.
Class to manage predefined suppliers products.
Class to manage products or services.
const TYPE_PRODUCT
Regular product.
Class to manage third parties objects (customers, suppliers, prospects...)
Class to manage price ask supplier.
updatePriceFournisseur($idProductFournPrice, $product, $user)
Upate ProductFournisseur.
cloture($user, $status, $note)
Close the askprice.
set_remise_percent($user, $remise)
Set an overall discount on the proposal.
generateDocument($modele, $outputlangs, $hidedetails=0, $hidedesc=0, $hideref=0, $moreparams=null)
Create a document onto disk according to template module.
deleteline($lineid)
Delete detail line.
info($id)
Object SupplierProposal Information.
create($user, $notrigger=0)
Create commercial proposal into database this->ref can be set or empty.
insert_discount($idremise)
Adding line of fixed discount in the proposal in DB.
static replaceProduct(DoliDB $db, $origin_id, $dest_id)
Function used to replace a product id with another one.
updateline($rowid, $pu, $qty, $remise_percent, $txtva, $txlocaltax1=0, $txlocaltax2=0, $desc='', $price_base_type='HT', $info_bits=0, $special_code=0, $fk_parent_line=0, $skip_update_total=0, $fk_fournprice=0, $pa_ht=0, $label='', $type=0, $array_options=0, $ref_supplier='', $fk_unit='', $pu_ht_devise=0)
Update a proposal line.
getNextNumRef($soc)
Returns the reference to the following non used Proposal used depending on the active numbering modul...
fetch($rowid, $ref='')
Load a proposal from database and its ligne array.
load_board($user, $mode)
Load indicators for dashboard (this->nbtodo and this->nbtodolate)
createFromClone(User $user, $fromid=0)
Load an object from its id and create a new one in database.
initAsSpecimen()
Initialise an instance with random values.
const STATUS_NOTSIGNED
Not signed quote, canceled.
addline($desc, $pu_ht, $qty, $txtva, $txlocaltax1=0, $txlocaltax2=0, $fk_product=0, $remise_percent=0, $price_base_type='HT', $pu_ttc=0, $info_bits=0, $type=0, $rang=-1, $special_code=0, $fk_parent_line=0, $fk_fournprice=0, $pa_ht=0, $label='', $array_options=0, $ref_supplier='', $fk_unit='', $origin='', $origin_id=0, $pu_ht_devise=0, $date_start=0, $date_end=0)
Add a proposal line into database (linked to product/service or not) Les parametres sont deja cense e...
const STATUS_DRAFT
Draft status.
liste_array($shortlist=0, $draft=0, $notcurrentuser=0, $socid=0, $limit=0, $offset=0, $sortfield='p.datec', $sortorder='DESC')
Return list of askprice (eventually filtered on user) into an array.
setDeliveryDate($user, $delivery_date)
Set delivery date.
LibStatut($status, $mode=1)
Return label of a status (draft, validated, ...)
__construct($db, $socid="", $supplier_proposalid=0)
Constructor.
static replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
Function used to replace a thirdparty id with another one.
updateOrCreatePriceFournisseur($user)
Add or update supplier price according to result of proposal.
load_state_board()
Load indicator this->nb of global stats widget.
reopen($user, $statut, $note='', $notrigger=0)
Reopen the commercial proposal.
const STATUS_VALIDATED
Validated status.
createPriceFournisseur($product, $user)
Create ProductFournisseur.
getLibStatut($mode=0)
Return label of status of proposal (draft, validated, ...)
set_remise_absolue($user, $remise)
Set an absolute overall discount on the proposal.
add_product($idproduct, $qty, $remise_percent=0)
Add line into array ->lines.
set_date_livraison($user, $delivery_date)
Set delivery date.
const STATUS_SIGNED
Signed quote.
getNomUrl($withpicto=0, $option='', $get_params='', $notooltip=0, $save_lastsearch_value=-1, $addlinktonotes=0)
Return clicable link of object (with eventually picto)
getKanbanView($option='', $arraydata=null)
Return clicable link of object (with eventually picto)
getTooltipContentArray($params)
getTooltipContentArray
valid($user, $notrigger=0)
Set status to validated.
const STATUS_CLOSE
Billed or closed/processed quote.
getLinesArray()
Retrieve an array of supplier proposal lines.
setDraft($user)
Set draft status.
Class to manage supplier_proposal lines.
fetch($rowid)
Retrieve the propal line object.
insert($notrigger=0)
Insert object line propal in database.
update_total()
Update DB line fields total_xxx Used by migration.
update($notrigger=0)
Update propal line object into DB.
__construct($db)
Class line Contructor.
Class to manage translations.
Class to manage Dolibarr users.
Definition: user.class.php:48
trait CommonIncoterm
Superclass for incoterm classes.
if(isModEnabled('facture') && $user->hasRight('facture', 'lire')) if((isModEnabled('fournisseur') &&empty($conf->global->MAIN_USE_NEW_SUPPLIERMOD) && $user->hasRight("fournisseur", "facture", "lire"))||(isModEnabled('supplier_invoice') && $user->hasRight("supplier_invoice", "lire"))) if(isModEnabled('don') && $user->hasRight('don', 'lire')) if(isModEnabled('tax') &&!empty($user->rights->tax->charges->lire)) if(isModEnabled('facture') &&isModEnabled('commande') && $user->hasRight("commande", "lire") &&empty($conf->global->WORKFLOW_DISABLE_CREATE_INVOICE_FROM_ORDER)) $sql
Social contributions to pay.
Definition: index.php:746
print *****$script_file(".$version.") pid c cd cd cd description as p label as s rowid
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)
Definition: files.lib.php:1485
dol_delete_file($file, $disableglob=0, $nophperrors=0, $nohook=0, $object=null, $allowdotdot=false, $indexdatabase=1, $nolog=0)
Remove a file or several files with a mask.
Definition: files.lib.php:1334
dol_dir_list($path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0, $nbsecondsold=0)
Scan a directory and return a list of files/directories.
Definition: files.lib.php:62
dol_delete_preview($object)
Delete all preview files linked to object instance.
Definition: files.lib.php:1537
dol_string_nohtmltag($stringtoclean, $removelinefeed=1, $pagecodeto='UTF-8', $strip_tags=0, $removedoublespaces=1)
Clean a string from all HTML tags and entities.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
dol_print_error($db='', $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
img_object($titlealt, $picto, $moreatt='', $pictoisfullpath=false, $srconly=0, $notitle=0)
Show a picto called object_picto (generic function)
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_print_date($time, $format='', $tzoutput='auto', $outputlangs='', $encodetooutput=false)
Output date in a string format according to outputlangs (or langs if not defined).
dol_now($mode='auto')
Return date for now.
getDolGlobalInt($key, $default=0)
Return dolibarr global constant int value.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=false, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
getLocalTaxesFromRate($vatrate, $local, $buyer, $seller, $firstparamisid=0)
Get type and rate of localtaxes for a particular vat rate/country of a thirdparty.
get_default_npr(Societe $thirdparty_seller, Societe $thirdparty_buyer, $idprod=0, $idprodfournprice=0)
Function that returns whether VAT must be recoverable collected VAT (e.g.
dolGetStatus($statusLabel='', $statusLabelShort='', $html='', $statusType='status0', $displayMode=0, $url='', $params=array())
Output the badge of a status.
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
get_localtax($vatrate, $local, $thirdparty_buyer="", $thirdparty_seller="", $vatnpr=0)
Return localtax rate for a particular vat, when selling a product with vat $vatrate,...
dol_sanitizeFileName($str, $newstr='_', $unaccent=1)
Clean a string to use it as a file name.
isModEnabled($module)
Is Dolibarr module enabled.
get_default_tva(Societe $thirdparty_seller, Societe $thirdparty_buyer, $idprod=0, $idprodfournprice=0)
Function that return vat rate of a product line (according to seller, buyer and product vat rate) VAT...
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
getEntity($element, $shared=1, $currentobject=null)
Get list of entity id to use.
dol_escape_htmltag($stringtoescape, $keepb=0, $keepn=0, $noescapetags='', $escapeonlyhtmltags=0, $cleanalsojavascript=0)
Returns text escaped for inclusion in HTML alt or title or value tags, or into values of HTML input f...
getMarginInfos($pvht, $remise_percent, $tva_tx, $localtax1_tx, $localtax2_tx, $fk_pa, $paht)
Return an array with margins information of a line.
div float
Buy price without taxes.
Definition: style.css.php:921
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:86