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;
350 $this->ismultientitymanaged = 1;
351 $this->isextrafieldmanaged = 1;
354 $this->labelStatus = array();
355 $this->labelStatus[-1] =
'StatusSendingCanceled';
356 $this->labelStatus[0] =
'StatusSendingDraft';
357 $this->labelStatus[1] =
'StatusSendingValidated';
358 $this->labelStatus[2] =
'StatusSendingProcessed';
361 $this->labelStatusShort = array();
362 $this->labelStatusShort[-1] =
'StatusSendingCanceledShort';
363 $this->labelStatusShort[0] =
'StatusSendingDraftShort';
364 $this->labelStatusShort[1] =
'StatusSendingValidatedShort';
365 $this->labelStatusShort[2] =
'StatusSendingProcessedShort';
374 public function getNextNumRef($soc)
376 global $langs,
$conf;
377 $langs->load(
"sendings");
386 $dirmodels = array_merge(array(
'/'), (array)
$conf->modules_parts[
'models']);
388 foreach ($dirmodels as $reldir) {
392 $mybool = ((bool) @include_once $dir.$file) || $mybool;
400 $obj =
new $classname();
401 '@phan-var-force ModelNumRefExpedition $obj';
403 $numref = $obj->getNextValue($soc, $this);
408 dol_print_error($this->db, get_class($this).
"::getNextNumRef ".$obj->error);
412 print $langs->trans(
"Error").
" ".$langs->trans(
"Error_EXPEDITION_ADDON_NUMBER_NotDefined");
424 public function create($user, $notrigger = 0)
428 require_once DOL_DOCUMENT_ROOT.
'/product/stock/class/mouvementstock.class.php';
433 $this->tracking_number = preg_replace(
'/\s+/',
'', $this->tracking_number);
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";
476 $sql .=
") VALUES (";
478 $sql .=
", ".((int) $this->entity);
479 $sql .=
", ".($this->ref_customer ?
"'".$this->db->escape($this->ref_customer).
"'" :
"null");
480 $sql .=
", ".($this->ref_ext ?
"'".$this->db->escape($this->ref_ext).
"'" :
"null");
481 $sql .=
", '".$this->db->idate($this->date_creation).
"'";
482 $sql .=
", ".((int) $user->id);
483 $sql .=
", ".($this->date_shipping > 0 ?
"'".$this->db->idate($this->date_shipping).
"'" :
"null");
484 $sql .=
", ".($this->date_delivery > 0 ?
"'".$this->db->idate($this->date_delivery).
"'" :
"null");
485 $sql .=
", ".($this->socid > 0 ? ((int) $this->socid) :
"null");
486 $sql .=
", ".($this->fk_project > 0 ? ((int) $this->fk_project) :
"null");
487 $sql .=
", ".($this->fk_delivery_address > 0 ? ((int) $this->fk_delivery_address) :
"null");
488 $sql .=
", ".($this->shipping_method_id > 0 ? ((int) $this->shipping_method_id) :
"null");
489 $sql .=
", '".$this->db->escape($this->tracking_number).
"'";
490 $sql .=
", ".(is_numeric($this->weight) ? (float) $this->weight :
'NULL');
491 $sql .=
", ".(is_numeric($this->sizeS) ? (float) $this->sizeS :
'NULL');
492 $sql .=
", ".(is_numeric($this->sizeW) ? (float) $this->sizeW :
'NULL');
493 $sql .=
", ".(is_numeric($this->sizeH) ? (float) $this->sizeH :
'NULL');
494 $sql .=
", ".($this->weight_units !=
'' ? (int) $this->weight_units :
'NULL');
495 $sql .=
", ".($this->size_units !=
'' ? (int) $this->size_units :
'NULL');
496 $sql .=
", ".(!empty($this->note_private) ?
"'".$this->db->escape($this->note_private).
"'" :
"null");
497 $sql .=
", ".(!empty($this->note_public) ?
"'".$this->db->escape($this->note_public).
"'" :
"null");
498 $sql .=
", ".(!empty($this->model_pdf) ?
"'".$this->db->escape($this->model_pdf).
"'" :
"null");
499 $sql .=
", ".((int) $this->fk_incoterms);
500 $sql .=
", '".$this->db->escape($this->location_incoterms).
"'";
501 $sql .=
", ".((int) $this->signed_status);
502 $sql .=
", ".((int) $this->billed);
505 dol_syslog(get_class($this).
"::create", LOG_DEBUG);
506 $resql = $this->db->query($sql);
508 $this->
id = $this->db->last_insert_id(MAIN_DB_PREFIX.
"expedition");
511 $initialref =
'(PROV'.$this->id.
')';
512 if (!empty($this->
ref)) {
513 $initialref = $this->ref;
516 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition";
517 $sql .=
" SET ref = '".$this->db->escape($initialref).
"'";
518 $sql .=
" WHERE rowid = ".((int) $this->
id);
520 dol_syslog(get_class($this).
"::create", LOG_DEBUG);
521 if ($this->db->query($sql)) {
522 $this->
ref = $initialref;
524 $num = count($this->lines);
525 $kits_list = array();
527 for ($i = 0; $i < $num; $i++) {
529 $objectsrc->fetch($this->lines[$i]->origin_line_id);
530 if ($this->lines[$i]->product_type ==
"9" && $objectsrc->special_code == SUBTOTALS_SPECIAL_CODE) {
531 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) {
538 $line = $this->lines[$i];
539 if ($line->fk_product > 0) {
540 if (!isset($kits_list[$line->fk_product])) {
541 if (!is_object($line->product)) {
542 $line_product =
new Product($this->db);
543 $result = $line_product->fetch($line->fk_product,
'',
'',
'', 1, 1, 1);
548 $line_product = $line->product;
552 $line_product->get_sousproduits_arbo();
553 $prods_arbo = $line_product->get_arbo_each_prod($line->qty);
554 if (count($prods_arbo) > 0) {
555 $kits_list[$line->fk_product] = array(
556 'arbo' => $prods_arbo,
557 'total_qty' => $line->qty,
561 $kits_list[$line->fk_product][
'total_qty'] += $line->qty;
567 $kits_id_cached = array();
568 $sub_kits_id_cached = array();
569 for ($i = 0; $i < $num; $i++) {
570 $line = $this->lines[$i];
573 if (!isset($kits_id_cached[$line->fk_product])) {
574 if (!isset($line->detail_batch) || (isset($kits_list[$line->fk_product]) && !
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE'))) {
575 $qty = isset($kits_list[$line->fk_product]) ? $kits_list[$line->fk_product][
'total_qty'] : $line->qty;
576 $warehouse_id = (isset($kits_list[$line->fk_product]) && !
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) ? 0 : $line->entrepot_id;
577 $line_id = $this->
create_line($warehouse_id, $line->origin_line_id, $qty, $line->rang, $line->array_options, 0, $line->fk_product);
581 if (isset($kits_list[$line->fk_product])) {
582 $kits_id_cached[$line->fk_product] = $line_id;
590 $line_id = $kits_id_cached[$line->fk_product];
594 if (isset($kits_list[$line->fk_product]) && !
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) {
595 $prods_arbo = $kits_list[$line->fk_product][
'arbo'];
596 $total_qty = $kits_list[$line->fk_product][
'total_qty'];
599 $parent_line_id = $line_id;
601 $product_child_id = 0;
602 foreach ($prods_arbo as $index => $product_child_arr) {
617 $product_child_level = (int) $product_child_arr[
'level'];
618 $product_child_incdec = !empty($product_child_arr[
'incdec']);
621 if ($product_child_level != $level_last) {
622 $parent_line_id = $line_id;
623 $parent_product_id = $product_child_id;
624 if (isset($kits_id_cached[$parent_product_id])) {
625 $parent_line_id = $kits_id_cached[$parent_product_id];
627 $kits_id_cached[$parent_product_id] = $parent_line_id;
633 $next_level = $product_child_level;
634 $next_index = $index + 1;
635 if (isset($prods_arbo[$next_index])) {
636 $next_level = (int) $prods_arbo[$next_index][
'level'];
638 if ($next_level > $product_child_level) {
643 $product_child_id = (int) $product_child_arr[
'id'];
644 $product_child_qty = (float) $product_child_arr[
'nb_total'];
645 $warehouse_id = $line->entrepot_id;
646 if ($is_kit || !$product_child_incdec) {
647 if (!$product_child_incdec) {
648 $product_child_qty = 0;
654 if (!isset($sub_kits_id_cached[$product_child_id]) || $warehouse_id > 0) {
655 $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);
663 if (empty($warehouse_id)) {
664 $sub_kits_id_cached[$product_child_id] = $line_id;
668 $level_last = $product_child_level;
674 if (!$error && $this->
id && $this->origin_id) {
681 if (!$error && $this->
id &&
getDolGlobalInt(
'SHIPPING_USE_ITS_OWN_CONTACTS') &&
getDolGlobalString(
'MAIN_PROPAGATE_CONTACTS_FROM_ORIGIN') && !empty($this->origin_type) && !empty($this->origin_id)) {
682 $originforcontact = $this->origin_type;
683 $originidforcontact = $this->origin_id;
685 $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";
686 $sqlcontact .=
" WHERE element_id = ".((int) $originidforcontact).
" AND ec.fk_c_type_contact = ctc.rowid AND ctc.element = '".$this->db->escape($originforcontact).
"'";
688 $resqlcontact = $this->db->query($sqlcontact);
690 while ($objcontact = $this->db->fetch_object($resqlcontact)) {
691 $this->
add_contact($objcontact->fk_socpeople, $objcontact->code, $objcontact->source);
706 if (!$error && !$notrigger) {
708 $result = $this->call_trigger(
'SHIPPING_CREATE', $user);
718 foreach ($this->errors as $errmsg) {
719 dol_syslog(get_class($this).
"::create ".$errmsg, LOG_ERR);
720 $this->error .= ($this->error ?
', '.$errmsg : $errmsg);
722 $this->db->rollback();
727 $this->db->rollback();
732 $this->error = $this->db->lasterror().
" - sql=$sql";
733 $this->db->rollback();
738 $this->error = $this->db->error().
" - sql=$sql";
739 $this->db->rollback();
757 public function create_line($entrepot_id, $origin_line_id, $qty, $rang = 0, $array_options = [], $parent_line_id = 0, $product_id = 0)
763 $expeditionline->fk_expedition = $this->id;
764 $expeditionline->entrepot_id = $entrepot_id;
765 $expeditionline->fk_elementdet = $origin_line_id;
766 $expeditionline->element_type = $this->origin;
767 $expeditionline->fk_parent = $parent_line_id;
768 $expeditionline->fk_product = $product_id;
769 $expeditionline->qty = $qty;
770 $expeditionline->rang = $rang;
771 $expeditionline->array_options = $array_options;
773 if (!($expeditionline->fk_product > 0)) {
775 $order_line->fetch($expeditionline->fk_elementdet);
776 $expeditionline->fk_product = $order_line->fk_product;
779 if (($lineId = $expeditionline->insert($user)) < 0) {
780 $this->errors[] = $expeditionline->error;
798 $stockLocationQty = array();
800 $tab = $line_ext->detail_batch;
802 foreach ($tab as $detbatch) {
803 if (!empty($detbatch->fk_warehouse)) {
804 if (empty($stockLocationQty[$detbatch->fk_warehouse])) {
805 $stockLocationQty[$detbatch->fk_warehouse] = 0;
807 $stockLocationQty[$detbatch->fk_warehouse] += $detbatch->qty;
811 foreach ($stockLocationQty as $stockLocation => $qty) {
812 $line_id = $this->
create_line($stockLocation, $line_ext->origin_line_id, $qty, $line_ext->rang, $array_options);
817 foreach ($tab as $detbatch) {
818 if ($detbatch->fk_warehouse == $stockLocation) {
819 if (!($detbatch->create($line_id) > 0)) {
820 $this->errors = $detbatch->errors;
844 public function fetch($id, $ref =
'', $ref_ext =
'', $notused =
'')
849 if (empty($id) && empty($ref) && empty($ref_ext)) {
853 $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";
854 $sql .=
", e.date_valid";
855 $sql .=
", e.weight, e.weight_units, e.size, e.size_units, e.width, e.height";
856 $sql .=
", e.date_expedition as date_expedition, e.model_pdf, e.fk_address, e.date_delivery";
857 $sql .=
", e.fk_shipping_method, e.tracking_number";
858 $sql .=
", e.note_private, e.note_public";
859 $sql .=
', e.fk_incoterms, e.location_incoterms';
860 $sql .=
', e.signed_status';
861 $sql .=
', i.libelle as label_incoterms';
862 $sql .=
', s.libelle as shipping_method';
863 $sql .=
", el.fk_source as origin_id, el.sourcetype as origin_type";
864 $sql .=
" FROM ".MAIN_DB_PREFIX.
"expedition as e";
865 $sql .=
" LEFT JOIN ".MAIN_DB_PREFIX.
"element_element as el ON el.fk_target = e.rowid AND el.targettype = '".$this->db->escape($this->element).
"'";
866 $sql .=
' LEFT JOIN '.MAIN_DB_PREFIX.
'c_incoterms as i ON e.fk_incoterms = i.rowid';
867 $sql .=
' LEFT JOIN '.MAIN_DB_PREFIX.
'c_shipment_mode as s ON e.fk_shipping_method = s.rowid';
868 $sql .=
" WHERE e.entity IN (".getEntity(
'expedition').
")";
870 $sql .=
" AND e.rowid = ".((int) $id);
873 $sql .=
" AND e.ref='".$this->db->escape($ref).
"'";
876 $sql .=
" AND e.ref_ext='".$this->db->escape($ref_ext).
"'";
879 dol_syslog(get_class($this).
"::fetch", LOG_DEBUG);
880 $result = $this->db->query($sql);
882 if ($this->db->num_rows($result)) {
883 $obj = $this->db->fetch_object($result);
885 $this->
id = (int) $obj->rowid;
886 $this->entity = $obj->entity;
887 $this->
ref = $obj->ref;
888 $this->socid = $obj->socid;
889 $this->ref_customer = $obj->ref_customer;
890 $this->ref_ext = $obj->ref_ext;
891 $this->
status = $obj->fk_statut;
892 $this->statut = $this->status;
893 $this->signed_status = $obj->signed_status;
894 $this->user_author_id = $obj->fk_user_author;
895 $this->fk_user_author = $obj->fk_user_author;
896 $this->user_creation_id = $obj->fk_user_author;
897 $this->date_creation = $this->db->jdate($obj->date_creation);
898 $this->date_valid = $this->db->jdate($obj->date_valid);
899 $this->
date = $this->db->jdate($obj->date_expedition);
900 $this->date_expedition = $this->db->jdate($obj->date_expedition);
901 $this->date_shipping = $this->db->jdate($obj->date_expedition);
902 $this->date_delivery = $this->db->jdate($obj->date_delivery);
903 $this->fk_delivery_address = $obj->fk_address;
904 $this->model_pdf = $obj->model_pdf;
905 $this->shipping_method_id = $obj->fk_shipping_method;
906 $this->shipping_method = $obj->shipping_method;
907 $this->tracking_number = $obj->tracking_number;
908 $this->origin = ($obj->origin_type ? $obj->origin_type :
'commande');
909 $this->origin_type = ($obj->origin_type ? $obj->origin_type :
'commande');
910 $this->origin_id = $obj->origin_id;
911 $this->billed = $obj->billed;
912 $this->fk_project = $obj->fk_project;
913 $this->signed_status = $obj->signed_status;
914 $this->trueWeight = $obj->weight;
915 $this->weight_units = $obj->weight_units;
917 $this->trueWidth = $obj->width;
918 $this->width_units = $obj->size_units;
919 $this->trueHeight = $obj->height;
920 $this->height_units = $obj->size_units;
921 $this->trueDepth = $obj->size;
922 $this->depth_units = $obj->size_units;
924 $this->note_public = $obj->note_public;
925 $this->note_private = $obj->note_private;
928 $this->trueSize = $obj->size.
"x".$obj->width.
"x".$obj->height;
929 $this->size_units = $obj->size_units;
932 $this->fk_incoterms = $obj->fk_incoterms;
933 $this->location_incoterms = $obj->location_incoterms;
934 $this->label_incoterms = $obj->label_incoterms;
936 $this->db->free($result);
949 if (!empty($this->multicurrency_code)) {
950 $this->multicurrency_code = $this->thirdparty->multicurrency_code;
952 if (
getDolGlobalString(
'MULTICURRENCY_USE_ORIGIN_TX') && !empty($this->thirdparty->multicurrency_tx)) {
953 $this->multicurrency_tx = $this->thirdparty->multicurrency_tx;
960 if (empty($obj->origin_id)) {
971 dol_syslog(get_class($this).
'::Fetch no expedition found', LOG_ERR);
972 $this->error =
'Shipment with id '.$id.
' not found';
976 $this->error = $this->db->error();
988 public function valid($user, $notrigger = 0)
992 require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
998 dol_syslog(get_class($this).
"::valid not in draft status", LOG_WARNING);
1002 if (!isset($this->socid)) {
1003 dol_syslog(get_class($this).
"::can't valid socid not set", LOG_WARNING);
1007 if (!((!
getDolGlobalString(
'MAIN_USE_ADVANCED_PERMS') && $user->hasRight(
'expedition',
'creer'))
1008 || (
getDolGlobalString(
'MAIN_USE_ADVANCED_PERMS') && $user->hasRight(
'expedition',
'shipping_advance',
'validate')))) {
1009 $this->error =
'Permission denied';
1010 dol_syslog(get_class($this).
"::valid ".$this->error, LOG_ERR);
1019 $soc =
new Societe($this->db);
1020 $soc->fetch($this->socid);
1023 $result = $soc->setAsCustomer();
1026 if (preg_match(
'/^[\(]?PROV/i', $this->
ref) || empty($this->
ref)) {
1027 $numref = $this->getNextNumRef($soc);
1028 } elseif (!empty($this->
ref)) {
1031 $numref =
"EXP".$this->id;
1038 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition SET";
1039 $sql .=
" ref = '".$this->db->escape($numref).
"'";
1040 $sql .=
", fk_statut = 1";
1041 $sql .=
", date_valid = '".$this->db->idate($now).
"'";
1042 $sql .=
", fk_user_valid = ".((int) $user->id);
1043 $sql .=
" WHERE rowid = ".((int) $this->
id);
1045 dol_syslog(get_class($this).
"::valid update expedition", LOG_DEBUG);
1046 $resql = $this->db->query($sql);
1048 $this->error = $this->db->lasterror();
1061 $triggerKey =
'SHIPPING_';
1062 if ($this->origin ==
'commande') {
1063 $triggerKey .=
'ORDER_SHIPMENTONPROCESS';
1065 $triggerKey .= strtoupper($this->origin).
'_SHIPMENTONPROCESS';
1074 if (!$error && !$notrigger) {
1076 $result = $this->call_trigger(
'SHIPPING_VALIDATE', $user);
1084 $this->oldref = $this->ref;
1087 if (preg_match(
'/^[\(]?PROV/i', $this->
ref)) {
1089 $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).
"'";
1090 $sql .=
" WHERE filename LIKE '".$this->db->escape($this->
ref).
"%' AND filepath = 'expedition/sending/".$this->db->escape($this->
ref).
"' and entity = ".((int)
$conf->entity);
1091 $resql = $this->db->query($sql);
1094 $this->error = $this->db->lasterror();
1096 $sql =
'UPDATE '.MAIN_DB_PREFIX.
"ecm_files set filepath = 'expedition/sending/".$this->db->escape($this->newref).
"'";
1097 $sql .=
" WHERE filepath = 'expedition/sending/".$this->db->escape($this->
ref).
"' and entity = ".((int)
$conf->entity);
1098 $resql = $this->db->query($sql);
1101 $this->error = $this->db->lasterror();
1107 $dirsource =
$conf->expedition->dir_output.
'/sending/'.$oldref;
1108 $dirdest =
$conf->expedition->dir_output.
'/sending/'.$newref;
1109 if (!$error && file_exists($dirsource)) {
1110 dol_syslog(get_class($this).
"::valid rename dir ".$dirsource.
" into ".$dirdest);
1112 if (@rename($dirsource, $dirdest)) {
1115 $listoffiles =
dol_dir_list(
$conf->expedition->dir_output.
'/sending/'.$newref,
'files', 1,
'^'.preg_quote($oldref,
'/'));
1116 foreach ($listoffiles as $fileentry) {
1117 $dirsource = $fileentry[
'name'];
1118 $dirdest = preg_replace(
'/^'.preg_quote($oldref,
'/').
'/', $newref, $dirsource);
1119 $dirsource = $fileentry[
'path'].
'/'.$dirsource;
1120 $dirdest = $fileentry[
'path'].
'/'.$dirdest;
1121 @rename($dirsource, $dirdest);
1130 $this->
ref = $numref;
1136 $this->db->commit();
1139 $this->db->rollback();
1153 if (empty($rows) || !is_array($rows)) {
1157 $rowToOrigin = array();
1158 $originToRows = array();
1160 $sql =
"SELECT rowid, fk_elementdet";
1161 $sql .=
" FROM ".$this->db->prefix().
"expeditiondet";
1162 $sql .=
" WHERE fk_expedition = ".((int) $this->
id);
1163 $sql .=
" ORDER BY rang ASC, rowid ASC";
1165 $resql = $this->db->query($sql);
1167 while ($obj = $this->db->fetch_object($resql)) {
1168 $rowid = (int) $obj->rowid;
1169 $originLine = (int) $obj->fk_elementdet;
1170 $rowToOrigin[$rowid] = $originLine;
1172 if (!isset($originToRows[$originLine])) {
1173 $originToRows[$originLine] = array();
1175 $originToRows[$originLine][] = $rowid;
1177 $this->db->free($resql);
1179 parent::line_ajaxorder($rows);
1183 $processedOrigins = array();
1186 foreach ($rows as $rowid) {
1187 $rowid = (int) $rowid;
1188 if (empty($rowid)) {
1192 $originLine = isset($rowToOrigin[$rowid]) ? $rowToOrigin[$rowid] : 0;
1193 if ($originLine > 0 && !empty($processedOrigins[$originLine])) {
1197 $rowidsToUpdate = array($rowid);
1198 if ($originLine > 0 && !empty($originToRows[$originLine])) {
1199 $rowidsToUpdate = $originToRows[$originLine];
1200 $processedOrigins[$originLine] = 1;
1203 foreach ($rowidsToUpdate as $childRowId) {
1222 if ($this->
status == self::STATUS_VALIDATED || $this->
status == self::STATUS_CLOSED) {
1224 include_once DOL_DOCUMENT_ROOT.
'/delivery/class/delivery.class.php';
1225 $delivery =
new Delivery($this->db);
1226 $result = $delivery->create_from_sending($user, $this->
id);
1251 public function addline($entrepot_id, $id, $qty, $array_options = [], $fk_product = 0, $fk_parent = 0)
1255 $num = count($this->lines);
1258 $line->entrepot_id = $entrepot_id;
1259 $line->origin_line_id = $id;
1260 $line->fk_elementdet = $id;
1261 $line->element_type =
'order';
1262 $line->fk_parent = $fk_parent;
1263 $line->fk_product = $fk_product;
1267 $orderline->fetch($id);
1270 $line->rang = $orderline->rang;
1271 $line->product_type = $orderline->product_type;
1272 if (!($line->fk_product > 0)) {
1273 $line->fk_product = $orderline->fk_product;
1276 if (
isModEnabled(
'stock') && !empty($orderline->fk_product)) {
1277 $product =
new Product($this->db);
1278 $product->fetch($orderline->fk_product);
1281 $langs->load(
"errors");
1282 $this->error = $langs->trans(
"ErrorWarehouseRequiredIntoShipmentLine");
1287 $productChildrenNb = 0;
1289 $productChildrenNb = $product->hasFatherOrChild(1);
1291 if ($productChildrenNb > 0) {
1292 $product_stock =
null;
1293 $product->loadStockForVirtualProduct(
'warehouseopen', $line->qty);
1294 if ($entrepot_id > 0) {
1295 if (isset($product->stock_warehouse[$entrepot_id])) {
1296 $product_stock = $product->stock_warehouse[$entrepot_id]->real;
1299 foreach ($product->stock_warehouse as $componentStockWarehouse) {
1300 if ($product_stock ===
null) {
1301 $product_stock = $componentStockWarehouse->real;
1303 $product_stock = min($product_stock, $componentStockWarehouse->real);
1307 if ($product_stock ===
null) {
1312 if ($entrepot_id > 0) {
1313 $product->load_stock(
'warehouseopen');
1314 $product_stock = $product->stock_warehouse[$entrepot_id]->real;
1316 $product_stock = $product->stock_reel;
1320 $product_type = $product->type;
1322 $isavirtualproduct = ($productChildrenNb > 0);
1327 || ($isavirtualproduct && !
getDolGlobalInt(
'STOCK_EXCLUDE_VIRTUAL_PRODUCTS'))
1330 if ($product->stockable_product == Product::ENABLED_STOCK && $product_stock < $qty) {
1331 $langs->load(
"errors");
1332 $this->error = $langs->trans(
'ErrorStockIsNotEnoughToAddProductOnShipment', $product->ref);
1333 $this->errorhidden =
'ErrorStockIsNotEnoughToAddProductOnShipment';
1344 if (
isModEnabled(
'productbatch') && !empty($line->fk_product) && !empty($orderline->product_tobatch)) {
1345 $this->error =
'ADDLINE_WAS_CALLED_INSTEAD_OF_ADDLINEBATCH '.$orderline->id.
' '.$line->fk_product;
1350 if (!
getDolGlobalString(
'MAIN_EXTRAFIELDS_DISABLED') && is_array($array_options) && count($array_options) > 0) {
1351 $line->array_options = $array_options;
1354 $this->lines[$num] = $line;
1372 public function addlinefree($qty, $element_type, $fk_product, $fk_unit, $rang, $description, $fk_parent, $array_options = [])
1374 global
$mysoc, $langs, $user;
1376 if ($this->
status == self::STATUS_DRAFT) {
1387 if ($ranktouse == -1) {
1388 $rangmax = $this->
line_max($fk_parent);
1389 $ranktouse = $rangmax + 1;
1394 $this->line->fk_expedition = $this->id;
1395 $this->line->element_type = $element_type;
1396 $this->line->fk_product = $fk_product;
1397 $this->line->description = $description;
1398 $this->line->desc = $description;
1399 $this->line->fk_parent = $fk_parent;
1400 $this->line->qty = (float) $qty;
1401 $this->line->fk_unit = $fk_unit;
1402 $this->line->rang = $ranktouse;
1404 if (is_array($array_options) && count($array_options) > 0) {
1405 $this->line->array_options = $array_options;
1408 $result = $this->line->insert($user);
1410 if (!isset($this->context[
'createfromclone'])) {
1411 if (!empty($fk_parent)) {
1413 } elseif ($ranktouse > 0 && $ranktouse <= count($this->lines)) {
1414 $linecount = count($this->lines);
1415 for ($ii = $ranktouse; $ii <= $linecount; $ii++) {
1419 $this->lines[] = $this->line;
1422 $this->db->commit();
1423 return $this->line->id;
1425 $this->error = $this->line->error;
1426 dol_syslog(get_class($this).
"::addlinefree error=".$this->error, LOG_ERR);
1427 $this->db->rollback();
1431 dol_syslog(get_class($this).
"::addlinefree status of shipment must be Draft to allow use of ->addlinefree()", LOG_ERR);
1451 public function updatelinefree($rowid, $qty, $element_type, $fk_product, $fk_unit, $rang, $description, $fk_parent, $notrigger, $array_options = array())
1453 global
$mysoc, $langs, $user;
1455 if ($this->
status == self::STATUS_DRAFT) {
1465 $qty = (float) $qty;
1466 $description = trim($description);
1471 $line->fetch($rowid);
1472 $line->fetch_optionals();
1474 if (!empty($line->fk_product)) {
1475 $product =
new Product($this->db);
1476 $result = $product->fetch($line->fk_product);
1477 $product_type = $product->type;
1480 $staticline = clone $line;
1482 $line->oldline = $staticline;
1483 $this->line = $line;
1484 $this->line->context = $this->context;
1485 $this->line->rang = $rang;
1486 $this->line->fk_expedition = $this->id;
1487 $this->line->element_type = $element_type;
1488 $this->line->fk_product = $fk_product;
1489 $this->line->qty = $qty;
1490 $this->line->fk_unit = $fk_unit;
1491 $this->line->fk_parent = $fk_parent;
1492 $this->line->description = $description;
1494 if (is_array($array_options) && count($array_options) > 0) {
1496 foreach ($array_options as $key => $value) {
1497 $this->line->array_options[$key] = $array_options[$key];
1501 $result = $this->line->update($user, $notrigger);
1504 if (!empty($fk_parent)) {
1508 $this->db->commit();
1511 $this->error = $this->line->error;
1513 $this->db->rollback();
1517 $this->error = get_class($this).
"::updatelinefree Shipment status makes operation forbidden";
1518 $this->errors = array(
'ShipmentStatusMakeOperationForbidden');
1532 public function addline_batch($dbatch, $array_options = [], $origin_line =
null)
1537 $num = count($this->lines);
1539 if ($dbatch[
'qty'] > 0 || ($dbatch[
'qty'] == 0 &&
getDolGlobalString(
'SHIPMENT_GETS_ALL_ORDER_PRODUCTS'))) {
1542 foreach ($dbatch[
'detail'] as $key => $value) {
1543 if ($value[
'q'] > 0 || ($value[
'q'] == 0 &&
getDolGlobalString(
'SHIPMENT_GETS_ALL_ORDER_PRODUCTS'))) {
1549 $ret = $linebatch->fetchFromStock($value[
'id_batch']);
1554 $linebatch->qty = $value[
'q'];
1555 if ($linebatch->qty == 0 &&
getDolGlobalString(
'SHIPMENT_GETS_ALL_ORDER_PRODUCTS')) {
1556 $linebatch->batch =
null;
1558 $tab[] = $linebatch;
1561 require_once DOL_DOCUMENT_ROOT.
'/product/class/productbatch.class.php';
1563 $prod_batch->fetch($value[
'id_batch']);
1565 if ($prod_batch->qty < $linebatch->qty) {
1566 $langs->load(
"errors");
1567 $this->errors[] = $langs->trans(
'ErrorStockIsNotEnoughToAddProductOnShipment', $prod_batch->fk_product);
1568 dol_syslog(get_class($this).
"::addline_batch error=Product ".$prod_batch->batch.
": ".$this->errorsToString(), LOG_ERR);
1574 if (is_object($linebatch)) {
1575 $line->entrepot_id = $linebatch->fk_warehouse;
1577 $line->origin_line_id = $dbatch[
'ix_l'];
1578 $line->fk_elementdet = $dbatch[
'ix_l'];
1579 $line->qty = $dbatch[
'qty'];
1580 $line->detail_batch = $tab;
1581 if (!($line->rang > 0)) {
1582 $line->rang = $origin_line->rang;
1584 if (!($line->fk_product > 0)) {
1585 $line->fk_product = $origin_line->fk_product;
1589 if (!
getDolGlobalString(
'MAIN_EXTRAFIELDS_DISABLED') && is_array($array_options) && count($array_options) > 0) {
1590 $line->array_options = $array_options;
1594 $this->lines[$num] = $line;
1607 public function update($user =
null, $notrigger = 0)
1614 if (isset($this->
ref)) {
1615 $this->
ref = trim($this->
ref);
1617 if (isset($this->entity)) {
1618 $this->entity = (int) $this->entity;
1620 if (isset($this->ref_customer)) {
1621 $this->ref_customer = trim($this->ref_customer);
1623 if (isset($this->socid)) {
1624 $this->socid = (int) $this->socid;
1626 if (isset($this->fk_user_author)) {
1627 $this->fk_user_author = (int) $this->fk_user_author;
1629 if (isset($this->fk_user_valid)) {
1630 $this->fk_user_valid = (int) $this->fk_user_valid;
1632 if (isset($this->fk_delivery_address)) {
1633 $this->fk_delivery_address = (int) $this->fk_delivery_address;
1635 if (isset($this->shipping_method_id)) {
1636 $this->shipping_method_id = (int) $this->shipping_method_id;
1638 if (isset($this->tracking_number)) {
1639 $this->tracking_number = trim($this->tracking_number);
1641 if (isset($this->statut)) {
1642 $this->statut = (int) $this->statut;
1644 if (isset($this->
status)) {
1647 if (isset($this->trueDepth)) {
1648 $this->trueDepth = trim($this->trueDepth);
1650 if (isset($this->trueWidth)) {
1651 $this->trueWidth = trim($this->trueWidth);
1653 if (isset($this->trueHeight)) {
1654 $this->trueHeight = trim($this->trueHeight);
1656 if (isset($this->size_units)) {
1657 $this->size_units = trim($this->size_units);
1659 if (isset($this->weight_units)) {
1660 $this->weight_units = (int) $this->weight_units;
1662 if (isset($this->trueWeight)) {
1663 $this->weight = trim((
string) $this->trueWeight);
1665 if (isset($this->note_private)) {
1666 $this->note_private = trim($this->note_private);
1668 if (isset($this->note_public)) {
1669 $this->note_public = trim($this->note_public);
1671 if (isset($this->model_pdf)) {
1672 $this->model_pdf = trim($this->model_pdf);
1674 if (!empty($this->date_expedition)) {
1675 $this->date_shipping = $this->date_expedition;
1682 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition SET";
1683 $sql .=
" ref = ".(isset($this->
ref) ?
"'".$this->db->escape($this->
ref).
"'" :
"null").
",";
1684 $sql .=
" ref_ext = ".(isset($this->ref_ext) ?
"'".$this->db->escape($this->ref_ext).
"'" :
"null").
",";
1685 $sql .=
" ref_customer = ".(isset($this->ref_customer) ?
"'".$this->db->escape($this->ref_customer).
"'" :
"null").
",";
1686 $sql .=
" fk_soc = ".(isset($this->socid) ? ((int) $this->socid) :
"null").
",";
1687 $sql .=
" date_creation = ".(dol_strlen($this->date_creation) != 0 ?
"'".$this->db->idate($this->date_creation).
"'" :
'null').
",";
1688 $sql .=
" fk_user_author = ".(isset($this->fk_user_author) ? ((int) $this->fk_user_author) :
"null").
",";
1689 $sql .=
" date_valid = ".(dol_strlen($this->date_valid) != 0 ?
"'".$this->db->idate($this->date_valid).
"'" :
'null').
",";
1690 $sql .=
" fk_user_valid = ".(isset($this->fk_user_valid) ? ((int) $this->fk_user_valid) :
"null").
",";
1691 $sql .=
" date_expedition = ".(dol_strlen($this->date_shipping) != 0 ?
"'".$this->db->idate($this->date_shipping).
"'" :
'null').
",";
1692 $sql .=
" date_delivery = ".(dol_strlen($this->date_delivery) != 0 ?
"'".$this->db->idate($this->date_delivery).
"'" :
'null').
",";
1693 $sql .=
" fk_address = ".(isset($this->fk_delivery_address) ? ((int) $this->fk_delivery_address) :
"null").
",";
1694 $sql .=
" fk_shipping_method = ".((isset($this->shipping_method_id) && $this->shipping_method_id > 0) ? ((
int) $this->shipping_method_id) :
"null").
",";
1695 $sql .=
" tracking_number = ".(isset($this->tracking_number) ?
"'".$this->db->escape($this->tracking_number).
"'" :
"null").
",";
1696 $sql .=
" fk_statut = ".(isset($this->
status) ? ((int) $this->
status) :
"null").
",";
1697 $sql .=
" fk_projet = ".(isset($this->fk_project) ? ((int) $this->fk_project) :
"null").
",";
1698 $sql .=
" height = ".(($this->trueHeight !=
'') ? ((
float) $this->trueHeight) :
"null").
",";
1699 $sql .=
" width = ".(($this->trueWidth !=
'') ? ((
float) $this->trueWidth) :
"null").
",";
1700 $sql .=
" size_units = ".(isset($this->size_units) ? ((int) $this->size_units) :
"null").
",";
1701 $sql .=
" size = ".(($this->trueDepth !=
'') ? ((
float) $this->trueDepth) :
"null").
",";
1702 $sql .=
" weight_units = ".(isset($this->weight_units) ? ((int) $this->weight_units) :
"null").
",";
1703 $sql .=
" weight = ".(($this->trueWeight !=
'') ? ((
float) $this->trueWeight) :
"null").
",";
1704 $sql .=
" note_private = ".(isset($this->note_private) ?
"'".$this->db->escape($this->note_private).
"'" :
"null").
",";
1705 $sql .=
" note_public = ".(isset($this->note_public) ?
"'".$this->db->escape($this->note_public).
"'" :
"null").
",";
1706 $sql .=
" model_pdf = ".(isset($this->model_pdf) ?
"'".$this->db->escape($this->model_pdf).
"'" :
"null").
",";
1707 $sql .=
" entity = ".((int)
$conf->entity);
1708 $sql .=
" WHERE rowid = ".((int) $this->
id);
1712 dol_syslog(get_class($this).
"::update", LOG_DEBUG);
1713 $resql = $this->db->query($sql);
1716 $this->errors[] =
"Error ".$this->db->lasterror();
1727 if (!$error && !$notrigger) {
1729 $result = $this->call_trigger(
'SHIPPING_MODIFY', $user);
1738 foreach ($this->errors as $errmsg) {
1739 dol_syslog(get_class($this).
"::update ".$errmsg, LOG_ERR);
1740 $this->error .= ($this->error ?
', '.$errmsg : $errmsg);
1742 $this->db->rollback();
1745 $this->db->commit();
1759 public function cancel($user, $notrigger = 0, $also_update_stock =
false)
1761 global
$conf, $langs, $user;
1763 require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
1772 if (isset($this->linkedObjectsIds[
'delivery']) && count($this->linkedObjectsIds[
'delivery']) > 0) {
1773 $this->error =
'ErrorThereIsSomeDeliveries';
1777 if (!$error && !$notrigger) {
1779 $result = $this->call_trigger(
'SHIPPING_CANCEL', $user);
1789 (
getDolGlobalString(
'STOCK_CALCULATE_ON_SHIPMENT_CLOSE') && $this->
status == self::STATUS_CLOSED && $also_update_stock));
1791 require_once DOL_DOCUMENT_ROOT.
"/product/stock/class/mouvementstock.class.php";
1793 $langs->load(
"agenda");
1797 $sql .=
" ed.fk_product";
1798 $sql .=
", ed.qty, ed.fk_entrepot, ed.rowid as expeditiondet_id";
1799 $sql .=
", SUM(".$this->db->ifsql(
"pa.rowid IS NOT NULL",
"1",
"0").
") as iskit";
1800 $sql .=
", ".$this->db->ifsql(
"pai.incdec IS NULL",
"1",
"pai.incdec").
" as incdec";
1801 $sql .=
" FROM ".$this->db->prefix().
"expeditiondet as ed";
1802 $sql .=
" LEFT JOIN ".$this->db->prefix().
"product_association as pa ON pa.fk_product_pere = ed.fk_product";
1803 $sql .=
" LEFT JOIN ".$this->db->prefix().
"expeditiondet as edp ON edp.rowid = ed.fk_parent";
1804 $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";
1805 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
1806 $sql .=
" GROUP BY ed.fk_product, ed.qty, ed.fk_entrepot, ed.rowid, pai.incdec";
1807 $sql .= $this->db->order(
"ed.rowid",
"DESC");
1809 dol_syslog(get_class($this).
"::delete select details", LOG_DEBUG);
1810 $resql = $this->db->query($sql);
1812 $cpt = $this->db->num_rows($resql);
1816 for ($i = 0; $i < $cpt; $i++) {
1817 dol_syslog(get_class($this).
"::delete movement index ".$i);
1818 $obj = $this->db->fetch_object($resql);
1819 $line_id = (int) $obj->expeditiondet_id;
1821 if ($can_update_stock && (empty($obj->iskit) ||
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) && !empty($obj->incdec)) {
1824 $mouvS->origin =
'';
1828 $lotArray = $shipmentlinebatch->fetchAll($obj->expeditiondet_id);
1829 if (!is_array($lotArray)) {
1831 $this->errors[] =
"Error ".$this->db->lasterror();
1835 if (empty($lotArray)) {
1839 $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));
1842 $this->errors = array_merge($this->errors, $mouvS->errors);
1848 foreach ($lotArray as $lot) {
1849 $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));
1852 $this->errors = array_merge($this->errors, $mouvS->errors);
1865 $res = $shipment_line->fetch($line_id);
1867 $result = $shipment_line->delete($user);
1870 $this->errors[] =
"Error ".$shipment_line->errorsToString();
1874 $this->errors[] =
"Error ".$shipment_line->errorsToString();
1884 $this->errors[] =
"Error ".$this->db->lasterror();
1897 $sql =
"SELECT rowid FROM ".$this->db->prefix().
"expedition";
1898 $sql .=
" WHERE rowid = ".((int) $this->
id);
1900 if ($this->db->query($sql)) {
1901 if (!empty($this->origin) && $this->origin_id > 0) {
1904 '@phan-var-force Facture|Commande $origin_object';
1915 $this->db->commit();
1919 if (!empty(
$conf->expedition->dir_output)) {
1920 $dir =
$conf->expedition->dir_output.
'/sending/'.$ref;
1921 $file = $dir.
'/'.$ref.
'.pdf';
1922 if (file_exists($file)) {
1927 if (file_exists($dir)) {
1929 $this->error = $langs->trans(
"ErrorCanNotDeleteDir", $dir);
1937 $this->error = $this->db->lasterror().
" - sql=$sql";
1938 $this->db->rollback();
1942 $this->db->rollback();
1946 $this->db->rollback();
1960 public function delete($user =
null, $notrigger = 0, $also_update_stock =
false)
1962 global
$conf, $langs;
1968 require_once DOL_DOCUMENT_ROOT.
'/core/lib/files.lib.php';
1977 if (isset($this->linkedObjectsIds[
'delivery']) && count($this->linkedObjectsIds[
'delivery']) > 0) {
1978 $this->error =
'ErrorThereIsSomeDeliveries';
1982 if (!$error && !$notrigger) {
1984 $result = $this->call_trigger(
'SHIPPING_DELETE', $user);
1994 (
getDolGlobalString(
'STOCK_CALCULATE_ON_SHIPMENT_CLOSE') && $this->
status == self::STATUS_CLOSED && $also_update_stock));
1996 require_once DOL_DOCUMENT_ROOT.
"/product/stock/class/mouvementstock.class.php";
1998 $langs->load(
"agenda");
2005 $sql .=
" ed.fk_product";
2006 $sql .=
", ed.qty, ed.fk_entrepot, ed.rowid as expeditiondet_id";
2007 $sql .=
", SUM(".$this->db->ifsql(
"pa.rowid IS NOT NULL",
"1",
"0").
") as iskit";
2008 $sql .=
", ".$this->db->ifsql(
"pai.incdec IS NULL",
"1",
"pai.incdec").
" as incdec";
2009 $sql .=
" FROM ".$this->db->prefix().
"expeditiondet as ed";
2010 $sql .=
" LEFT JOIN ".$this->db->prefix().
"product_association as pa ON pa.fk_product_pere = ed.fk_product";
2011 $sql .=
" LEFT JOIN ".$this->db->prefix().
"expeditiondet as edp ON edp.rowid = ed.fk_parent";
2012 $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";
2013 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
2014 $sql .=
" GROUP BY ed.fk_product, ed.qty, ed.fk_entrepot, ed.rowid, pai.incdec";
2015 $sql .= $this->db->order(
"ed.rowid",
"DESC");
2017 dol_syslog(get_class($this).
"::delete select details", LOG_DEBUG);
2018 $resql = $this->db->query($sql);
2020 $cpt = $this->db->num_rows($resql);
2021 for ($i = 0; $i < $cpt; $i++) {
2022 dol_syslog(get_class($this).
"::delete movement index ".$i);
2023 $obj = $this->db->fetch_object($resql);
2024 $line_id = (int) $obj->expeditiondet_id;
2026 if ($can_update_stock && (empty($obj->iskit) ||
getDolGlobalInt(
'PRODUIT_SOUSPRODUITS_ALSO_ENABLE_PARENT_STOCK_MOVE')) && !empty($obj->incdec)) {
2029 $mouvS->origin =
'';
2031 $lotArray = $shipmentlinebatch->fetchAll($line_id);
2032 if (!is_array($lotArray)) {
2034 $this->errors[] =
"Error ".$this->db->lasterror();
2036 if (empty($lotArray)) {
2040 $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));
2043 $this->errors = array_merge($this->errors, $mouvS->errors);
2049 foreach ($lotArray as $lot) {
2050 $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));
2053 $this->errors = array_merge($this->errors, $mouvS->errors);
2066 $res = $shipment_line->fetch($line_id);
2068 $result = $shipment_line->delete($user);
2071 $this->errors[] =
"Error ".$shipment_line->errorsToString();
2075 $this->errors[] =
"Error ".$shipment_line->errorsToString();
2085 $this->errors[] =
"Error ".$this->db->lasterror();
2110 $sql =
"DELETE FROM ".$this->db->prefix().
"expedition";
2111 $sql .=
" WHERE rowid = ".((int) $this->
id);
2113 if ($this->db->query($sql)) {
2114 if (!empty($this->origin) && $this->origin_id > 0) {
2117 '@phan-var-force Facture|Commande $origin_object';
2127 $this->db->commit();
2135 if (!empty(
$conf->expedition->dir_output)) {
2136 $dir =
$conf->expedition->dir_output .
'/sending/' . $ref;
2137 $file = $dir .
'/' . $ref .
'.pdf';
2138 if (file_exists($file)) {
2143 if (file_exists($dir)) {
2145 $this->error = $langs->trans(
"ErrorCanNotDeleteDir", $dir);
2153 $this->error = $this->db->lasterror().
" - sql=$sql";
2154 $this->db->rollback();
2158 $this->db->rollback();
2162 $this->db->rollback();
2178 $this->lines = array();
2183 $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";
2184 $sql .=
", cd.total_ht, cd.total_localtax1, cd.total_localtax2, cd.total_ttc, cd.total_tva";
2185 $sql .=
", cd.fk_remise_except, cd.fk_product_fournisseur_price as fk_fournprice";
2186 $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";
2187 $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";
2188 $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";
2189 $sql .=
", p.ref as product_ref, p.label as product_label, p.fk_product_type, p.barcode as product_barcode";
2190 $sql .=
", p.weight, p.weight_units, p.length, p.length_units, p.width, p.width_units, p.height, p.height_units";
2191 $sql .=
", p.surface, p.surface_units, p.volume, p.volume_units, p.tosell as product_tosell, p.tobuy as product_tobuy";
2192 $sql .=
", p.tobatch as product_tobatch, p.stockable_product";
2193 $sql .=
" FROM ".MAIN_DB_PREFIX.
"expeditiondet as ed, ".MAIN_DB_PREFIX.
"commandedet as cd";
2194 $sql .=
" LEFT JOIN ".MAIN_DB_PREFIX.
"product as p ON p.rowid = cd.fk_product";
2195 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
2196 $sql .=
" AND ed.fk_elementdet = cd.rowid";
2197 $sql .=
" ORDER BY CASE WHEN ed.rang IS NULL OR ed.rang = 0 THEN cd.rang ELSE ed.rang END, ed.rowid";
2199 dol_syslog(get_class($this).
"::fetch_lines", LOG_DEBUG);
2200 $resql = $this->db->query($sql);
2202 include_once DOL_DOCUMENT_ROOT.
'/core/lib/price.lib.php';
2204 $num = $this->db->num_rows($resql);
2210 $this->total_ht = 0;
2211 $this->total_tva = 0;
2212 $this->total_ttc = 0;
2213 $this->total_localtax1 = 0;
2214 $this->total_localtax2 = 0;
2216 $this->multicurrency_total_ht = 0;
2217 $this->multicurrency_total_tva = 0;
2218 $this->multicurrency_total_ttc = 0;
2224 $obj = $this->db->fetch_object($resql);
2226 if ($originline > 0 && $originline == $obj->fk_elementdet) {
2228 $line->entrepot_id = 0;
2229 $line->qty_shipped += $obj->qty_shipped;
2232 $line->entrepot_id = $obj->fk_entrepot;
2233 $line->qty_shipped = $obj->qty_shipped;
2237 $detail_entrepot->entrepot_id = $obj->fk_entrepot;
2238 $detail_entrepot->qty_shipped = $obj->qty_shipped;
2239 $detail_entrepot->line_id = $obj->line_id;
2240 $line->details_entrepot[] = $detail_entrepot;
2242 $line->line_id = $obj->line_id;
2243 $line->rowid = $obj->line_id;
2244 $line->id = $obj->line_id;
2246 $line->fk_origin =
'orderline';
2248 $line->fk_element = $obj->fk_element;
2249 $line->origin_id = $obj->fk_element;
2250 $line->fk_elementdet = $obj->fk_elementdet;
2251 $line->origin_line_id = $obj->fk_elementdet;
2252 $line->element_type = $obj->element_type;
2254 $line->fk_expedition = $this->id;
2256 $line->stockable_product = $obj->stockable_product;
2257 $line->product_type = $obj->product_type;
2258 $line->fk_product = $obj->fk_product;
2259 $line->fk_product_type = $obj->fk_product_type;
2260 $line->ref = $obj->product_ref;
2261 $line->product_ref = $obj->product_ref;
2262 $line->product_label = $obj->product_label;
2263 $line->libelle = $obj->product_label;
2264 $line->product_barcode = $obj->product_barcode;
2265 $line->product_tosell = $obj->product_tosell;
2266 $line->product_tobuy = $obj->product_tobuy;
2267 $line->product_tobatch = $obj->product_tobatch;
2268 $line->fk_fournprice = $obj->fk_fournprice;
2269 $line->label = $obj->custom_label;
2270 $line->description = $obj->description;
2271 $line->qty_asked = $obj->qty_asked;
2272 $line->rang = $obj->rang;
2273 $line->weight = $obj->weight;
2274 $line->weight_units = $obj->weight_units;
2275 $line->length = $obj->length;
2276 $line->length_units = $obj->length_units;
2277 $line->width = $obj->width;
2278 $line->width_units = $obj->width_units;
2279 $line->height = $obj->height;
2280 $line->height_units = $obj->height_units;
2281 $line->surface = $obj->surface;
2282 $line->surface_units = $obj->surface_units;
2283 $line->volume = $obj->volume;
2284 $line->volume_units = $obj->volume_units;
2285 $line->stockable_product = $obj->stockable_product;
2286 $line->fk_unit = $obj->fk_unit;
2288 $line->extraparams = !empty($obj->extraparams) ? (array) json_decode($obj->extraparams,
true) : array();
2290 $line->pa_ht = $obj->pa_ht;
2293 $localtax_array = array(0 => $obj->localtax1_type, 1 => $obj->localtax1_tx, 2 => $obj->localtax2_type, 3 => $obj->localtax2_tx);
2294 $localtax1_tx =
get_localtax($obj->tva_tx, 1, $this->thirdparty);
2295 $localtax2_tx =
get_localtax($obj->tva_tx, 2, $this->thirdparty);
2298 $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);
2299 $line->desc = $obj->description;
2300 $line->qty = $line->qty_shipped;
2301 $line->total_ht = (float) $tabprice[0];
2302 $line->total_localtax1 = (float) $tabprice[9];
2303 $line->total_localtax2 = (float) $tabprice[10];
2304 $line->total_ttc = (float) $tabprice[2];
2305 $line->total_tva = (float) $tabprice[1];
2306 $line->vat_src_code = $obj->vat_src_code;
2307 $line->tva_tx = $obj->tva_tx;
2308 $line->localtax1_tx = $obj->localtax1_tx;
2309 $line->localtax2_tx = $obj->localtax2_tx;
2310 $line->info_bits = $obj->info_bits;
2311 $line->price = $obj->price;
2312 $line->subprice = $obj->subprice;
2313 $line->fk_remise_except = $obj->fk_remise_except;
2314 $line->remise_percent = $obj->remise_percent;
2316 $this->total_ht += $tabprice[0];
2317 $this->total_tva += $tabprice[1];
2318 $this->total_ttc += $tabprice[2];
2319 $this->total_localtax1 += $tabprice[9];
2320 $this->total_localtax2 += $tabprice[10];
2322 $line->date_start = $this->db->jdate($obj->date_start);
2323 $line->date_end = $this->db->jdate($obj->date_end);
2325 $line->special_code = $obj->special_code;
2328 $this->fk_multicurrency = $obj->fk_multicurrency;
2329 $this->multicurrency_code = $obj->multicurrency_code;
2330 $line->multicurrency_subprice = $obj->multicurrency_subprice;
2331 $line->multicurrency_total_ht = $obj->multicurrency_total_ht;
2332 $line->multicurrency_total_tva = $obj->multicurrency_total_tva;
2333 $line->multicurrency_total_ttc = $obj->multicurrency_total_ttc;
2335 $this->multicurrency_total_ht += $obj->multicurrency_total_ht;
2336 $this->multicurrency_total_tva += $obj->multicurrency_total_tva;
2337 $this->multicurrency_total_ttc += $obj->multicurrency_total_ttc;
2340 if (
isModEnabled(
'productbatch') && $obj->line_id > 0 && $obj->product_tobatch > 0) {
2341 if ($originline != $obj->fk_elementdet) {
2342 $line->detail_batch = array();
2344 $newdetailbatch = $shipmentlinebatch->fetchAll($obj->line_id, $obj->fk_product);
2346 if (is_array($newdetailbatch)) {
2347 if ($originline != $obj->fk_elementdet) {
2348 $line->detail_batch = $newdetailbatch;
2350 $line->detail_batch = array_merge($line->detail_batch, $newdetailbatch);
2357 $detail_children = array();
2358 $line_child_list = array();
2359 $res = $line->findAllChild($line->id, $line_child_list, 1);
2361 foreach ($line_child_list as $child_line) {
2362 foreach ($child_line as $child_obj) {
2363 $child_product_id = (int) $child_obj->fk_product;
2364 $child_warehouse_id = (int) $child_obj->fk_warehouse;
2366 if ($child_warehouse_id > 0) {
2368 if (!isset($detail_children[$child_product_id])) {
2369 $detail_children[$child_product_id] = array();
2371 if (!isset($detail_children[$child_product_id][$child_warehouse_id])) {
2372 $detail_children[$child_product_id][$child_warehouse_id] = 0;
2374 $detail_children[$child_product_id][$child_warehouse_id] += $child_obj->qty;
2379 $line->detail_children = $detail_children;
2382 $line->fetch_optionals();
2384 if ($originline != $obj->fk_elementdet) {
2385 $this->lines[$lineindex] = $line;
2388 $line->total_ht += $tabprice[0];
2389 $line->total_localtax1 += $tabprice[9];
2390 $line->total_localtax2 += $tabprice[10];
2391 $line->total_ttc += $tabprice[2];
2392 $line->total_tva += $tabprice[1];
2396 $originline = $obj->fk_elementdet;
2398 $this->db->free($resql);
2401 $this->error = $this->db->error();
2418 $this->lines = array();
2420 $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';
2421 $sql .=
' FROM '.MAIN_DB_PREFIX.$this->table_element_line.
' as ed';
2422 $sql .=
' LEFT JOIN '.MAIN_DB_PREFIX.
'product as p ON (p.rowid = ed.fk_product)';
2423 $sql .=
' WHERE ed.fk_expedition = '.((int) $this->
id);
2424 $sql .=
' ORDER BY ed.rang, ed.rowid';
2426 dol_syslog(get_class($this).
"::fetch_lines_free", LOG_DEBUG);
2427 $result = $this->db->query($sql);
2429 $num = $this->db->num_rows($result);
2433 $objp = $this->db->fetch_object($result);
2437 $line->rowid = $objp->rowid;
2438 $line->id = $objp->rowid;
2439 $line->fk_expedition = $this->id;
2440 $line->description = $objp->description;
2441 $line->qty = $objp->qty;
2442 $line->fk_entrepot = $objp->fk_entrepot;
2443 $line->fk_product = $objp->fk_product;
2444 $line->rang = $objp->rang;
2445 $line->fk_element = $objp->fk_element;
2446 $line->fk_unit = $objp->fk_unit;
2447 $line->fk_elementdet = $objp->fk_elementdet;
2448 $line->fk_element_type = $objp->element_type;
2449 $line->fetch_optionals();
2451 $this->lines[$i] = $line;
2456 $this->db->free($result);
2460 $this->error = $this->db->error();
2486 if ($this->
status == self::STATUS_DRAFT) {
2492 $line->fetch($lineid);
2494 if ($line->delete($user) > 0) {
2497 $this->db->commit();
2500 $this->db->rollback();
2504 $this->error =
'ErrorDeleteLineNotAllowedByObjectStatus';
2518 global
$conf, $langs;
2520 $langs->load(
'sendings');
2522 $nofetch = !empty($params[
'nofetch']);
2525 $datas[
'picto'] =
img_picto(
'', $this->picto).
' <u class="paddingrightonly">'.$langs->trans(
"Shipment").
'</u>';
2526 if (isset($this->
status)) {
2527 $datas[
'picto'] .=
' '.$this->getLibStatut(5);
2529 $datas[
'ref'] =
'<br><b>'.$langs->trans(
'Ref').
':</b> '.$this->ref;
2530 $datas[
'refcustomer'] =
'<br><b>'.$langs->trans(
'RefCustomer').
':</b> '.($this->ref_customer ? $this->ref_customer : $this->ref_client);
2532 $langs->load(
'companies');
2533 if (empty($this->thirdparty)) {
2536 $datas[
'customer'] =
'<br><b>'.$langs->trans(
'Customer').
':</b> '.$this->thirdparty->getNomUrl(1,
'', 0, 1);
2553 public function getNomUrl($withpicto = 0, $option =
'', $max = 0, $short = 0, $notooltip = 0, $save_lastsearch_value = -1)
2555 global $langs, $hookmanager;
2560 'objecttype' => $this->element,
2561 'option' => $option,
2564 $classfortooltip =
'classfortooltip';
2567 $classfortooltip =
'classforajaxtooltip';
2568 $dataparams =
' data-params="'.dol_escape_htmltag(json_encode($params)).
'"';
2574 $url = DOL_URL_ROOT.
'/expedition/card.php?id='.$this->id;
2580 if ($option !==
'nolink') {
2582 $add_save_lastsearch_values = ($save_lastsearch_value == 1 ? 1 : 0);
2583 if ($save_lastsearch_value == -1 && isset($_SERVER[
"PHP_SELF"]) && preg_match(
'/list\.php/', $_SERVER[
"PHP_SELF"])) {
2584 $add_save_lastsearch_values = 1;
2586 if ($add_save_lastsearch_values) {
2587 $url .=
'&save_lastsearch_values=1';
2592 if (empty($notooltip)) {
2594 $label = $langs->trans(
"Shipment");
2595 $linkclose .=
' alt="'.dolPrintHTMLForAttribute($label).
'"';
2597 $linkclose .= ($label ?
' title="'.dolPrintHTMLForAttribute($label).
'"' :
' title="tocomplete"');
2598 $linkclose .= $dataparams.
' class="'.$classfortooltip.
'"';
2601 $linkstart =
'<a href="'.$url.
'"';
2602 $linkstart .= $linkclose.
'>';
2605 $result .= $linkstart;
2607 $result .=
img_object(($notooltip ?
'' : $label), ($this->picto ? $this->picto :
'generic'), ($notooltip ? (($withpicto != 2) ?
'class="paddingright"' :
'') :
'class="'.(($withpicto != 2) ?
'paddingright ' :
'').
'"'), 0, 0, $notooltip ? 0 : 1);
2609 if ($withpicto != 2) {
2610 $result .= $this->ref;
2612 $result .= $linkend;
2614 $hookmanager->initHooks(array($this->element .
'dao'));
2615 $parameters = array(
'id' => $this->
id,
'getnomurl' => &$result);
2616 $reshook = $hookmanager->executeHooks(
'getNomUrl', $parameters, $this, $action);
2618 $result = $hookmanager->resPrint;
2620 $result .= $hookmanager->resPrint;
2649 $labelStatus = $langs->transnoentitiesnoconv($this->labelStatus[$status]);
2650 $labelStatusShort = $langs->transnoentitiesnoconv($this->labelStatusShort[$status]);
2652 $statusType =
'status'.$status;
2653 if ($status == self::STATUS_VALIDATED) {
2654 $statusType =
'status4';
2656 if ($status == self::STATUS_CLOSED) {
2657 $statusType =
'status6';
2659 if ($status == self::STATUS_CANCELED) {
2660 $statusType =
'status9';
2664 $status_label = $this->signed_status ? $labelStatus . $signed_label : $labelStatus;
2665 $status_label_short = $this->signed_status ? $labelStatusShort . $signed_label : $labelStatusShort;
2667 return dolGetStatus($status_label, $status_label_short,
'', $statusType, $mode);
2679 global $langs,
$conf;
2681 $selected = (empty($arraydata[
'selected']) ? 0 : $arraydata[
'selected']);
2683 $return =
'<div class="box-flex-item box-flex-grow-zero">';
2684 $return .=
'<div class="info-box info-box-sm">';
2685 $return .=
'<div class="info-box-icon bg-infobox-action">';
2687 $return .=
'</div>';
2688 $return .=
'<div class="info-box-content">';
2689 $return .=
'<span class="info-box-ref inline-block tdoverflowmax150 valignmiddle">' . $this->
getNomUrl() .
'</span>';
2690 if ($selected >= 0) {
2691 $return .=
'<input id="cb'.$this->id.
'" class="flat checkforselect fright" type="checkbox" name="toselect[]" value="'.$this->
id.
'"'.($selected ?
' checked="checked"' :
'').
'>';
2693 $return .=
'<br><div class="info-box-ref tdoverflowmax150">'.$this->thirdparty->getNomUrl(1).
'</div>';
2694 $return .=
'<div class="info-box-ref amount">'.price($this->total_ht, 0, $langs, 0, -1, -1,
getDolCurrency()).
' '.$langs->trans(
'HT').
'</div>';
2695 $return .=
'<div class="info-box-status">'.$this->getLibStatut(3).
'</div>';
2696 $return .=
'</div>';
2697 $return .=
'</div>';
2698 $return .=
'</div>';
2716 dol_syslog(get_class($this).
"::initAsSpecimen");
2719 $order->initAsSpecimen();
2723 $this->
ref =
'SPECIMEN';
2724 $this->specimen = 1;
2726 $this->livraison_id = 0;
2728 $this->date_creation = $now;
2729 $this->date_valid = $now;
2730 $this->date_delivery = $now + 24 * 3600;
2731 $this->date_expedition = $now + 24 * 3600;
2733 $this->entrepot_id = 0;
2734 $this->fk_delivery_address = 0;
2737 $this->commande_id = 0;
2738 $this->commande = $order;
2740 $this->origin_id = 1;
2741 $this->origin_type =
'commande';
2743 $this->note_private =
'Private note';
2744 $this->note_public =
'Public note';
2748 while ($xnbp < $nbp) {
2750 $line->product_desc = $langs->trans(
"Description").
" ".$xnbp;
2751 $line->product_label = $langs->trans(
"Description").
" ".$xnbp;
2753 $line->qty_asked = 5;
2754 $line->qty_shipped = 4;
2755 $line->fk_product = $this->commande->lines[$xnbp]->fk_product;
2757 $line->weight = 1.123456;
2758 $line->weight_units = 0;
2760 $line->volume = 2.34567;
2761 $line->volume_unit = 0;
2763 $this->lines[] = $line;
2794 if ($user->hasRight(
'expedition',
'creer')) {
2795 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition";
2796 $sql .=
" SET date_delivery = ".($delivery_date ?
"'".$this->db->idate($delivery_date).
"'" :
'null');
2797 $sql .=
" WHERE rowid = ".((int) $this->
id);
2799 dol_syslog(get_class($this).
"::setDeliveryDate", LOG_DEBUG);
2800 $resql = $this->db->query($sql);
2802 $this->date_delivery = $delivery_date;
2805 $this->error = $this->db->error();
2822 if ($user->hasRight(
'expedition',
'creer')) {
2823 $sql =
"UPDATE ".MAIN_DB_PREFIX.
"expedition";
2824 $sql .=
" SET date_expedition = ".($shipping_date ?
"'".$this->db->idate($shipping_date).
"'" :
'null');
2825 $sql .=
" WHERE rowid = ".((int) $this->
id);
2827 dol_syslog(get_class($this).
"::setShippingDate", LOG_DEBUG);
2828 $resql = $this->db->query($sql);
2830 $this->date_shipping = $shipping_date;
2833 $this->error = $this->db->error();
2853 $sql =
"SELECT em.rowid, em.code, em.libelle as label";
2854 $sql .=
" FROM ".MAIN_DB_PREFIX.
"c_shipment_mode as em";
2855 $sql .=
" WHERE em.active = 1";
2856 $sql .=
" ORDER BY em.libelle ASC";
2858 $resql = $this->db->query($sql);
2860 while ($obj = $this->db->fetch_object($resql)) {
2861 $label = $langs->trans(
'SendingMethod'.$obj->code);
2862 $this->meths[(int) $obj->rowid] = ($label !=
'SendingMethod'.$obj->code ? $label : $obj->label);
2879 $this->listmeths = [];
2882 $sql =
"SELECT em.rowid, em.code, em.libelle as label, em.description, em.tracking, em.active";
2883 $sql .=
" FROM ".MAIN_DB_PREFIX.
"c_shipment_mode as em";
2885 $sql .=
" WHERE em.rowid=".((int) $id);
2888 $resql = $this->db->query($sql);
2890 while ($obj = $this->db->fetch_object($resql)) {
2891 $label = $langs->trans(
'SendingMethod'.$obj->code);
2892 $this->listmeths[$i] = [
2893 'rowid' => (int) $obj->rowid,
2894 'code' => $obj->code,
2895 'libelle' => ($label !=
'SendingMethod'.$obj->code ? $label : $obj->label),
2896 'description' => $obj->description,
2897 'tracking' => $obj->tracking,
2898 'active' => (
int) $obj->active,
2914 if (!empty($this->shipping_method_id)) {
2915 $sql =
"SELECT em.code, em.tracking";
2916 $sql .=
" FROM ".MAIN_DB_PREFIX.
"c_shipment_mode as em";
2917 $sql .=
" WHERE em.rowid = ".((int) $this->shipping_method_id);
2919 $resql = $this->db->query($sql);
2921 if ($obj = $this->db->fetch_object($resql)) {
2922 $tracking = (
string) $obj->tracking;
2927 if (!empty($tracking) && !empty($value)) {
2928 $url = str_replace(
'{TRACKID}', $value, $tracking);
2929 $this->tracking_url = sprintf(
'<a target="_blank" rel="noopener noreferrer" href="%s">%s</a>', $url, $value);
2931 $this->tracking_url = $value;
2947 if ($this->
status == self::STATUS_CLOSED) {
2952 if ($this->
status == self::STATUS_DRAFT) {
2953 $this->
valid($user);
2959 if (empty($this->date_shipping)) {
2960 $sql .=
", date_expedition = '".$this->db->escape($this->db->idate(
dol_now())).
"'";
2962 $sql .=
" WHERE rowid = ".((int) $this->
id).
" AND fk_statut > 0";
2964 $resql = $this->db->query($sql);
2967 if ($this->origin_type ==
'commande' && $this->origin_id > 0) {
2969 $order->fetch($this->origin_id);
2971 $order->loadExpeditions(self::STATUS_CLOSED);
2973 $shipments_match_order = 1;
2974 foreach ($order->lines as $line) {
2975 $lineid = $line->id;
2977 if (($line->product_type == 0 ||
getDolGlobalString(
'STOCK_SUPPORTS_SERVICES')) && $order->expeditions[$lineid] != $qty) {
2978 $shipments_match_order = 0;
2979 $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';
2984 if ($shipments_match_order) {
2985 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');
2987 $order->cloture($user);
3005 $result = $this->call_trigger(
'SHIPPING_CLOSED', $user);
3016 $this->db->commit();
3022 $this->db->rollback();
3041 require_once DOL_DOCUMENT_ROOT .
'/product/stock/class/mouvementstock.class.php';
3043 $langs->load(
"agenda");
3047 $sql .=
" ed.rowid as edid, ed.fk_product, ed.qty, ed.fk_entrepot";
3048 $sql .=
", cd.rowid as cdid";
3049 $sql .=
", cd.subprice";
3050 $sql .=
", edb.rowid as edbrowid, edb.eatby, edb.sellby, edb.batch, edb.qty as edbqty, edb.fk_origin_stock";
3052 $sql .=
" FROM " . $this->db->prefix() .
"expeditiondet as ed";
3053 $sql .=
" LEFT JOIN " . $this->db->prefix() .
"commandedet as cd ON cd.rowid = ed.fk_elementdet";
3054 $sql .=
" LEFT JOIN " . $this->db->prefix() .
"expeditiondet_batch as edb on edb.fk_expeditiondet = ed.rowid";
3055 $sql .=
" INNER JOIN " . $this->db->prefix() .
"expedition as e ON ed.fk_expedition = e.rowid";
3056 $sql .=
" WHERE ed.fk_expedition = " . ((int) $this->
id);
3059 dol_syslog(get_class($this) .
"::valid select details", LOG_DEBUG);
3060 $resql = $this->db->query($sql);
3062 $cpt = $this->db->num_rows($resql);
3063 for ($i = 0; $i < $cpt; $i++) {
3064 $obj = $this->db->fetch_object($resql);
3065 if (empty($obj->edbrowid)) {
3068 $qty = $obj->edbqty;
3070 if ($qty <= 0 || ($qty < 0 && !
getDolGlobalInt(
'SHIPMENT_ALLOW_NEGATIVE_QTY'))) {
3073 dol_syslog(get_class($this) .
"::valid movement index " . $i .
" ed.rowid=" . $obj->edid .
" edb.rowid=" . $obj->edbrowid);
3076 $mouvS->origin = &$this;
3077 $mouvS->setOrigin($this->element, $this->
id, $obj->cdid, $obj->edid);
3079 if (empty($obj->edbrowid)) {
3083 $result = $mouvS->livraison($user, $obj->fk_product, $obj->fk_entrepot, $qty, $obj->subprice, $langs->trans($labelmovement, $obj->ref));
3093 $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);
3103 $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)";
3104 $resqldelete = $this->db->query($sqldelete);
3108 $this->error = $this->db->lasterror();
3109 $this->errors[] = $this->db->lasterror();
3132 $sql =
'UPDATE '.MAIN_DB_PREFIX.
'expedition SET billed = 1';
3133 $sql .=
" WHERE rowid = ".((int) $this->
id).
' AND fk_statut > 0';
3135 $resql = $this->db->query($sql);
3140 $result = $this->call_trigger(
'SHIPPING_BILLED', $user);
3147 $this->errors[] = $this->db->lasterror;
3150 if (empty($error)) {
3151 $this->db->commit();
3154 $this->db->rollback();
3169 if ($this->
status <= self::STATUS_DRAFT) {
3173 return $this->
setStatusCommon($user, self::STATUS_DRAFT, $notrigger,
'SHIPMENT_UNVALIDATE');
3183 global $langs, $user;
3188 if ($this->
status == self::STATUS_VALIDATED) {
3194 $oldbilled = $this->billed;
3196 $sql =
'UPDATE '.MAIN_DB_PREFIX.
'expedition SET fk_statut = 1';
3197 $sql .=
" WHERE rowid = ".((int) $this->
id).
' AND fk_statut > 0';
3199 $resql = $this->db->query($sql);
3207 require_once DOL_DOCUMENT_ROOT.
'/product/stock/class/mouvementstock.class.php';
3209 $langs->load(
"agenda");
3213 $sql =
"SELECT cd.fk_product, cd.subprice,";
3214 $sql .=
" ed.rowid, ed.qty, ed.fk_entrepot,";
3215 $sql .=
" edb.rowid as edbrowid, edb.eatby, edb.sellby, edb.batch, edb.qty as edbqty, edb.fk_origin_stock";
3216 $sql .=
" FROM ".MAIN_DB_PREFIX.
"commandedet as cd,";
3217 $sql .=
" ".MAIN_DB_PREFIX.
"expeditiondet as ed";
3218 $sql .=
" LEFT JOIN ".MAIN_DB_PREFIX.
"expeditiondet_batch as edb on edb.fk_expeditiondet = ed.rowid";
3219 $sql .=
" WHERE ed.fk_expedition = ".((int) $this->
id);
3220 $sql .=
" AND cd.rowid = ed.fk_elementdet";
3222 dol_syslog(get_class($this).
"::valid select details", LOG_DEBUG);
3223 $resql = $this->db->query($sql);
3225 $cpt = $this->db->num_rows($resql);
3226 for ($i = 0; $i < $cpt; $i++) {
3227 $obj = $this->db->fetch_object($resql);
3228 if (empty($obj->edbrowid)) {
3231 $qty = $obj->edbqty;
3236 dol_syslog(get_class($this).
"::reopen expedition movement index ".$i.
" ed.rowid=".$obj->rowid.
" edb.rowid=".$obj->edbrowid);
3240 $mouvS->origin = &$this;
3241 $mouvS->setOrigin($this->element, $this->
id);
3243 if (empty($obj->edbrowid)) {
3247 $result = $mouvS->livraison($user, $obj->fk_product, $obj->fk_entrepot, -$qty, $obj->subprice, $langs->trans(
"ShipmentUnClassifyCloseddInDolibarr", $this->ref));
3257 $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);
3266 $this->error = $this->db->lasterror();
3273 $result = $this->call_trigger(
'SHIPPING_REOPEN', $user);
3280 $this->errors[] = $this->db->lasterror();
3284 $this->db->commit();
3289 $this->billed = $oldbilled;
3290 $this->db->rollback();
3306 public function generateDocument($modele, $outputlangs, $hidedetails = 0, $hidedesc = 0, $hideref = 0, $moreparams =
null)
3308 $outputlangs->load(
"products");
3313 if (!empty($this->model_pdf)) {
3314 $modele = $this->model_pdf;
3320 $modelpath =
"core/modules/expedition/doc/";
3324 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.
line_ajaxorder($rows)
Override to keep every expeditiondet row that belongs to the displayed line in sync when drag & drop ...
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', ...)
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.
print $langs trans("Show") . '< td style="' . $timeColor . '" align="center"> s</td > badge status0 badge status4 badge status3 Error badge status8< td align="center">< span class="badge ' . $badge . '"></span ></td >< td align="center">< a href="#" class="button button-small" onclick="openLogModal(this)" data-req="' . dol_escape_htmltag($reqSafe) . '" data-res="' . dol_escape_htmltag($resSafe) . '" data-err="' . dol_escape_htmltag($errSafe) . '">< span class="fa fa-search-plus"></span ></a ></td ></tr >< tr >< td colspan="' . $colspan . '" class="opacitymedium"></td ></tr ></table ></div ></form > logModal none logModal none s a JSON string
buildzip.php
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.
print $langs trans('Date')." left Ref Label right Qty right Price right TotalHT right TotalTTC right right right right right right right right right centpercent right TotalHT right n right VAT right n right TotalVAT right n No sujeto a RE IRPF right TotalLT1 right n right TotalLT2 right n right TotalTTC right n takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency takeposcustomercurrency right TotalTTC takeposcustomercurrency right takeposcustomercurrency n right Paid right PaymentTypeShortLIQ right SELECT p pos_change as p datep as date
$conf db user
Active Directory does not allow anonymous connections.