40require_once DOL_DOCUMENT_ROOT.
'/core/class/commonobject.class.php';
41require_once DOL_DOCUMENT_ROOT.
"/expedition/class/expeditionligne.class.php";
42require_once DOL_DOCUMENT_ROOT.
'/core/class/commonincoterm.class.php';
44 require_once DOL_DOCUMENT_ROOT.
'/comm/propal/class/propal.class.php';
47 require_once DOL_DOCUMENT_ROOT.
'/commande/class/commande.class.php';
49require_once DOL_DOCUMENT_ROOT.
'/expedition/class/expeditionlinebatch.class.php';
50require_once DOL_DOCUMENT_ROOT.
'/core/class/commonsignedobject.class.php';
51require_once DOL_DOCUMENT_ROOT.
'/subtotals/class/commonsubtotal.class.php';
61 use CommonSignedObject;
67 public $element =
"shipping";
72 public $fk_element =
"fk_expedition";
77 public $table_element =
"expedition";
82 public $table_element_line =
"expeditiondet";
87 public $class_element_line =
'ExpeditionLigne';
92 public $picto =
'dolly';
98 public $fields = array();
104 public $user_author_id;
110 public $fk_user_author;
116 public $fk_user_valid;
133 public $ref_customer;
143 public $tracking_number;
148 public $tracking_url;
161 public $weight_units;
177 public $height_units;
194 public $livraison_id;
199 public $multicurrency_subprice;
229 public $date_delivery;
243 public $date_expedition;
249 public $date_shipping;
278 public $lines = array();
284 public $fk_multicurrency;
289 public $multicurrency_code;
293 public $multicurrency_tx;
297 public $multicurrency_total_ht;
301 public $multicurrency_total_tva;
305 public $multicurrency_total_ttc;
352 $this->ismultientitymanaged = 1;
353 $this->isextrafieldmanaged = 1;
356 $this->labelStatus = array();
357 $this->labelStatus[-1] =
'StatusSendingCanceled';
358 $this->labelStatus[0] =
'StatusSendingDraft';
359 $this->labelStatus[1] =
'StatusSendingValidated';
360 $this->labelStatus[2] =
'StatusSendingProcessed';
363 $this->labelStatusShort = array();
364 $this->labelStatusShort[-1] =
'StatusSendingCanceledShort';
365 $this->labelStatusShort[0] =
'StatusSendingDraftShort';
366 $this->labelStatusShort[1] =
'StatusSendingValidatedShort';
367 $this->labelStatusShort[2] =
'StatusSendingProcessedShort';
376 public function getNextNumRef($soc)
378 global $langs, $conf;
379 $langs->load(
"sendings");
388 $dirmodels = array_merge(array(
'/'), (array) $conf->modules_parts[
'models']);
390 foreach ($dirmodels as $reldir) {
394 $mybool = ((bool) @include_once $dir.$file) || $mybool;
402 $obj =
new $classname();
403 '@phan-var-force ModelNumRefExpedition $obj';
405 $numref = $obj->getNextValue($soc, $this);
410 dol_print_error($this->db, get_class($this).
"::getNextNumRef ".$obj->error);
414 print $langs->trans(
"Error").
" ".$langs->trans(
"Error_EXPEDITION_ADDON_NUMBER_NotDefined");
426 public function create($user, $notrigger = 0)
430 require_once DOL_DOCUMENT_ROOT.
'/product/stock/class/mouvementstock.class.php';
435 if (empty($this->fk_project)) {
436 $this->fk_project = 0;
438 if (empty($this->date_creation)) {
439 $this->date_creation = $now;
441 if (empty($this->date_shipping) && !empty($this->date_expedition)) {
442 $this->date_shipping = $this->date_expedition;
450 $sql =
"INSERT INTO ".MAIN_DB_PREFIX.
"expedition (";
453 $sql .=
", ref_customer";
455 $sql .=
", date_creation";
456 $sql .=
", fk_user_author";
457 $sql .=
", date_expedition";
458 $sql .=
", date_delivery";
460 $sql .=
", fk_projet";
461 $sql .=
", fk_address";
462 $sql .=
", fk_shipping_method";
463 $sql .=
", tracking_number";
468 $sql .=
", weight_units";
469 $sql .=
", size_units";
470 $sql .=
", note_private";
471 $sql .=
", note_public";
472 $sql .=
", model_pdf";
473 $sql .=
", fk_incoterms, location_incoterms";
474 $sql .=
", signed_status";
475 $sql .=
") VALUES (";
477 $sql .=
", ".((int) $this->entity);
478 $sql .=
", ".($this->ref_customer ?
"'".$this->db->escape($this->ref_customer).
"'" :
"null");
479 $sql .=
", ".($this->ref_ext ?
"'".$this->db->escape($this->ref_ext).
"'" :
"null");
480 $sql .=
", '".$this->db->idate($this->date_creation).
"'";
481 $sql .=
", ".((int) $user->id);
482 $sql .=
", ".($this->date_shipping > 0 ?
"'".$this->db->idate($this->date_shipping).
"'" :
"null");
483 $sql .=
", ".($this->date_delivery > 0 ?
"'".$this->db->idate($this->date_delivery).
"'" :
"null");
484 $sql .=
", ".($this->socid > 0 ? ((int) $this->socid) :
"null");
485 $sql .=
", ".($this->fk_project > 0 ? ((int) $this->fk_project) :
"null");
486 $sql .=
", ".($this->fk_delivery_address > 0 ? ((int) $this->fk_delivery_address) :
"null");
487 $sql .=
", ".($this->shipping_method_id > 0 ? ((int) $this->shipping_method_id) :
"null");
488 $sql .=
", '".$this->db->escape($this->tracking_number).
"'";
489 $sql .=
", ".(is_numeric($this->weight) ? (float) $this->weight :
'NULL');
490 $sql .=
", ".(is_numeric($this->sizeS) ? (float) $this->sizeS :
'NULL');
491 $sql .=
", ".(is_numeric($this->sizeW) ? (float) $this->sizeW :
'NULL');
492 $sql .=
", ".(is_numeric($this->sizeH) ? (float) $this->sizeH :
'NULL');
493 $sql .=
", ".($this->weight_units !=
'' ? (int) $this->weight_units :
'NULL');
494 $sql .=
", ".($this->size_units !=
'' ? (int) $this->size_units :
'NULL');
495 $sql .=
", ".(!empty($this->note_private) ?
"'".$this->db->escape($this->note_private).
"'" :
"null");
496 $sql .=
", ".(!empty($this->note_public) ?
"'".$this->db->escape($this->note_public).
"'" :
"null");
497 $sql .=
", ".(!empty($this->model_pdf) ?
"'".$this->db->escape($this->model_pdf).
"'" :
"null");
498 $sql .=
", ".((int) $this->fk_incoterms);
499 $sql .=
", '".$this->db->escape($this->location_incoterms).
"'";
500 $sql .=
", ".((int) $this->signed_status);
503 dol_syslog(get_class($this).
"::create", LOG_DEBUG);
504 $resql = $this->db->query($sql);
506 $this->
id = $this->db->last_insert_id(MAIN_DB_PREFIX.
"expedition");
509 $initialref =
'(PROV'.$this->id.
')';
510 if (!empty($this->
ref)) {
511 $initialref = $this->ref;
514 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition";
515 $sql .=
" SET ref = '".$this->db->escape($initialref).
"'";
516 $sql .=
" WHERE rowid = ".((int) $this->
id);
518 dol_syslog(get_class($this).
"::create", LOG_DEBUG);
519 if ($this->db->query($sql)) {
520 $this->
ref = $initialref;
522 $num = count($this->lines);
523 $kits_list = array();
525 for ($i = 0; $i < $num; $i++) {
527 $objectsrc->fetch($this->lines[$i]->origin_line_id);
528 if ($this->lines[$i]->product_type ==
"9" && $objectsrc->special_code == SUBTOTALS_SPECIAL_CODE) {
529 if ($this->
create_line($this->lines[$i]->entrepot_id, $this->lines[$i]->origin_line_id, $this->lines[$i]->qty, $this->lines[$i]->rang, $this->lines[$i]->array_options) <= 0) {
536 $line = $this->lines[$i];
537 if ($line->fk_product > 0) {
538 if (!isset($kits_list[$line->fk_product])) {
539 if (!is_object($line->product)) {
540 $line_product =
new Product($this->db);
541 $result = $line_product->fetch($line->fk_product,
'',
'',
'', 1, 1, 1);
546 $line_product = $line->product;
550 $line_product->get_sousproduits_arbo();
551 $prods_arbo = $line_product->get_arbo_each_prod($line->qty);
552 if (count($prods_arbo) > 0) {
553 $kits_list[$line->fk_product] = array(
554 'arbo' => $prods_arbo,
555 'total_qty' => $line->qty,
559 $kits_list[$line->fk_product][
'total_qty'] += $line->qty;
565 $kits_id_cached = array();
566 $sub_kits_id_cached = array();
567 for ($i = 0; $i < $num; $i++) {
568 $line = $this->lines[$i];
571 if (!isset($kits_id_cached[$line->fk_product])) {
572 if (!isset($line->detail_batch) || (isset($kits_list[$line->fk_product]) && !
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE'))) {
573 $qty = isset($kits_list[$line->fk_product]) ? $kits_list[$line->fk_product][
'total_qty'] : $line->qty;
574 $warehouse_id = (isset($kits_list[$line->fk_product]) && !
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) ? 0 : $line->entrepot_id;
575 $line_id = $this->
create_line($warehouse_id, $line->origin_line_id, $qty, $line->rang, $line->array_options, 0, $line->fk_product);
579 if (isset($kits_list[$line->fk_product])) {
580 $kits_id_cached[$line->fk_product] = $line_id;
588 $line_id = $kits_id_cached[$line->fk_product];
592 if (isset($kits_list[$line->fk_product]) && !
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) {
593 $prods_arbo = $kits_list[$line->fk_product][
'arbo'];
594 $total_qty = $kits_list[$line->fk_product][
'total_qty'];
597 $parent_line_id = $line_id;
599 $product_child_id = 0;
600 foreach ($prods_arbo as $index => $product_child_arr) {
615 $product_child_level = (int) $product_child_arr[
'level'];
616 $product_child_incdec = !empty($product_child_arr[
'incdec']);
619 if ($product_child_level != $level_last) {
620 $parent_line_id = $line_id;
621 $parent_product_id = $product_child_id;
622 if (isset($kits_id_cached[$parent_product_id])) {
623 $parent_line_id = $kits_id_cached[$parent_product_id];
625 $kits_id_cached[$parent_product_id] = $parent_line_id;
631 $next_level = $product_child_level;
632 $next_index = $index + 1;
633 if (isset($prods_arbo[$next_index])) {
634 $next_level = (int) $prods_arbo[$next_index][
'level'];
636 if ($next_level > $product_child_level) {
641 $product_child_id = (int) $product_child_arr[
'id'];
642 $product_child_qty = (float) $product_child_arr[
'nb_total'];
643 $warehouse_id = $line->entrepot_id;
644 if ($is_kit || !$product_child_incdec) {
645 if (!$product_child_incdec) {
646 $product_child_qty = 0;
652 if (!isset($sub_kits_id_cached[$product_child_id]) || $warehouse_id > 0) {
653 $line_id = $this->
create_line($warehouse_id, ($parent_line_id ? 0 : $line->origin_line_id), $product_child_qty, $line->rang, $line->array_options, $parent_line_id, $product_child_id);
661 if (empty($warehouse_id)) {
662 $sub_kits_id_cached[$product_child_id] = $line_id;
666 $level_last = $product_child_level;
672 if (!$error && $this->
id && $this->origin_id) {
679 if (!$error && $this->
id &&
getDolGlobalInt(
'SHIPPING_USE_ITS_OWN_CONTACTS') &&
getDolGlobalString(
'MAIN_PROPAGATE_CONTACTS_FROM_ORIGIN') && !empty($this->origin_type) && !empty($this->origin_id)) {
680 $originforcontact = $this->origin_type;
681 $originidforcontact = $this->origin_id;
683 $sqlcontact =
"SELECT ctc.code, ctc.source, ec.fk_socpeople FROM ".MAIN_DB_PREFIX.
"element_contact as ec, ".MAIN_DB_PREFIX.
"c_type_contact as ctc";
684 $sqlcontact .=
" WHERE element_id = ".((int) $originidforcontact).
" AND ec.fk_c_type_contact = ctc.rowid AND ctc.element = '".$this->db->escape($originforcontact).
"'";
686 $resqlcontact = $this->db->query($sqlcontact);
688 while ($objcontact = $this->db->fetch_object($resqlcontact)) {
689 $this->
add_contact($objcontact->fk_socpeople, $objcontact->code, $objcontact->source);
704 if (!$error && !$notrigger) {
706 $result = $this->call_trigger(
'SHIPPING_CREATE', $user);
716 foreach ($this->errors as $errmsg) {
717 dol_syslog(get_class($this).
"::create ".$errmsg, LOG_ERR);
718 $this->error .= ($this->error ?
', '.$errmsg : $errmsg);
720 $this->db->rollback();
725 $this->db->rollback();
730 $this->error = $this->db->lasterror().
" - sql=$sql";
731 $this->db->rollback();
736 $this->error = $this->db->error().
" - sql=$sql";
737 $this->db->rollback();
755 public function create_line($entrepot_id, $origin_line_id, $qty, $rang = 0, $array_options = [], $parent_line_id = 0, $product_id = 0)
761 $expeditionline->fk_expedition = $this->id;
762 $expeditionline->entrepot_id = $entrepot_id;
763 $expeditionline->fk_elementdet = $origin_line_id;
764 $expeditionline->element_type = $this->origin;
765 $expeditionline->fk_parent = $parent_line_id;
766 $expeditionline->fk_product = $product_id;
767 $expeditionline->qty = $qty;
768 $expeditionline->rang = $rang;
769 $expeditionline->array_options = $array_options;
771 if (!($expeditionline->fk_product > 0)) {
773 $order_line->fetch($expeditionline->fk_elementdet);
774 $expeditionline->fk_product = $order_line->fk_product;
777 if (($lineId = $expeditionline->insert($user)) < 0) {
778 $this->errors[] = $expeditionline->error;
796 $stockLocationQty = array();
798 $tab = $line_ext->detail_batch;
800 foreach ($tab as $detbatch) {
801 if (!empty($detbatch->fk_warehouse)) {
802 if (empty($stockLocationQty[$detbatch->fk_warehouse])) {
803 $stockLocationQty[$detbatch->fk_warehouse] = 0;
805 $stockLocationQty[$detbatch->fk_warehouse] += $detbatch->qty;
809 foreach ($stockLocationQty as $stockLocation => $qty) {
810 $line_id = $this->
create_line($stockLocation, $line_ext->origin_line_id, $qty, $line_ext->rang, $array_options);
815 foreach ($tab as $detbatch) {
816 if ($detbatch->fk_warehouse == $stockLocation) {
817 if (!($detbatch->create($line_id) > 0)) {
818 $this->errors = $detbatch->errors;
842 public function fetch($id, $ref =
'', $ref_ext =
'', $notused =
'')
847 if (empty($id) && empty($ref) && empty($ref_ext)) {
851 $sql =
"SELECT e.rowid, e.entity, e.ref, e.fk_soc as socid, e.date_creation, e.ref_customer, e.ref_ext, e.fk_user_author, e.fk_statut, e.signed_status, e.fk_projet as fk_project, e.billed";
852 $sql .=
", e.date_valid";
853 $sql .=
", e.weight, e.weight_units, e.size, e.size_units, e.width, e.height";
854 $sql .=
", e.date_expedition as date_expedition, e.model_pdf, e.fk_address, e.date_delivery";
855 $sql .=
", e.fk_shipping_method, e.tracking_number";
856 $sql .=
", e.note_private, e.note_public";
857 $sql .=
', e.fk_incoterms, e.location_incoterms';
858 $sql .=
', e.signed_status';
859 $sql .=
', i.libelle as label_incoterms';
860 $sql .=
', s.libelle as shipping_method';
861 $sql .=
", el.fk_source as origin_id, el.sourcetype as origin_type";
862 $sql .=
" FROM ".MAIN_DB_PREFIX.
"expedition as e";
863 $sql .=
" LEFT JOIN ".MAIN_DB_PREFIX.
"element_element as el ON el.fk_target = e.rowid AND el.targettype = '".$this->db->escape($this->element).
"'";
864 $sql .=
' LEFT JOIN '.MAIN_DB_PREFIX.
'c_incoterms as i ON e.fk_incoterms = i.rowid';
865 $sql .=
' LEFT JOIN '.MAIN_DB_PREFIX.
'c_shipment_mode as s ON e.fk_shipping_method = s.rowid';
866 $sql .=
" WHERE e.entity IN (".getEntity(
'expedition').
")";
868 $sql .=
" AND e.rowid = ".((int) $id);
871 $sql .=
" AND e.ref='".$this->db->escape($ref).
"'";
874 $sql .=
" AND e.ref_ext='".$this->db->escape($ref_ext).
"'";
877 dol_syslog(get_class($this).
"::fetch", LOG_DEBUG);
878 $result = $this->db->query($sql);
880 if ($this->db->num_rows($result)) {
881 $obj = $this->db->fetch_object($result);
883 $this->
id = (int) $obj->rowid;
884 $this->entity = $obj->entity;
885 $this->
ref = $obj->ref;
886 $this->socid = $obj->socid;
887 $this->ref_customer = $obj->ref_customer;
888 $this->ref_ext = $obj->ref_ext;
889 $this->
status = $obj->fk_statut;
890 $this->statut = $this->status;
891 $this->signed_status = $obj->signed_status;
892 $this->user_author_id = $obj->fk_user_author;
893 $this->fk_user_author = $obj->fk_user_author;
894 $this->user_creation_id = $obj->fk_user_author;
895 $this->date_creation = $this->db->jdate($obj->date_creation);
896 $this->date_valid = $this->db->jdate($obj->date_valid);
897 $this->
date = $this->db->jdate($obj->date_expedition);
898 $this->date_expedition = $this->db->jdate($obj->date_expedition);
899 $this->date_shipping = $this->db->jdate($obj->date_expedition);
900 $this->date_delivery = $this->db->jdate($obj->date_delivery);
901 $this->fk_delivery_address = $obj->fk_address;
902 $this->model_pdf = $obj->model_pdf;
903 $this->shipping_method_id = $obj->fk_shipping_method;
904 $this->shipping_method = $obj->shipping_method;
905 $this->tracking_number = $obj->tracking_number;
906 $this->origin = ($obj->origin_type ? $obj->origin_type :
'commande');
907 $this->origin_type = ($obj->origin_type ? $obj->origin_type :
'commande');
908 $this->origin_id = $obj->origin_id;
909 $this->billed = $obj->billed;
910 $this->fk_project = $obj->fk_project;
911 $this->signed_status = $obj->signed_status;
912 $this->trueWeight = $obj->weight;
913 $this->weight_units = $obj->weight_units;
915 $this->trueWidth = $obj->width;
916 $this->width_units = $obj->size_units;
917 $this->trueHeight = $obj->height;
918 $this->height_units = $obj->size_units;
919 $this->trueDepth = $obj->size;
920 $this->depth_units = $obj->size_units;
922 $this->note_public = $obj->note_public;
923 $this->note_private = $obj->note_private;
926 $this->trueSize = $obj->size.
"x".$obj->width.
"x".$obj->height;
927 $this->size_units = $obj->size_units;
930 $this->fk_incoterms = $obj->fk_incoterms;
931 $this->location_incoterms = $obj->location_incoterms;
932 $this->label_incoterms = $obj->label_incoterms;
934 $this->db->free($result);
947 if (!empty($this->multicurrency_code)) {
948 $this->multicurrency_code = $this->thirdparty->multicurrency_code;
950 if (
getDolGlobalString(
'MULTICURRENCY_USE_ORIGIN_TX') && !empty($this->thirdparty->multicurrency_tx)) {
951 $this->multicurrency_tx = $this->thirdparty->multicurrency_tx;
958 if (empty($obj->origin_id)) {
969 dol_syslog(get_class($this).
'::Fetch no expedition found', LOG_ERR);
970 $this->error =
'Shipment with id '.$id.
' not found';
974 $this->error = $this->db->error();
986 public function valid($user, $notrigger = 0)
990 require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
996 dol_syslog(get_class($this).
"::valid not in draft status", LOG_WARNING);
1000 if (!isset($this->socid)) {
1001 dol_syslog(get_class($this).
"::can't valid socid not set", LOG_WARNING);
1005 if (!((!
getDolGlobalString(
'MAIN_USE_ADVANCED_PERMS') && $user->hasRight(
'expedition',
'creer'))
1006 || (
getDolGlobalString(
'MAIN_USE_ADVANCED_PERMS') && $user->hasRight(
'expedition',
'shipping_advance',
'validate')))) {
1007 $this->error =
'Permission denied';
1008 dol_syslog(get_class($this).
"::valid ".$this->error, LOG_ERR);
1017 $soc =
new Societe($this->db);
1018 $soc->fetch($this->socid);
1021 $result = $soc->setAsCustomer();
1024 if (preg_match(
'/^[\(]?PROV/i', $this->
ref) || empty($this->
ref)) {
1025 $numref = $this->getNextNumRef($soc);
1026 } elseif (!empty($this->
ref)) {
1027 $numref = (string) $this->
ref;
1029 $numref =
"EXP".$this->id;
1036 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition SET";
1037 $sql .=
" ref = '".$this->db->escape($numref).
"'";
1038 $sql .=
", fk_statut = 1";
1039 $sql .=
", date_valid = '".$this->db->idate($now).
"'";
1040 $sql .=
", fk_user_valid = ".((int) $user->id);
1041 $sql .=
" WHERE rowid = ".((int) $this->
id);
1043 dol_syslog(get_class($this).
"::valid update expedition", LOG_DEBUG);
1044 $resql = $this->db->query($sql);
1046 $this->error = $this->db->lasterror();
1059 $triggerKey =
'SHIPPING_';
1060 if ($this->origin ==
'commande') {
1061 $triggerKey .=
'ORDER_SHIPMENTONPROCESS';
1063 $triggerKey .= strtoupper($this->origin).
'_SHIPMENTONPROCESS';
1072 if (!$error && !$notrigger) {
1074 $result = $this->call_trigger(
'SHIPPING_VALIDATE', $user);
1082 $this->oldref = $this->ref;
1085 if (preg_match(
'/^[\(]?PROV/i', $this->
ref)) {
1087 $sql =
'UPDATE '.MAIN_DB_PREFIX.
"ecm_files set filename = CONCAT('".$this->db->escape($this->newref).
"', SUBSTR(filename, ".(strlen($this->
ref) + 1).
")), filepath = 'expedition/sending/".$this->db->escape($this->newref).
"'";
1088 $sql .=
" WHERE filename LIKE '".$this->db->escape($this->
ref).
"%' AND filepath = 'expedition/sending/".$this->db->escape($this->
ref).
"' and entity = ".((int) $conf->entity);
1089 $resql = $this->db->query($sql);
1092 $this->error = $this->db->lasterror();
1094 $sql =
'UPDATE '.MAIN_DB_PREFIX.
"ecm_files set filepath = 'expedition/sending/".$this->db->escape($this->newref).
"'";
1095 $sql .=
" WHERE filepath = 'expedition/sending/".$this->db->escape($this->
ref).
"' and entity = ".((int) $conf->entity);
1096 $resql = $this->db->query($sql);
1099 $this->error = $this->db->lasterror();
1105 $dirsource = $conf->expedition->dir_output.
'/sending/'.$oldref;
1106 $dirdest = $conf->expedition->dir_output.
'/sending/'.$newref;
1107 if (!$error && file_exists($dirsource)) {
1108 dol_syslog(get_class($this).
"::valid rename dir ".$dirsource.
" into ".$dirdest);
1110 if (@rename($dirsource, $dirdest)) {
1113 $listoffiles =
dol_dir_list($conf->expedition->dir_output.
'/sending/'.$newref,
'files', 1,
'^'.preg_quote($oldref,
'/'));
1114 foreach ($listoffiles as $fileentry) {
1115 $dirsource = $fileentry[
'name'];
1116 $dirdest = preg_replace(
'/^'.preg_quote($oldref,
'/').
'/', $newref, $dirsource);
1117 $dirsource = $fileentry[
'path'].
'/'.$dirsource;
1118 $dirdest = $fileentry[
'path'].
'/'.$dirdest;
1119 @rename($dirsource, $dirdest);
1128 $this->
ref = $numref;
1134 $this->db->commit();
1137 $this->db->rollback();
1154 if ($this->
status == self::STATUS_VALIDATED || $this->
status == self::STATUS_CLOSED) {
1156 include_once DOL_DOCUMENT_ROOT.
'/delivery/class/delivery.class.php';
1157 $delivery =
new Delivery($this->db);
1158 $result = $delivery->create_from_sending($user, $this->
id);
1183 public function addline($entrepot_id, $id, $qty, $array_options = [], $fk_product = 0, $fk_parent = 0)
1187 $num = count($this->lines);
1190 $line->entrepot_id = $entrepot_id;
1191 $line->origin_line_id = $id;
1192 $line->fk_elementdet = $id;
1193 $line->element_type =
'order';
1194 $line->fk_parent = $fk_parent;
1195 $line->fk_product = $fk_product;
1199 $orderline->fetch($id);
1202 $line->rang = $orderline->rang;
1203 $line->product_type = $orderline->product_type;
1204 if (!($line->fk_product > 0)) {
1205 $line->fk_product = $orderline->fk_product;
1208 if (
isModEnabled(
'stock') && !empty($orderline->fk_product)) {
1209 $product =
new Product($this->db);
1210 $product->fetch($orderline->fk_product);
1213 $langs->load(
"errors");
1214 $this->error = $langs->trans(
"ErrorWarehouseRequiredIntoShipmentLine");
1219 $productChildrenNb = 0;
1221 $productChildrenNb = $product->hasFatherOrChild(1);
1223 if ($productChildrenNb > 0) {
1224 $product_stock =
null;
1225 $product->loadStockForVirtualProduct(
'warehouseopen', $line->qty);
1226 if ($entrepot_id > 0) {
1227 if (isset($product->stock_warehouse[$entrepot_id])) {
1228 $product_stock = $product->stock_warehouse[$entrepot_id]->real;
1231 foreach ($product->stock_warehouse as $componentStockWarehouse) {
1232 if ($product_stock ===
null) {
1233 $product_stock = $componentStockWarehouse->real;
1235 $product_stock = min($product_stock, $componentStockWarehouse->real);
1239 if ($product_stock ===
null) {
1244 if ($entrepot_id > 0) {
1245 $product->load_stock(
'warehouseopen');
1246 $product_stock = $product->stock_warehouse[$entrepot_id]->real;
1248 $product_stock = $product->stock_reel;
1252 $product_type = $product->type;
1254 $isavirtualproduct = ($productChildrenNb > 0);
1259 || ($isavirtualproduct && !
getDolGlobalInt(
'STOCK_EXCLUDE_VIRTUAL_PRODUCTS'))
1262 if ($product->stockable_product == Product::ENABLED_STOCK && $product_stock < $qty) {
1263 $langs->load(
"errors");
1264 $this->error = $langs->trans(
'ErrorStockIsNotEnoughToAddProductOnShipment', $product->ref);
1265 $this->errorhidden =
'ErrorStockIsNotEnoughToAddProductOnShipment';
1276 if (
isModEnabled(
'productbatch') && !empty($line->fk_product) && !empty($orderline->product_tobatch)) {
1277 $this->error =
'ADDLINE_WAS_CALLED_INSTEAD_OF_ADDLINEBATCH '.$orderline->id.
' '.$line->fk_product;
1282 if (!
getDolGlobalString(
'MAIN_EXTRAFIELDS_DISABLED') && is_array($array_options) && count($array_options) > 0) {
1283 $line->array_options = $array_options;
1286 $this->lines[$num] = $line;
1304 public function addlinefree($qty, $element_type, $fk_product, $fk_unit, $rang, $description, $fk_parent, $array_options = [])
1306 global
$mysoc, $langs, $user;
1308 if ($this->
status == self::STATUS_DRAFT) {
1319 if ($ranktouse == -1) {
1320 $rangmax = $this->
line_max($fk_parent);
1321 $ranktouse = $rangmax + 1;
1326 $this->line->fk_expedition = $this->id;
1327 $this->line->element_type = $element_type;
1328 $this->line->fk_product = $fk_product;
1329 $this->line->description = $description;
1330 $this->line->desc = $description;
1331 $this->line->fk_parent = $fk_parent;
1332 $this->line->qty = (float) $qty;
1333 $this->line->fk_unit = $fk_unit;
1334 $this->line->rang = $ranktouse;
1336 if (is_array($array_options) && count($array_options) > 0) {
1337 $this->line->array_options = $array_options;
1340 $result = $this->line->insert($user);
1342 if (!isset($this->context[
'createfromclone'])) {
1343 if (!empty($fk_parent)) {
1345 } elseif ($ranktouse > 0 && $ranktouse <= count($this->lines)) {
1346 $linecount = count($this->lines);
1347 for ($ii = $ranktouse; $ii <= $linecount; $ii++) {
1351 $this->lines[] = $this->line;
1354 $this->db->commit();
1355 return $this->line->id;
1357 $this->error = $this->line->error;
1358 dol_syslog(get_class($this).
"::addlinefree error=".$this->error, LOG_ERR);
1359 $this->db->rollback();
1363 dol_syslog(get_class($this).
"::addlinefree status of shipment must be Draft to allow use of ->addlinefree()", LOG_ERR);
1383 public function updatelinefree($rowid, $qty, $element_type, $fk_product, $fk_unit, $rang, $description, $fk_parent, $notrigger, $array_options = array())
1385 global
$mysoc, $langs, $user;
1387 if ($this->
status == self::STATUS_DRAFT) {
1397 $qty = (float) $qty;
1398 $description = trim($description);
1403 $line->fetch($rowid);
1404 $line->fetch_optionals();
1406 if (!empty($line->fk_product)) {
1407 $product =
new Product($this->db);
1408 $result = $product->fetch($line->fk_product);
1409 $product_type = $product->type;
1412 $staticline = clone $line;
1414 $line->oldline = $staticline;
1415 $this->line = $line;
1416 $this->line->context = $this->context;
1417 $this->line->rang = $rang;
1418 $this->line->fk_expedition = $this->id;
1419 $this->line->element_type = $element_type;
1420 $this->line->fk_product = $fk_product;
1421 $this->line->qty = $qty;
1422 $this->line->fk_unit = $fk_unit;
1423 $this->line->fk_parent = $fk_parent;
1424 $this->line->description = $description;
1426 if (is_array($array_options) && count($array_options) > 0) {
1428 foreach ($array_options as $key => $value) {
1429 $this->line->array_options[$key] = $array_options[$key];
1433 $result = $this->line->update($user, $notrigger);
1436 if (!empty($fk_parent)) {
1440 $this->db->commit();
1443 $this->error = $this->line->error;
1445 $this->db->rollback();
1449 $this->error = get_class($this).
"::updatelinefree Shipment status makes operation forbidden";
1450 $this->errors = array(
'ShipmentStatusMakeOperationForbidden');
1464 public function addline_batch($dbatch, $array_options = [], $origin_line =
null)
1469 $num = count($this->lines);
1471 if ($dbatch[
'qty'] > 0 || ($dbatch[
'qty'] == 0 &&
getDolGlobalString(
'SHIPMENT_GETS_ALL_ORDER_PRODUCTS'))) {
1474 foreach ($dbatch[
'detail'] as $key => $value) {
1475 if ($value[
'q'] > 0 || ($value[
'q'] == 0 &&
getDolGlobalString(
'SHIPMENT_GETS_ALL_ORDER_PRODUCTS'))) {
1481 $ret = $linebatch->fetchFromStock($value[
'id_batch']);
1486 $linebatch->qty = $value[
'q'];
1487 if ($linebatch->qty == 0 &&
getDolGlobalString(
'SHIPMENT_GETS_ALL_ORDER_PRODUCTS')) {
1488 $linebatch->batch =
null;
1490 $tab[] = $linebatch;
1493 require_once DOL_DOCUMENT_ROOT.
'/product/class/productbatch.class.php';
1495 $prod_batch->fetch($value[
'id_batch']);
1497 if ($prod_batch->qty < $linebatch->qty) {
1498 $langs->load(
"errors");
1499 $this->errors[] = $langs->trans(
'ErrorStockIsNotEnoughToAddProductOnShipment', $prod_batch->fk_product);
1500 dol_syslog(get_class($this).
"::addline_batch error=Product ".$prod_batch->batch.
": ".$this->errorsToString(), LOG_ERR);
1506 if (is_object($linebatch)) {
1507 $line->entrepot_id = $linebatch->fk_warehouse;
1509 $line->origin_line_id = $dbatch[
'ix_l'];
1510 $line->fk_elementdet = $dbatch[
'ix_l'];
1511 $line->qty = $dbatch[
'qty'];
1512 $line->detail_batch = $tab;
1513 if (!($line->rang > 0)) {
1514 $line->rang = $origin_line->rang;
1516 if (!($line->fk_product > 0)) {
1517 $line->fk_product = $origin_line->fk_product;
1521 if (!
getDolGlobalString(
'MAIN_EXTRAFIELDS_DISABLED') && is_array($array_options) && count($array_options) > 0) {
1522 $line->array_options = $array_options;
1526 $this->lines[$num] = $line;
1539 public function update($user =
null, $notrigger = 0)
1546 if (isset($this->
ref)) {
1547 $this->
ref = trim($this->
ref);
1549 if (isset($this->entity)) {
1550 $this->entity = (int) $this->entity;
1552 if (isset($this->ref_customer)) {
1553 $this->ref_customer = trim($this->ref_customer);
1555 if (isset($this->socid)) {
1556 $this->socid = (int) $this->socid;
1558 if (isset($this->fk_user_author)) {
1559 $this->fk_user_author = (int) $this->fk_user_author;
1561 if (isset($this->fk_user_valid)) {
1562 $this->fk_user_valid = (int) $this->fk_user_valid;
1564 if (isset($this->fk_delivery_address)) {
1565 $this->fk_delivery_address = (int) $this->fk_delivery_address;
1567 if (isset($this->shipping_method_id)) {
1568 $this->shipping_method_id = (int) $this->shipping_method_id;
1570 if (isset($this->tracking_number)) {
1571 $this->tracking_number = trim($this->tracking_number);
1573 if (isset($this->statut)) {
1574 $this->statut = (int) $this->statut;
1576 if (isset($this->
status)) {
1579 if (isset($this->trueDepth)) {
1580 $this->trueDepth = trim($this->trueDepth);
1582 if (isset($this->trueWidth)) {
1583 $this->trueWidth = trim($this->trueWidth);
1585 if (isset($this->trueHeight)) {
1586 $this->trueHeight = trim($this->trueHeight);
1588 if (isset($this->size_units)) {
1589 $this->size_units = trim($this->size_units);
1591 if (isset($this->weight_units)) {
1592 $this->weight_units = (int) $this->weight_units;
1594 if (isset($this->trueWeight)) {
1595 $this->weight = trim((
string) $this->trueWeight);
1597 if (isset($this->note_private)) {
1598 $this->note_private = trim($this->note_private);
1600 if (isset($this->note_public)) {
1601 $this->note_public = trim($this->note_public);
1603 if (isset($this->model_pdf)) {
1604 $this->model_pdf = trim($this->model_pdf);
1606 if (!empty($this->date_expedition)) {
1607 $this->date_shipping = $this->date_expedition;
1614 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition SET";
1615 $sql .=
" ref = ".(isset($this->
ref) ?
"'".$this->db->escape($this->
ref).
"'" :
"null").
",";
1616 $sql .=
" ref_ext = ".(isset($this->ref_ext) ?
"'".$this->db->escape($this->ref_ext).
"'" :
"null").
",";
1617 $sql .=
" ref_customer = ".(isset($this->ref_customer) ?
"'".$this->db->escape($this->ref_customer).
"'" :
"null").
",";
1618 $sql .=
" fk_soc = ".(isset($this->socid) ? ((int) $this->socid) :
"null").
",";
1619 $sql .=
" date_creation = ".(dol_strlen($this->date_creation) != 0 ?
"'".$this->db->idate($this->date_creation).
"'" :
'null').
",";
1620 $sql .=
" fk_user_author = ".(isset($this->fk_user_author) ? ((int) $this->fk_user_author) :
"null").
",";
1621 $sql .=
" date_valid = ".(dol_strlen($this->date_valid) != 0 ?
"'".$this->db->idate($this->date_valid).
"'" :
'null').
",";
1622 $sql .=
" fk_user_valid = ".(isset($this->fk_user_valid) ? ((int) $this->fk_user_valid) :
"null").
",";
1623 $sql .=
" date_expedition = ".(dol_strlen($this->date_shipping) != 0 ?
"'".$this->db->idate($this->date_shipping).
"'" :
'null').
",";
1624 $sql .=
" date_delivery = ".(dol_strlen($this->date_delivery) != 0 ?
"'".$this->db->idate($this->date_delivery).
"'" :
'null').
",";
1625 $sql .=
" fk_address = ".(isset($this->fk_delivery_address) ? ((int) $this->fk_delivery_address) :
"null").
",";
1626 $sql .=
" fk_shipping_method = ".((isset($this->shipping_method_id) && $this->shipping_method_id > 0) ? ((
int) $this->shipping_method_id) :
"null").
",";
1627 $sql .=
" tracking_number = ".(isset($this->tracking_number) ?
"'".$this->db->escape($this->tracking_number).
"'" :
"null").
",";
1628 $sql .=
" fk_statut = ".(isset($this->
status) ? ((int) $this->
status) :
"null").
",";
1629 $sql .=
" fk_projet = ".(isset($this->fk_project) ? ((int) $this->fk_project) :
"null").
",";
1630 $sql .=
" height = ".(($this->trueHeight !=
'') ? ((
float) $this->trueHeight) :
"null").
",";
1631 $sql .=
" width = ".(($this->trueWidth !=
'') ? ((
float) $this->trueWidth) :
"null").
",";
1632 $sql .=
" size_units = ".(isset($this->size_units) ? ((int) $this->size_units) :
"null").
",";
1633 $sql .=
" size = ".(($this->trueDepth !=
'') ? ((
float) $this->trueDepth) :
"null").
",";
1634 $sql .=
" weight_units = ".(isset($this->weight_units) ? ((int) $this->weight_units) :
"null").
",";
1635 $sql .=
" weight = ".(($this->trueWeight !=
'') ? ((
float) $this->trueWeight) :
"null").
",";
1636 $sql .=
" note_private = ".(isset($this->note_private) ?
"'".$this->db->escape($this->note_private).
"'" :
"null").
",";
1637 $sql .=
" note_public = ".(isset($this->note_public) ?
"'".$this->db->escape($this->note_public).
"'" :
"null").
",";
1638 $sql .=
" model_pdf = ".(isset($this->model_pdf) ?
"'".$this->db->escape($this->model_pdf).
"'" :
"null").
",";
1639 $sql .=
" entity = ".((int) $conf->entity);
1640 $sql .=
" WHERE rowid = ".((int) $this->
id);
1644 dol_syslog(get_class($this).
"::update", LOG_DEBUG);
1645 $resql = $this->db->query($sql);
1648 $this->errors[] =
"Error ".$this->db->lasterror();
1659 if (!$error && !$notrigger) {
1661 $result = $this->call_trigger(
'SHIPPING_MODIFY', $user);
1670 foreach ($this->errors as $errmsg) {
1671 dol_syslog(get_class($this).
"::update ".$errmsg, LOG_ERR);
1672 $this->error .= ($this->error ?
', '.$errmsg : $errmsg);
1674 $this->db->rollback();
1677 $this->db->commit();
1691 public function cancel($user, $notrigger = 0, $also_update_stock =
false)
1693 global $conf, $langs, $user;
1695 require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
1704 if (isset($this->linkedObjectsIds[
'delivery']) && count($this->linkedObjectsIds[
'delivery']) > 0) {
1705 $this->error =
'ErrorThereIsSomeDeliveries';
1709 if (!$error && !$notrigger) {
1711 $result = $this->call_trigger(
'SHIPPING_CANCEL', $user);
1721 (
getDolGlobalString(
'STOCK_CALCULATE_ON_SHIPMENT_CLOSE') && $this->
status == self::STATUS_CLOSED && $also_update_stock));
1723 require_once DOL_DOCUMENT_ROOT.
"/product/stock/class/mouvementstock.class.php";
1725 $langs->load(
"agenda");
1729 $sql .=
" ed.fk_product";
1730 $sql .=
", ed.qty, ed.fk_entrepot, ed.rowid as expeditiondet_id";
1731 $sql .=
", SUM(".$this->db->ifsql(
"pa.rowid IS NOT NULL",
"1",
"0").
") as iskit";
1732 $sql .=
", ".$this->db->ifsql(
"pai.incdec IS NULL",
"1",
"pai.incdec").
" as incdec";
1733 $sql .=
" FROM ".$this->db->prefix().
"expeditiondet as ed";
1734 $sql .=
" LEFT JOIN ".$this->db->prefix().
"product_association as pa ON pa.fk_product_pere = ed.fk_product";
1735 $sql .=
" LEFT JOIN ".$this->db->prefix().
"expeditiondet as edp ON edp.rowid = ed.fk_parent";
1736 $sql .=
" LEFT JOIN ".$this->db->prefix().
"product_association as pai ON pai.fk_product_pere = edp.fk_product AND pai.fk_product_fils = ed.fk_product";
1737 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
1738 $sql .=
" GROUP BY ed.fk_product, ed.qty, ed.fk_entrepot, ed.rowid, pai.incdec";
1739 $sql .= $this->db->order(
"ed.rowid",
"DESC");
1741 dol_syslog(get_class($this).
"::delete select details", LOG_DEBUG);
1742 $resql = $this->db->query($sql);
1744 $cpt = $this->db->num_rows($resql);
1748 for ($i = 0; $i < $cpt; $i++) {
1749 dol_syslog(get_class($this).
"::delete movement index ".$i);
1750 $obj = $this->db->fetch_object($resql);
1751 $line_id = (int) $obj->expeditiondet_id;
1753 if ($can_update_stock && (empty($obj->iskit) ||
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) && !empty($obj->incdec)) {
1756 $mouvS->origin =
'';
1760 $lotArray = $shipmentlinebatch->fetchAll($obj->expeditiondet_id);
1761 if (!is_array($lotArray)) {
1763 $this->errors[] =
"Error ".$this->db->lasterror();
1767 if (empty($lotArray)) {
1771 $result = $mouvS->reception($user, $obj->fk_product, $obj->fk_entrepot, $obj->qty, 0, $langs->trans(
"ShipmentCanceledInDolibarr", $this->ref),
'',
'',
'',
'', 0,
'', 0, (empty($obj->iskit) ? 1 : 0));
1774 $this->errors = array_merge($this->errors, $mouvS->errors);
1780 foreach ($lotArray as $lot) {
1781 $result = $mouvS->reception($user, $obj->fk_product, $obj->fk_entrepot, $lot->qty, 0, $langs->trans(
"ShipmentCanceledInDolibarr", $this->ref), $lot->eatby, $lot->sellby, (
string) $lot->batch,
'', 0,
'', 0, (empty($obj->iskit) ? 1 : 0));
1784 $this->errors = array_merge($this->errors, $mouvS->errors);
1797 $res = $shipment_line->fetch($line_id);
1799 $result = $shipment_line->delete($user);
1802 $this->errors[] =
"Error ".$shipment_line->errorsToString();
1806 $this->errors[] =
"Error ".$shipment_line->errorsToString();
1816 $this->errors[] =
"Error ".$this->db->lasterror();
1829 $sql =
"SELECT rowid FROM ".$this->db->prefix().
"expedition";
1830 $sql .=
" WHERE rowid = ".((int) $this->
id);
1832 if ($this->db->query($sql)) {
1833 if (!empty($this->origin) && $this->origin_id > 0) {
1836 '@phan-var-force Facture|Commande $origin_object';
1847 $this->db->commit();
1851 if (!empty($conf->expedition->dir_output)) {
1852 $dir = $conf->expedition->dir_output.
'/sending/'.$ref;
1853 $file = $dir.
'/'.$ref.
'.pdf';
1854 if (file_exists($file)) {
1859 if (file_exists($dir)) {
1861 $this->error = $langs->trans(
"ErrorCanNotDeleteDir", $dir);
1869 $this->error = $this->db->lasterror().
" - sql=$sql";
1870 $this->db->rollback();
1874 $this->db->rollback();
1878 $this->db->rollback();
1892 public function delete($user =
null, $notrigger = 0, $also_update_stock =
false)
1894 global $conf, $langs;
1900 require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
1909 if (isset($this->linkedObjectsIds[
'delivery']) && count($this->linkedObjectsIds[
'delivery']) > 0) {
1910 $this->error =
'ErrorThereIsSomeDeliveries';
1914 if (!$error && !$notrigger) {
1916 $result = $this->call_trigger(
'SHIPPING_DELETE', $user);
1926 (
getDolGlobalString(
'STOCK_CALCULATE_ON_SHIPMENT_CLOSE') && $this->
status == self::STATUS_CLOSED && $also_update_stock));
1928 require_once DOL_DOCUMENT_ROOT.
"/product/stock/class/mouvementstock.class.php";
1930 $langs->load(
"agenda");
1937 $sql .=
" ed.fk_product";
1938 $sql .=
", ed.qty, ed.fk_entrepot, ed.rowid as expeditiondet_id";
1939 $sql .=
", SUM(".$this->db->ifsql(
"pa.rowid IS NOT NULL",
"1",
"0").
") as iskit";
1940 $sql .=
", ".$this->db->ifsql(
"pai.incdec IS NULL",
"1",
"pai.incdec").
" as incdec";
1941 $sql .=
" FROM ".$this->db->prefix().
"expeditiondet as ed";
1942 $sql .=
" LEFT JOIN ".$this->db->prefix().
"product_association as pa ON pa.fk_product_pere = ed.fk_product";
1943 $sql .=
" LEFT JOIN ".$this->db->prefix().
"expeditiondet as edp ON edp.rowid = ed.fk_parent";
1944 $sql .=
" LEFT JOIN ".$this->db->prefix().
"product_association as pai ON pai.fk_product_pere = edp.fk_product AND pai.fk_product_fils = ed.fk_product";
1945 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
1946 $sql .=
" GROUP BY ed.fk_product, ed.qty, ed.fk_entrepot, ed.rowid, pai.incdec";
1947 $sql .= $this->db->order(
"ed.rowid",
"DESC");
1949 dol_syslog(get_class($this).
"::delete select details", LOG_DEBUG);
1950 $resql = $this->db->query($sql);
1952 $cpt = $this->db->num_rows($resql);
1953 for ($i = 0; $i < $cpt; $i++) {
1954 dol_syslog(get_class($this).
"::delete movement index ".$i);
1955 $obj = $this->db->fetch_object($resql);
1956 $line_id = (int) $obj->expeditiondet_id;
1958 if ($can_update_stock && (empty($obj->iskit) ||
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) && !empty($obj->incdec)) {
1961 $mouvS->origin =
'';
1963 $lotArray = $shipmentlinebatch->fetchAll($line_id);
1964 if (!is_array($lotArray)) {
1966 $this->errors[] =
"Error ".$this->db->lasterror();
1968 if (empty($lotArray)) {
1972 $result = $mouvS->reception($user, $obj->fk_product, $obj->fk_entrepot, $obj->qty, 0, $langs->trans(
"ShipmentDeletedInDolibarr", $this->ref),
'',
'',
'',
'', 0,
'', 0, (empty($obj->iskit) ? 1 : 0));
1975 $this->errors = array_merge($this->errors, $mouvS->errors);
1981 foreach ($lotArray as $lot) {
1982 $result = $mouvS->reception($user, $obj->fk_product, $obj->fk_entrepot, $lot->qty, 0, $langs->trans(
"ShipmentDeletedInDolibarr", $this->ref), $lot->eatby, $lot->sellby, (
string) $lot->batch,
'', 0,
'', 0, (empty($obj->iskit) ? 1 : 0));
1985 $this->errors = array_merge($this->errors, $mouvS->errors);
1998 $res = $shipment_line->fetch($line_id);
2000 $result = $shipment_line->delete($user);
2003 $this->errors[] =
"Error ".$shipment_line->errorsToString();
2007 $this->errors[] =
"Error ".$shipment_line->errorsToString();
2017 $this->errors[] =
"Error ".$this->db->lasterror();
2042 $sql =
"DELETE FROM ".$this->db->prefix().
"expedition";
2043 $sql .=
" WHERE rowid = ".((int) $this->
id);
2045 if ($this->db->query($sql)) {
2046 if (!empty($this->origin) && $this->origin_id > 0) {
2049 '@phan-var-force Facture|Commande $origin_object';
2059 $this->db->commit();
2067 if (!empty($conf->expedition->dir_output)) {
2068 $dir = $conf->expedition->dir_output .
'/sending/' . $ref;
2069 $file = $dir .
'/' . $ref .
'.pdf';
2070 if (file_exists($file)) {
2075 if (file_exists($dir)) {
2077 $this->error = $langs->trans(
"ErrorCanNotDeleteDir", $dir);
2085 $this->error = $this->db->lasterror().
" - sql=$sql";
2086 $this->db->rollback();
2090 $this->db->rollback();
2094 $this->db->rollback();
2110 $this->lines = array();
2115 $sql =
"SELECT cd.rowid, cd.fk_product, cd.label as custom_label, cd.description, cd.qty as qty_asked, cd.product_type, cd.fk_unit";
2116 $sql .=
", cd.total_ht, cd.total_localtax1, cd.total_localtax2, cd.total_ttc, cd.total_tva";
2117 $sql .=
", cd.fk_remise_except, cd.fk_product_fournisseur_price as fk_fournprice";
2118 $sql .=
", cd.vat_src_code, cd.tva_tx, cd.localtax1_tx, cd.localtax2_tx, cd.localtax1_type, cd.localtax2_type, cd.info_bits, cd.price, cd.subprice, cd.remise_percent,cd.buy_price_ht as pa_ht";
2119 $sql .=
", cd.fk_multicurrency, cd.multicurrency_code, cd.multicurrency_subprice, cd.multicurrency_total_ht, cd.multicurrency_total_tva, cd.multicurrency_total_ttc, cd.rang, cd.date_start, cd.date_end, cd.special_code";
2120 $sql .=
", ed.rowid as line_id, ed.qty as qty_shipped, ed.fk_element, ed.fk_elementdet, ed.element_type, ed.fk_entrepot, ed.extraparams";
2121 $sql .=
", p.ref as product_ref, p.label as product_label, p.fk_product_type, p.barcode as product_barcode";
2122 $sql .=
", p.weight, p.weight_units, p.length, p.length_units, p.width, p.width_units, p.height, p.height_units";
2123 $sql .=
", p.surface, p.surface_units, p.volume, p.volume_units, p.tosell as product_tosell, p.tobuy as product_tobuy";
2124 $sql .=
", p.tobatch as product_tobatch, p.stockable_product";
2125 $sql .=
" FROM ".MAIN_DB_PREFIX.
"expeditiondet as ed, ".MAIN_DB_PREFIX.
"commandedet as cd";
2126 $sql .=
" LEFT JOIN ".MAIN_DB_PREFIX.
"product as p ON p.rowid = cd.fk_product";
2127 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
2128 $sql .=
" AND ed.fk_elementdet = cd.rowid";
2129 $sql .=
" ORDER BY cd.rang, ed.fk_elementdet";
2131 dol_syslog(get_class($this).
"::fetch_lines", LOG_DEBUG);
2132 $resql = $this->db->query($sql);
2134 include_once DOL_DOCUMENT_ROOT.
'/core/lib/price.lib.php';
2136 $num = $this->db->num_rows($resql);
2142 $this->total_ht = 0;
2143 $this->total_tva = 0;
2144 $this->total_ttc = 0;
2145 $this->total_localtax1 = 0;
2146 $this->total_localtax2 = 0;
2148 $this->multicurrency_total_ht = 0;
2149 $this->multicurrency_total_tva = 0;
2150 $this->multicurrency_total_ttc = 0;
2156 $obj = $this->db->fetch_object($resql);
2158 if ($originline > 0 && $originline == $obj->fk_elementdet) {
2160 $line->entrepot_id = 0;
2161 $line->qty_shipped += $obj->qty_shipped;
2164 $line->entrepot_id = $obj->fk_entrepot;
2165 $line->qty_shipped = $obj->qty_shipped;
2169 $detail_entrepot->entrepot_id = $obj->fk_entrepot;
2170 $detail_entrepot->qty_shipped = $obj->qty_shipped;
2171 $detail_entrepot->line_id = $obj->line_id;
2172 $line->details_entrepot[] = $detail_entrepot;
2174 $line->line_id = $obj->line_id;
2175 $line->rowid = $obj->line_id;
2176 $line->id = $obj->line_id;
2178 $line->fk_origin =
'orderline';
2180 $line->fk_element = $obj->fk_element;
2181 $line->origin_id = $obj->fk_element;
2182 $line->fk_elementdet = $obj->fk_elementdet;
2183 $line->origin_line_id = $obj->fk_elementdet;
2184 $line->element_type = $obj->element_type;
2186 $line->fk_expedition = $this->id;
2188 $line->stockable_product = $obj->stockable_product;
2189 $line->product_type = $obj->product_type;
2190 $line->fk_product = $obj->fk_product;
2191 $line->fk_product_type = $obj->fk_product_type;
2192 $line->ref = $obj->product_ref;
2193 $line->product_ref = $obj->product_ref;
2194 $line->product_label = $obj->product_label;
2195 $line->libelle = $obj->product_label;
2196 $line->product_barcode = $obj->product_barcode;
2197 $line->product_tosell = $obj->product_tosell;
2198 $line->product_tobuy = $obj->product_tobuy;
2199 $line->product_tobatch = $obj->product_tobatch;
2200 $line->fk_fournprice = $obj->fk_fournprice;
2201 $line->label = $obj->custom_label;
2202 $line->description = $obj->description;
2203 $line->qty_asked = $obj->qty_asked;
2204 $line->rang = $obj->rang;
2205 $line->weight = $obj->weight;
2206 $line->weight_units = $obj->weight_units;
2207 $line->length = $obj->length;
2208 $line->length_units = $obj->length_units;
2209 $line->width = $obj->width;
2210 $line->width_units = $obj->width_units;
2211 $line->height = $obj->height;
2212 $line->height_units = $obj->height_units;
2213 $line->surface = $obj->surface;
2214 $line->surface_units = $obj->surface_units;
2215 $line->volume = $obj->volume;
2216 $line->volume_units = $obj->volume_units;
2217 $line->stockable_product = $obj->stockable_product;
2218 $line->fk_unit = $obj->fk_unit;
2220 $line->extraparams = !empty($obj->extraparams) ? (array) json_decode($obj->extraparams,
true) : array();
2222 $line->pa_ht = $obj->pa_ht;
2225 $localtax_array = array(0 => $obj->localtax1_type, 1 => $obj->localtax1_tx, 2 => $obj->localtax2_type, 3 => $obj->localtax2_tx);
2226 $localtax1_tx =
get_localtax($obj->tva_tx, 1, $this->thirdparty);
2227 $localtax2_tx =
get_localtax($obj->tva_tx, 2, $this->thirdparty);
2230 $tabprice =
calcul_price_total($obj->qty_shipped, $obj->subprice, $obj->remise_percent, $obj->tva_tx, $localtax1_tx, $localtax2_tx, 0,
'HT', $obj->info_bits, $obj->fk_product_type,
$mysoc, $localtax_array);
2231 $line->desc = $obj->description;
2232 $line->qty = $line->qty_shipped;
2233 $line->total_ht = (float) $tabprice[0];
2234 $line->total_localtax1 = (float) $tabprice[9];
2235 $line->total_localtax2 = (float) $tabprice[10];
2236 $line->total_ttc = (float) $tabprice[2];
2237 $line->total_tva = (float) $tabprice[1];
2238 $line->vat_src_code = $obj->vat_src_code;
2239 $line->tva_tx = $obj->tva_tx;
2240 $line->localtax1_tx = $obj->localtax1_tx;
2241 $line->localtax2_tx = $obj->localtax2_tx;
2242 $line->info_bits = $obj->info_bits;
2243 $line->price = $obj->price;
2244 $line->subprice = $obj->subprice;
2245 $line->fk_remise_except = $obj->fk_remise_except;
2246 $line->remise_percent = $obj->remise_percent;
2248 $this->total_ht += $tabprice[0];
2249 $this->total_tva += $tabprice[1];
2250 $this->total_ttc += $tabprice[2];
2251 $this->total_localtax1 += $tabprice[9];
2252 $this->total_localtax2 += $tabprice[10];
2254 $line->date_start = $this->db->jdate($obj->date_start);
2255 $line->date_end = $this->db->jdate($obj->date_end);
2257 $line->special_code = $obj->special_code;
2260 $this->fk_multicurrency = $obj->fk_multicurrency;
2261 $this->multicurrency_code = $obj->multicurrency_code;
2262 $line->multicurrency_subprice = $obj->multicurrency_subprice;
2263 $line->multicurrency_total_ht = $obj->multicurrency_total_ht;
2264 $line->multicurrency_total_tva = $obj->multicurrency_total_tva;
2265 $line->multicurrency_total_ttc = $obj->multicurrency_total_ttc;
2267 $this->multicurrency_total_ht += $obj->multicurrency_total_ht;
2268 $this->multicurrency_total_tva += $obj->multicurrency_total_tva;
2269 $this->multicurrency_total_ttc += $obj->multicurrency_total_ttc;
2271 if ($originline != $obj->fk_elementdet) {
2272 $line->detail_batch = array();
2276 if (
isModEnabled(
'productbatch') && $obj->line_id > 0 && $obj->product_tobatch > 0) {
2277 $newdetailbatch = $shipmentlinebatch->fetchAll($obj->line_id, $obj->fk_product);
2279 if (is_array($newdetailbatch)) {
2280 if ($originline != $obj->fk_elementdet) {
2281 $line->detail_batch = $newdetailbatch;
2283 $line->detail_batch = array_merge($line->detail_batch, $newdetailbatch);
2290 $detail_children = array();
2291 $line_child_list = array();
2292 $res = $line->findAllChild($line->id, $line_child_list, 1);
2294 foreach ($line_child_list as $child_line) {
2295 foreach ($child_line as $child_obj) {
2296 $child_product_id = (int) $child_obj->fk_product;
2297 $child_warehouse_id = (int) $child_obj->fk_warehouse;
2299 if ($child_warehouse_id > 0) {
2301 if (!isset($detail_children[$child_product_id])) {
2302 $detail_children[$child_product_id] = array();
2304 if (!isset($detail_children[$child_product_id][$child_warehouse_id])) {
2305 $detail_children[$child_product_id][$child_warehouse_id] = 0;
2307 $detail_children[$child_product_id][$child_warehouse_id] += $child_obj->qty;
2312 $line->detail_children = $detail_children;
2315 $line->fetch_optionals();
2317 if ($originline != $obj->fk_elementdet) {
2318 $this->lines[$lineindex] = $line;
2321 $line->total_ht += $tabprice[0];
2322 $line->total_localtax1 += $tabprice[9];
2323 $line->total_localtax2 += $tabprice[10];
2324 $line->total_ttc += $tabprice[2];
2325 $line->total_tva += $tabprice[1];
2329 $originline = $obj->fk_elementdet;
2331 $this->db->free($resql);
2334 $this->error = $this->db->error();
2351 $this->lines = array();
2353 $sql =
'SELECT ed.rowid, ed.fk_expedition, ed.fk_entrepot, ed.fk_product, ed.fk_unit, ed.description, ed.fk_elementdet, ed.fk_element, ed.element_type, ed.qty, ed.rang';
2354 $sql .=
' FROM '.MAIN_DB_PREFIX.$this->table_element_line.
' as ed';
2355 $sql .=
' LEFT JOIN '.MAIN_DB_PREFIX.
'product as p ON (p.rowid = ed.fk_product)';
2356 $sql .=
' WHERE ed.fk_expedition = '.((int) $this->
id);
2357 $sql .=
' ORDER BY ed.rang, ed.rowid';
2359 dol_syslog(get_class($this).
"::fetch_lines_free", LOG_DEBUG);
2360 $result = $this->db->query($sql);
2362 $num = $this->db->num_rows($result);
2366 $objp = $this->db->fetch_object($result);
2370 $line->rowid = $objp->rowid;
2371 $line->id = $objp->rowid;
2372 $line->fk_expedition = $this->id;
2373 $line->description = $objp->description;
2374 $line->qty = $objp->qty;
2375 $line->fk_entrepot = $objp->fk_entrepot;
2376 $line->fk_product = $objp->fk_product;
2377 $line->rang = $objp->rang;
2378 $line->fk_element = $objp->fk_element;
2379 $line->fk_unit = $objp->fk_unit;
2380 $line->fk_elementdet = $objp->fk_elementdet;
2381 $line->fk_element_type = $objp->element_type;
2382 $line->fetch_optionals();
2384 $this->lines[$i] = $line;
2389 $this->db->free($result);
2393 $this->error = $this->db->error();
2419 if ($this->
status == self::STATUS_DRAFT) {
2425 $line->fetch($lineid);
2427 if ($line->delete($user) > 0) {
2430 $this->db->commit();
2433 $this->db->rollback();
2437 $this->error =
'ErrorDeleteLineNotAllowedByObjectStatus';
2451 global $conf, $langs;
2453 $langs->load(
'sendings');
2455 $nofetch = !empty($params[
'nofetch']);
2458 $datas[
'picto'] =
img_picto(
'', $this->picto).
' <u class="paddingrightonly">'.$langs->trans(
"Shipment").
'</u>';
2459 if (isset($this->
status)) {
2460 $datas[
'picto'] .=
' '.$this->getLibStatut(5);
2462 $datas[
'ref'] =
'<br><b>'.$langs->trans(
'Ref').
':</b> '.$this->ref;
2463 $datas[
'refcustomer'] =
'<br><b>'.$langs->trans(
'RefCustomer').
':</b> '.($this->ref_customer ? $this->ref_customer : $this->ref_client);
2465 $langs->load(
'companies');
2466 if (empty($this->thirdparty)) {
2469 $datas[
'customer'] =
'<br><b>'.$langs->trans(
'Customer').
':</b> '.$this->thirdparty->getNomUrl(1,
'', 0, 1);
2486 public function getNomUrl($withpicto = 0, $option =
'', $max = 0, $short = 0, $notooltip = 0, $save_lastsearch_value = -1)
2488 global $langs, $hookmanager;
2493 'objecttype' => $this->element,
2494 'option' => $option,
2497 $classfortooltip =
'classfortooltip';
2500 $classfortooltip =
'classforajaxtooltip';
2501 $dataparams =
' data-params="'.dol_escape_htmltag(json_encode($params)).
'"';
2507 $url = DOL_URL_ROOT.
'/expedition/card.php?id='.$this->id;
2513 if ($option !==
'nolink') {
2515 $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
2516 if ($save_lastsearch_value == -1 && isset($_SERVER[
"PHP_SELF"]) && preg_match(
'/list\.php/', $_SERVER[
"PHP_SELF"])) {
2517 $add_save_lastsearch_values = 1;
2519 if ($add_save_lastsearch_values) {
2520 $url .=
'&save_lastsearch_values=1';
2525 if (empty($notooltip)) {
2527 $label = $langs->trans(
"Shipment");
2528 $linkclose .=
' alt="'.dolPrintHTMLForAttribute($label).
'"';
2530 $linkclose .= ($label ?
' title="'.dolPrintHTMLForAttribute($label).
'"' :
' title="tocomplete"');
2531 $linkclose .= $dataparams.
' class="'.$classfortooltip.
'"';
2534 $linkstart =
'<a href="'.$url.
'"';
2535 $linkstart .= $linkclose.
'>';
2538 $result .= $linkstart;
2540 $result .=
img_object(($notooltip ?
'' : $label), ($this->picto ? $this->picto :
'generic'), ($notooltip ? (($withpicto != 2) ?
'class="paddingright"' :
'') :
'class="'.(($withpicto != 2) ?
'paddingright ' :
'').
'"'), 0, 0, $notooltip ? 0 : 1);
2542 if ($withpicto != 2) {
2543 $result .= $this->ref;
2545 $result .= $linkend;
2547 $hookmanager->initHooks(array($this->element .
'dao'));
2548 $parameters = array(
'id' => $this->
id,
'getnomurl' => &$result);
2549 $reshook = $hookmanager->executeHooks(
'getNomUrl', $parameters, $this, $action);
2551 $result = $hookmanager->resPrint;
2553 $result .= $hookmanager->resPrint;
2582 $labelStatus = $langs->transnoentitiesnoconv($this->labelStatus[$status]);
2583 $labelStatusShort = $langs->transnoentitiesnoconv($this->labelStatusShort[$status]);
2585 $statusType =
'status'.$status;
2586 if ($status == self::STATUS_VALIDATED) {
2587 $statusType =
'status4';
2589 if ($status == self::STATUS_CLOSED) {
2590 $statusType =
'status6';
2592 if ($status == self::STATUS_CANCELED) {
2593 $statusType =
'status9';
2597 $status_label = $this->signed_status ? $labelStatus . $signed_label : $labelStatus;
2598 $status_label_short = $this->signed_status ? $labelStatusShort . $signed_label : $labelStatusShort;
2600 return dolGetStatus($status_label, $status_label_short,
'', $statusType, $mode);
2612 global $langs, $conf;
2614 $selected = (empty($arraydata[
'selected']) ? 0 : $arraydata[
'selected']);
2616 $return =
'<div class="box-flex-item box-flex-grow-zero">';
2617 $return .=
'<div class="info-box info-box-sm">';
2618 $return .=
'<div class="info-box-icon bg-infobox-action">';
2620 $return .=
'</div>';
2621 $return .=
'<div class="info-box-content">';
2622 $return .=
'<span class="info-box-ref inline-block tdoverflowmax150 valignmiddle">' . $this->
getNomUrl() .
'</span>';
2623 if ($selected >= 0) {
2624 $return .=
'<input id="cb'.$this->id.
'" class="flat checkforselect fright" type="checkbox" name="toselect[]" value="'.$this->
id.
'"'.($selected ?
' checked="checked"' :
'').
'>';
2626 $return .=
'<br><div class="info-box-ref tdoverflowmax150">'.$this->thirdparty->getNomUrl(1).
'</div>';
2627 $return .=
'<div class="info-box-ref amount">'.price($this->total_ht, 0, $langs, 0, -1, -1,
getDolCurrency()).
' '.$langs->trans(
'HT').
'</div>';
2628 $return .=
'<div class="info-box-status">'.$this->getLibStatut(3).
'</div>';
2629 $return .=
'</div>';
2630 $return .=
'</div>';
2631 $return .=
'</div>';
2649 dol_syslog(get_class($this).
"::initAsSpecimen");
2652 $order->initAsSpecimen();
2656 $this->
ref =
'SPECIMEN';
2657 $this->specimen = 1;
2659 $this->livraison_id = 0;
2661 $this->date_creation = $now;
2662 $this->date_valid = $now;
2663 $this->date_delivery = $now + 24 * 3600;
2664 $this->date_expedition = $now + 24 * 3600;
2666 $this->entrepot_id = 0;
2667 $this->fk_delivery_address = 0;
2670 $this->commande_id = 0;
2671 $this->commande = $order;
2673 $this->origin_id = 1;
2674 $this->origin_type =
'commande';
2676 $this->note_private =
'Private note';
2677 $this->note_public =
'Public note';
2681 while ($xnbp < $nbp) {
2683 $line->product_desc = $langs->trans(
"Description").
" ".$xnbp;
2684 $line->product_label = $langs->trans(
"Description").
" ".$xnbp;
2686 $line->qty_asked = 5;
2687 $line->qty_shipped = 4;
2688 $line->fk_product = $this->commande->lines[$xnbp]->fk_product;
2690 $line->weight = 1.123456;
2691 $line->weight_units = 0;
2693 $line->volume = 2.34567;
2694 $line->volume_unit = 0;
2696 $this->lines[] = $line;
2727 if ($user->hasRight(
'expedition',
'creer')) {
2728 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition";
2729 $sql .=
" SET date_delivery = ".($delivery_date ?
"'".$this->db->idate($delivery_date).
"'" :
'null');
2730 $sql .=
" WHERE rowid = ".((int) $this->
id);
2732 dol_syslog(get_class($this).
"::setDeliveryDate", LOG_DEBUG);
2733 $resql = $this->db->query($sql);
2735 $this->date_delivery = $delivery_date;
2738 $this->error = $this->db->error();
2755 if ($user->hasRight(
'expedition',
'creer')) {
2756 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition";
2757 $sql .=
" SET date_expedition = ".($shipping_date ?
"'".$this->db->idate($shipping_date).
"'" :
'null');
2758 $sql .=
" WHERE rowid = ".((int) $this->
id);
2760 dol_syslog(get_class($this).
"::setShippingDate", LOG_DEBUG);
2761 $resql = $this->db->query($sql);
2763 $this->date_shipping = $shipping_date;
2766 $this->error = $this->db->error();
2786 $sql =
"SELECT em.rowid, em.code, em.libelle as label";
2787 $sql .=
" FROM ".MAIN_DB_PREFIX.
"c_shipment_mode as em";
2788 $sql .=
" WHERE em.active = 1";
2789 $sql .=
" ORDER BY em.libelle ASC";
2791 $resql = $this->db->query($sql);
2793 while ($obj = $this->db->fetch_object($resql)) {
2794 $label = $langs->trans(
'SendingMethod'.$obj->code);
2795 $this->meths[(int) $obj->rowid] = ($label !=
'SendingMethod'.$obj->code ? $label : $obj->label);
2812 $this->listmeths = [];
2815 $sql =
"SELECT em.rowid, em.code, em.libelle as label, em.description, em.tracking, em.active";
2816 $sql .=
" FROM ".MAIN_DB_PREFIX.
"c_shipment_mode as em";
2818 $sql .=
" WHERE em.rowid=".((int) $id);
2821 $resql = $this->db->query($sql);
2823 while ($obj = $this->db->fetch_object($resql)) {
2824 $label = $langs->trans(
'SendingMethod'.$obj->code);
2825 $this->listmeths[$i] = [
2826 'rowid' => (int) $obj->rowid,
2827 'code' => $obj->code,
2828 'libelle' => ($label !=
'SendingMethod'.$obj->code ? $label : $obj->label),
2829 'description' => $obj->description,
2830 'tracking' => $obj->tracking,
2831 'active' => (
int) $obj->active,
2847 if (!empty($this->shipping_method_id)) {
2848 $sql =
"SELECT em.code, em.tracking";
2849 $sql .=
" FROM ".MAIN_DB_PREFIX.
"c_shipment_mode as em";
2850 $sql .=
" WHERE em.rowid = ".((int) $this->shipping_method_id);
2852 $resql = $this->db->query($sql);
2854 if ($obj = $this->db->fetch_object($resql)) {
2855 $tracking = (string) $obj->tracking;
2860 if (!empty($tracking) && !empty($value)) {
2861 $url = str_replace(
'{TRACKID}', $value, $tracking);
2862 $this->tracking_url = sprintf(
'<a target="_blank" rel="noopener noreferrer" href="%s">%s</a>', $url, $value);
2864 $this->tracking_url = $value;
2880 if ($this->
status == self::STATUS_CLOSED) {
2887 if (empty($this->date_shipping)) {
2888 $sql .=
", date_expedition = '".$this->db->escape($this->db->idate(
dol_now())).
"'";
2890 $sql .=
" WHERE rowid = ".((int) $this->
id).
" AND fk_statut > 0";
2892 $resql = $this->db->query($sql);
2895 if ($this->origin_type ==
'commande' && $this->origin_id > 0) {
2897 $order->fetch($this->origin_id);
2899 $order->loadExpeditions(self::STATUS_CLOSED);
2901 $shipments_match_order = 1;
2902 foreach ($order->lines as $line) {
2903 $lineid = $line->id;
2905 if (($line->product_type == 0 ||
getDolGlobalString(
'STOCK_SUPPORTS_SERVICES')) && $order->expeditions[$lineid] != $qty) {
2906 $shipments_match_order = 0;
2907 $text =
'Qty for order line id '.$lineid.
' is '.$qty.
'. However in the shipments with status Expedition::STATUS_CLOSED='.self::STATUS_CLOSED.
' we have qty = '.$order->expeditions[$lineid].
', so we can t close order';
2912 if ($shipments_match_order) {
2913 dol_syslog(
"Qty for the ".count($order->lines).
" lines of the origin order is same than qty for lines in the shipment we close (shipments_match_order is true), with new status Expedition::STATUS_CLOSED=".self::STATUS_CLOSED.
', so we close order');
2915 $order->cloture($user);
2933 $result = $this->call_trigger(
'SHIPPING_CLOSED', $user);
2944 $this->db->commit();
2950 $this->db->rollback();
2969 require_once DOL_DOCUMENT_ROOT .
'/product/stock/class/mouvementstock.class.php';
2971 $langs->load(
"agenda");
2975 $sql .=
" ed.rowid as edid, ed.fk_product, ed.qty, ed.fk_entrepot";
2976 $sql .=
", cd.rowid as cdid";
2977 $sql .=
", cd.subprice";
2978 $sql .=
", edb.rowid as edbrowid, edb.eatby, edb.sellby, edb.batch, edb.qty as edbqty, edb.fk_origin_stock";
2980 $sql .=
" FROM " . $this->db->prefix() .
"expeditiondet as ed";
2981 $sql .=
" LEFT JOIN " . $this->db->prefix() .
"commandedet as cd ON cd.rowid = ed.fk_elementdet";
2982 $sql .=
" LEFT JOIN " . $this->db->prefix() .
"expeditiondet_batch as edb on edb.fk_expeditiondet = ed.rowid";
2983 $sql .=
" INNER JOIN " . $this->db->prefix() .
"expedition as e ON ed.fk_expedition = e.rowid";
2984 $sql .=
" WHERE ed.fk_expedition = " . ((int) $this->
id);
2987 dol_syslog(get_class($this) .
"::valid select details", LOG_DEBUG);
2988 $resql = $this->db->query($sql);
2990 $cpt = $this->db->num_rows($resql);
2991 for ($i = 0; $i < $cpt; $i++) {
2992 $obj = $this->db->fetch_object($resql);
2993 if (empty($obj->edbrowid)) {
2996 $qty = $obj->edbqty;
2998 if ($qty <= 0 || ($qty < 0 && !
getDolGlobalInt(
'SHIPMENT_ALLOW_NEGATIVE_QTY'))) {
3001 dol_syslog(get_class($this) .
"::valid movement index " . $i .
" ed.rowid=" . $obj->edid .
" edb.rowid=" . $obj->edbrowid);
3004 $mouvS->origin = &$this;
3005 $mouvS->setOrigin($this->element, $this->
id, $obj->cdid, $obj->edid);
3007 if (empty($obj->edbrowid)) {
3011 $result = $mouvS->livraison($user, $obj->fk_product, $obj->fk_entrepot, $qty, $obj->subprice, $langs->trans($labelmovement, $obj->ref));
3021 $result = $mouvS->livraison($user, $obj->fk_product, $obj->fk_entrepot, $qty, $obj->subprice, $langs->trans($labelmovement, $obj->ref),
'', $this->db->jdate($obj->eatby), $this->db->jdate($obj->sellby), $obj->batch, $obj->fk_origin_stock);
3031 $sqldelete =
"DELETE FROM ".$this->db->prefix().
"product_stock WHERE reel = 0 AND rowid NOT IN (SELECT fk_product_stock FROM ".$this->db->prefix().
"product_batch as pb)";
3032 $resqldelete = $this->db->query($sqldelete);
3036 $this->error = $this->db->lasterror();
3037 $this->errors[] = $this->db->lasterror();
3060 $sql =
'UPDATE '.MAIN_DB_PREFIX.
'expedition SET billed = 1';
3061 $sql .=
" WHERE rowid = ".((int) $this->
id).
' AND fk_statut > 0';
3063 $resql = $this->db->query($sql);
3068 $result = $this->call_trigger(
'SHIPPING_BILLED', $user);
3075 $this->errors[] = $this->db->lasterror;
3078 if (empty($error)) {
3079 $this->db->commit();
3082 $this->db->rollback();
3097 if ($this->
status <= self::STATUS_DRAFT) {
3101 return $this->
setStatusCommon($user, self::STATUS_DRAFT, $notrigger,
'SHIPMENT_UNVALIDATE');
3111 global $langs, $user;
3116 if ($this->
status == self::STATUS_VALIDATED) {
3122 $oldbilled = $this->billed;
3124 $sql =
'UPDATE '.MAIN_DB_PREFIX.
'expedition SET fk_statut = 1';
3125 $sql .=
" WHERE rowid = ".((int) $this->
id).
' AND fk_statut > 0';
3127 $resql = $this->db->query($sql);
3135 require_once DOL_DOCUMENT_ROOT.
'/product/stock/class/mouvementstock.class.php';
3137 $langs->load(
"agenda");
3141 $sql =
"SELECT cd.fk_product, cd.subprice,";
3142 $sql .=
" ed.rowid, ed.qty, ed.fk_entrepot,";
3143 $sql .=
" edb.rowid as edbrowid, edb.eatby, edb.sellby, edb.batch, edb.qty as edbqty, edb.fk_origin_stock";
3144 $sql .=
" FROM ".MAIN_DB_PREFIX.
"commandedet as cd,";
3145 $sql .=
" ".MAIN_DB_PREFIX.
"expeditiondet as ed";
3146 $sql .=
" LEFT JOIN ".MAIN_DB_PREFIX.
"expeditiondet_batch as edb on edb.fk_expeditiondet = ed.rowid";
3147 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
3148 $sql .=
" AND cd.rowid = ed.fk_elementdet";
3150 dol_syslog(get_class($this).
"::valid select details", LOG_DEBUG);
3151 $resql = $this->db->query($sql);
3153 $cpt = $this->db->num_rows($resql);
3154 for ($i = 0; $i < $cpt; $i++) {
3155 $obj = $this->db->fetch_object($resql);
3156 if (empty($obj->edbrowid)) {
3159 $qty = $obj->edbqty;
3164 dol_syslog(get_class($this).
"::reopen expedition movement index ".$i.
" ed.rowid=".$obj->rowid.
" edb.rowid=".$obj->edbrowid);
3168 $mouvS->origin = &$this;
3169 $mouvS->setOrigin($this->element, $this->
id);
3171 if (empty($obj->edbrowid)) {
3175 $result = $mouvS->livraison($user, $obj->fk_product, $obj->fk_entrepot, -$qty, $obj->subprice, $langs->trans(
"ShipmentUnClassifyCloseddInDolibarr", $this->ref));
3185 $result = $mouvS->livraison($user, $obj->fk_product, $obj->fk_entrepot, -$qty, $obj->subprice, $langs->trans(
"ShipmentUnClassifyCloseddInDolibarr", $this->ref),
'', $this->db->jdate($obj->eatby), $this->db->jdate($obj->sellby), $obj->batch, $obj->fk_origin_stock);
3194 $this->error = $this->db->lasterror();
3201 $result = $this->call_trigger(
'SHIPPING_REOPEN', $user);
3208 $this->errors[] = $this->db->lasterror();
3212 $this->db->commit();
3217 $this->billed = $oldbilled;
3218 $this->db->rollback();
3234 public function generateDocument($modele, $outputlangs, $hidedetails = 0, $hidedesc = 0, $hideref = 0, $moreparams =
null)
3236 $outputlangs->load(
"products");
3241 if (!empty($this->model_pdf)) {
3242 $modele = $this->model_pdf;
3248 $modelpath =
"core/modules/expedition/doc/";
3252 return $this->
commonGenerateDocument($modelpath, $modele, $outputlangs, $hidedetails, $hidedesc, $hideref, $moreparams);
Class to manage customers orders.
const STATUS_SHIPMENTONPROCESS
Shipment on process.
const STATUS_VALIDATED
Validated status.
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.
setStatut($status, $elementId=null, $elementType='', $trigkey='', $fieldstatus='')
Set status of an object.
add_object_linked($origin=null, $origin_id=null, $f_user=null, $notrigger=0)
Add an object link into llx_element_element.
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.
deleteObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $rowid=0, $f_user=null, $notrigger=0)
Delete all links between an object $this.
setErrorsFromObject($object)
setErrorsFromObject
updateRangOfLine($rowid, $rang)
Update position of line (rang)
deleteExtraFields()
Delete all extra fields values for the current object.
setStatusCommon($user, $status, $notrigger=0, $triggercode='')
Set to a status.
fetchObjectLinked($sourceid=null, $sourcetype='', $targetid=null, $targettype='', $clause='OR', $alsosametype=1, $orderby='sourcetype', $loadalsoobjects=1)
Fetch array of objects linked to current object (object of enabled modules only).
errorsToString()
Method to output saved errors.
static commonReplaceThirdparty(DoliDB $dbs, $origin_id, $dest_id, array $tables, $ignoreerrors=0)
Function used to replace a thirdparty id with another one.
line_max($fk_parent_line=0)
Get max value used for position of line (rang)
fetch_origin()
Read linked origin object.
insertExtraFields($trigger='', $userused=null)
Add/Update all extra fields values for the current object.
delete_linked_contact($source='', $code='')
Delete all links between an object $this and all its contacts in llx_element_contact.
add_contact($fk_socpeople, $type_contact, $source='external', $notrigger=0)
Add a link between element $this->element and a contact.
Class to manage receptions.
Class to manage Dolibarr database access.
getNomUrl($withpicto=0, $option='', $max=0, $short=0, $notooltip=0, $save_lastsearch_value=-1)
Return clickable link of object (with eventually picto)
addline_batch($dbatch, $array_options=[], $origin_line=null)
Add a shipment line with batch record.
create_delivery($user)
Create a delivery receipt from a shipment.
setDraft($user, $notrigger=0)
Set draft status.
const STATUS_SHIPMENT_IN_PROGRESS
Expedition in progress -> package exit the warehouse and is now in the truck or into the hand of the ...
getUrlTrackingStatus($value='')
Forge an set tracking url.
setClosed()
Classify the shipping as closed (this records also the stock movement)
__construct($db)
Constructor.
create_line($entrepot_id, $origin_line_id, $qty, $rang=0, $array_options=[], $parent_line_id=0, $product_id=0)
Create a expedition line.
addline($entrepot_id, $id, $qty, $array_options=[], $fk_product=0, $fk_parent=0)
Add an expedition line.
create($user, $notrigger=0)
Create expedition en base.
LibStatut($status, $mode)
Return label of a status.
setBilled()
Classify the shipping as invoiced (used for example by trigger when WORKFLOW_SHIPPING_CLASSIFY_BILLED...
generateDocument($modele, $outputlangs, $hidedetails=0, $hidedesc=0, $hideref=0, $moreparams=null)
Create a document onto disk according to template module.
getTooltipContentArray($params)
getTooltipContentArray
addlinefree($qty, $element_type, $fk_product, $fk_unit, $rang, $description, $fk_parent, $array_options=[])
Add a simple expedition line.
setDeliveryDate($user, $delivery_date)
Set the planned delivery date.
getKanbanView($option='', $arraydata=null)
Return clickable link of object (with eventually picto)
setShippingDate($user, $shipping_date)
Set the shipping date.
const STATUS_DRAFT
Draft status.
const STATUS_CANCELED
Canceled status.
getLibStatut($mode=0)
Return status label.
set_date_livraison($user, $delivery_date)
Set delivery date.
initAsSpecimen()
Initialise an instance with random values.
updatelinefree($rowid, $qty, $element_type, $fk_product, $fk_unit, $rang, $description, $fk_parent, $notrigger, $array_options=array())
Update a simple expedition line.
const STATUS_CLOSED
Closed status -> parcel was received by customer / end of process prev status : validated or shipment...
const STATUS_VALIDATED
Validated status -> parcel is ready to be sent prev status : draft next status : closed or shipment_i...
create_line_batch($line_ext, $array_options=[])
Create the detail of the expedition line.
manageStockMvtOnEvt($user, $labelmovement='ShipmentClassifyClosedInDolibarr')
Manage Stock MVt onb Close or valid Shipment.
valid($user, $notrigger=0)
Validate object and update stock if option enabled.
update($user=null, $notrigger=0)
Update database.
fetch_lines_free()
Load lines of simple shipment.
fetch_delivery_methods()
Fetch deliveries method and return an array.
fetch($id, $ref='', $ref_ext='', $notused='')
Get object and lines from database.
reOpen()
Classify the shipping as validated/opened.
deleteLine($user, $lineid)
Delete detail line.
getLinesArray()
Create an array of shipment lines.
static replaceThirdparty(DoliDB $dbs, $origin_id, $dest_id)
Function used to replace a thirdparty id with another one.
list_delivery_methods($id=0)
Fetch all deliveries method and return an array.
Class to manage lines of shipment.
CRUD class for batch number management within shipment.
Class to manage stock movements.
Class to manage order lines.
Class to manage products or services.
const TYPE_SERVICE
Service.
Manage record for batch number management.
Class to manage third parties objects (customers, suppliers, prospects...)
print $langs trans("Ref").' m titre as m m statut as status
Or an array listing all the potential status of the object: array: int of the status => translated la...
getLibSignedStatus(int $mode=0)
Returns the label for signed status.
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.
dol_delete_dir_recursive($dir, $count=0, $nophperrors=0, $onlysub=0, &$countdeleted=0, $indexdatabase=1, $nolog=0, $level=0)
Remove a directory $dir and its subdirectories (or only files and subdirectories)
dol_dir_list($utf8_path, $types="all", $recursive=0, $filter="", $excludefilter=null, $sortcriteria="name", $sortorder=SORT_ASC, $mode=0, $nohook=0, $relativename="", $donotfollowsymlinks=0, $nbsecondsold=0)
Scan a directory and return a list of files/directories.
dol_now($mode='gmt')
Return date for now.
setEntity($currentobject)
Set entity id to use when to create an object.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2, $allowothertags=array())
Show picto whatever it's its name (generic function)
GETPOSTINT($paramname, $method=0)
Return the value of a $_GET or $_POST supervariable, converted into integer.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
img_object($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $allowothertags=array())
Show a picto called object_picto (generic function)
dol_sanitizeFileName($str, $newstr='_', $unaccent=1, $includequotes=0, $allowdash=0)
Clean a string to use it as a file name.
dol_strlen($string, $stringencoding='UTF-8')
Make a strlen call.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
getDolCurrency()
Return the main currency ('EUR', 'USD', ...)
dolGetStatus($statusLabel='', $statusLabelShort='', $html='', $statusType='status0', $displayMode=0, $url='', $params=array())
Output the badge of a status.
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
dol_print_error($db=null, $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
isModEnabled($module)
Is Dolibarr module enabled.
get_localtax($vatrate, $local, $thirdparty_buyer=null, $thirdparty_seller=null, $vatnpr=0)
Return localtax rate for a particular VAT rate, when selling a product with vat $vatrate,...
dol_syslog($message, $level=LOG_INFO, $ident=0, $suffixinfilename='', $restricttologhandler='', $logcontext=null)
Write log message into outputs.
calcul_price_total($qty, $pu, $remise_percent_ligne, $txtva, $uselocaltax1_rate, $uselocaltax2_rate, $remise_percent_global, $price_base_type, $info_bits, $type, $seller=null, $localtaxes_array=[], $progress=100, $multicurrency_tx=1, $pu_devise=0, $multicurrency_code='')
Calculate totals (net, vat, ...) of a line.
if(getDolGlobalString( 'TAKEPOS_SHOW_CUSTOMER')) print $langs trans('Date')." left Label right Qty right Price right TotalHT right TotalTTC right right right right right right right right right centpercent right TotalHT right n right VAT right n right TotalVAT right n No sujeto a RE IRPF right TotalLT1 right n right TotalLT2 right n right TotalTTC right n takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency right TotalTTC takeposcustomercurrency right takeposcustomercurrency n right PaymentTypeShortLIQ right SELECT p pos_change as p datep as date
$conf db user
Active Directory does not allow anonymous connections.