* Copyright (C) 2023 Christian Humpel * Copyright (C) 2023 Vincent de Grandpré * Copyright (C) 2024-2025 Frédéric France * Copyright (C) 2024-2025 MDW * Copyright (C) 2024 Alexandre Spangaro * Copyright (C) 2025 Noé Cendrier * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ /** * \file mo_production.php * \ingroup mrp * \brief Page to make production on a MO */ // Load Dolibarr environment require '../main.inc.php'; require_once DOL_DOCUMENT_ROOT.'/bom/class/bom.class.php'; require_once DOL_DOCUMENT_ROOT.'/core/class/html.formcompany.class.php'; require_once DOL_DOCUMENT_ROOT.'/core/class/html.formfile.class.php'; require_once DOL_DOCUMENT_ROOT.'/core/class/html.formprojet.class.php'; require_once DOL_DOCUMENT_ROOT.'/mrp/class/mo.class.php'; require_once DOL_DOCUMENT_ROOT.'/mrp/lib/mrp_mo.lib.php'; require_once DOL_DOCUMENT_ROOT.'/projet/class/project.class.php'; require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php'; require_once DOL_DOCUMENT_ROOT.'/product/class/html.formproduct.class.php'; require_once DOL_DOCUMENT_ROOT.'/product/stock/class/entrepot.class.php'; require_once DOL_DOCUMENT_ROOT.'/product/stock/class/productlot.class.php'; require_once DOL_DOCUMENT_ROOT.'/product/stock/class/mouvementstock.class.php'; require_once DOL_DOCUMENT_ROOT.'/workstation/class/workstation.class.php'; require_once DOL_DOCUMENT_ROOT.'/mrp/class/mooptracking.class.php'; /** * @var Conf $conf * @var DoliDB $db * @var HookManager $hookmanager * @var Translate $langs * @var User $user */ // Load translation files required by the page $langs->loadLangs(array("mrp", "stocks", "other", "product", "productbatch")); // Get parameters $id = GETPOSTINT('id'); $ref = GETPOST('ref', 'alpha'); $action = GETPOST('action', 'aZ09'); $confirm = GETPOST('confirm', 'alpha'); $cancel = GETPOST('cancel'); $contextpage = GETPOST('contextpage', 'aZ') ? GETPOST('contextpage', 'aZ') : 'mocard'; // To manage different context of search $backtopage = GETPOST('backtopage', 'alpha'); $lineid = GETPOSTINT('lineid'); $fk_movement = GETPOSTINT('fk_movement'); $fk_default_warehouse = GETPOSTINT('fk_default_warehouse'); $collapse = GETPOST('collapse', 'aZ09comma'); // Initialize a technical objects $object = new Mo($db); $extrafields = new ExtraFields($db); $diroutputmassaction = $conf->mrp->dir_output.'/temp/massgeneration/'.$user->id; $objectline = new MoLine($db); $hookmanager->initHooks(array('moproduction', 'globalcard')); // Note that conf->hooks_modules contains array // Fetch optionals attributes and labels $extrafields->fetch_name_optionals_label($object->table_element); $search_array_options = $extrafields->getOptionalsFromPost($object->table_element, '', 'search_'); // Initialize array of search criteria $search_all = GETPOST("search_all", 'alpha'); $search = array(); foreach ($object->fields as $key => $val) { if (GETPOST('search_'.$key, 'alpha')) { $search[$key] = GETPOST('search_'.$key, 'alpha'); } } if (empty($action) && empty($id) && empty($ref)) { $action = 'view'; } // Load object include DOL_DOCUMENT_ROOT.'/core/actions_fetchobject.inc.php'; // Must be 'include', not 'include_once'. // Security check - Protection if external user //if ($user->socid > 0) accessforbidden(); //if ($user->socid > 0) $socid = $user->socid; $isdraft = (($object->status == $object::STATUS_DRAFT) ? 1 : 0); $result = restrictedArea($user, 'mrp', $object->id, 'mrp_mo', '', 'fk_soc', 'rowid', $isdraft); // Permissions $permissionnote = $user->hasRight('mrp', 'write'); // Used by the include of actions_setnotes.inc.php $permissiondellink = $user->hasRight('mrp', 'write'); // Used by the include of actions_dellink.inc.php $permissiontoadd = $user->hasRight('mrp', 'write'); // Used by the include of actions_addupdatedelete.inc.php and actions_lineupdown.inc.php $permissiontodelete = $user->hasRight('mrp', 'delete') || ($permissiontoadd && isset($object->status) && $object->status == $object::STATUS_DRAFT); $permissiontoproduce = $permissiontoadd; $permissiontoupdatecost = $user->hasRight('bom', 'read'); // User who can define cost must have knowledge of pricing $upload_dir = $conf->mrp->multidir_output[isset($object->entity) ? $object->entity : 1]; /* * Actions */ $parameters = array(); $reshook = $hookmanager->executeHooks('doActions', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks if ($reshook < 0) { setEventMessages($hookmanager->error, $hookmanager->errors, 'errors'); } if (empty($reshook)) { $error = 0; $backurlforlist = DOL_URL_ROOT.'/mrp/mo_list.php'; if (empty($backtopage) || ($cancel && empty($id))) { //var_dump($backurlforlist);exit; if (empty($id) && (($action != 'add' && $action != 'create') || $cancel)) { $backtopage = $backurlforlist; } else { $backtopage = DOL_URL_ROOT.'/mrp/mo_production.php?id='.($id > 0 ? $id : '__ID__'); } } $triggermodname = 'MO_MODIFY'; // Name of trigger action code to execute when we modify record if ($action == 'confirm_cancel' && $confirm == 'yes' && !empty($permissiontoadd)) { $also_cancel_consumed_and_produced_lines = (GETPOST('alsoCancelConsumedAndProducedLines', 'alpha') ? 1 : 0); $result = $object->cancel($user, 0, (bool) $also_cancel_consumed_and_produced_lines); if ($result > 0) { // Purge custom operation progress rows for this (now-cancelled) MO. $db->query("DELETE FROM ".MAIN_DB_PREFIX."mrp_mo_optracking WHERE fk_mo = ".(int)$object->id); header("Location: " . DOL_URL_ROOT.'/mrp/mo_card.php?id=' . $object->id); exit; } else { $action = ''; setEventMessages($object->error, $object->errors, 'errors'); } } elseif ($action == 'confirm_delete' && $confirm == 'yes' && !empty($permissiontodelete)) { $also_cancel_consumed_and_produced_lines = (GETPOST('alsoCancelConsumedAndProducedLines', 'alpha') ? 1 : 0); $result = $object->delete($user, 0, (bool) $also_cancel_consumed_and_produced_lines); if ($result > 0) { header("Location: " . $backurlforlist); exit; } else { $action = ''; setEventMessages($object->error, $object->errors, 'errors'); } } // Actions cancel, add, update, delete or clone include DOL_DOCUMENT_ROOT.'/core/actions_addupdatedelete.inc.php'; // Actions when linking object each other include DOL_DOCUMENT_ROOT.'/core/actions_dellink.inc.php'; // Actions when printing a doc from card include DOL_DOCUMENT_ROOT.'/core/actions_printing.inc.php'; // Actions to send emails $triggersendname = 'MO_SENTBYMAIL'; $autocopy = 'MAIN_MAIL_AUTOCOPY_MO_TO'; $trackid = 'mo'.$object->id; include DOL_DOCUMENT_ROOT.'/core/actions_sendmails.inc.php'; // Action to move up and down lines of object //include DOL_DOCUMENT_ROOT.'/core/actions_lineupdown.inc.php'; // Must be 'include', not 'include_once' if ($action == 'set_thirdparty' && $permissiontoadd) { $object->setValueFrom('fk_soc', GETPOSTINT('fk_soc'), '', null, 'date', '', $user, $triggermodname); } if ($action == 'classin' && $permissiontoadd) { $object->setProject(GETPOSTINT('projectid')); } if ($action == 'confirm_reopen' && $permissiontoadd) { $result = $object->setStatut($object::STATUS_INPROGRESS, 0, '', 'MRP_REOPEN'); } if (($action == 'confirm_addconsumeline' && GETPOST('addconsumelinebutton') && $permissiontoadd) || ($action == 'confirm_addproduceline' && GETPOST('addproducelinebutton') && $permissiontoadd)) { $moline = new MoLine($db); // Line to produce $moline->fk_mo = $object->id; $moline->qty = GETPOSTFLOAT('qtytoadd'); $moline->fk_product = GETPOSTINT('productidtoadd'); if (GETPOST('addconsumelinebutton')) { $moline->role = 'toconsume'; } else { $moline->role = 'toproduce'; } $moline->origin_type = 'free'; // free consume line $moline->position = 0; // Is it a product or a service ? if (!empty($moline->fk_product)) { $tmpproduct = new Product($db); $tmpproduct->fetch($moline->fk_product); if ($tmpproduct->type == Product::TYPE_SERVICE) { $moline->fk_default_workstation = $tmpproduct->fk_default_workstation; $moline->disable_stock_change = 1; if ($tmpproduct->duration_unit) { $moline->qty = $tmpproduct->duration_value; include_once DOL_DOCUMENT_ROOT.'/core/class/cunits.class.php'; $cunits = new CUnits($db); $res = $cunits->fetch(0, '', $tmpproduct->duration_unit, 'time'); if ($res > 0) { $moline->fk_unit = $cunits->id; } } } else { $moline->disable_stock_change = 0; if (getDolGlobalInt('PRODUCT_USE_UNITS')) { $moline->fk_unit = $tmpproduct->fk_unit; } } } // Extrafields $extralabelsline = $extrafields->fetch_name_optionals_label($object->table_element_line); $array_options = $extrafields->getOptionalsFromPost($object->table_element_line); // Unset extrafield if (is_array($extralabelsline)) { // Get extra fields foreach ($extralabelsline as $key => $value) { unset($_POST["options_".$key]); } } if (is_array($array_options) && count($array_options) > 0) { $moline->array_options = $array_options; } $resultline = $moline->create($user, 0); // Never use triggers here if ($resultline <= 0) { $error++; setEventMessages($moline->error, $moline->errors, 'errors'); } $action = ''; // Redirect to refresh the tab information header("Location: ".$_SERVER["PHP_SELF"].'?id='.$object->id); exit; } // Compute operation-complete flag once here so both the backend guards and the // View section below share the same fresh result. $object->lines is already // loaded by Mo::fetch() (which calls fetchLines() internally). $opsComplete = true; // default true when no operations are defined $_tracker = new MoOpTracking($db); $_operations = MoOpTracking::getOperationsFromMo($object); $_cumuls = array(); if ($object->qty > 0 && !empty($_operations)) { $_cumuls = $_tracker->getCumulByMo($object->id); foreach ($_operations as $_ws) { if ((float) ($_cumuls[$_ws['fk_ws']] ?? 0) < (float) $object->qty) { $opsComplete = false; break; } } } // Backend guard: block confirm_consumeandproduceall when routing ops incomplete. if ($action == 'confirm_consumeandproduceall' && !$opsComplete) { setEventMessages($langs->trans('MoOpTrackingNotComplete'), null, 'errors'); $action = 'consumeandproduceall'; // return to that view without processing } if (in_array($action, array('confirm_consumeorproduce', 'confirm_consumeandproduceall')) && $permissiontoproduce) { $stockmove = new MouvementStock($db); $labelmovement = GETPOST('inventorylabel', 'alphanohtml'); $codemovement = GETPOST('inventorycode', 'alphanohtml'); $db->begin(); $pos = 0; // Process line to consume foreach ($object->lines as $line) { if ($line->role == 'toconsume') { $tmpproduct = new Product($db); $tmpproduct->fetch($line->fk_product); $i = 1; while (GETPOSTISSET('qty-'.$line->id.'-'.$i)) { $qtytoprocess = (float) price2num(GETPOST('qty-'.$line->id.'-'.$i)); if ($qtytoprocess != 0) { // Check warehouse is set if we should have to if (GETPOSTISSET('idwarehouse-'.$line->id.'-'.$i)) { // If there is a warehouse to set if (!(GETPOST('idwarehouse-'.$line->id.'-'.$i) > 0)) { // If there is no warehouse set. $langs->load("errors"); setEventMessages($langs->trans("ErrorFieldRequiredForProduct", $langs->transnoentitiesnoconv("Warehouse"), $tmpproduct->ref), null, 'errors'); $error++; } if ($tmpproduct->status_batch && (!GETPOST('batch-'.$line->id.'-'.$i))) { $langs->load("errors"); setEventMessages($langs->trans("ErrorFieldRequiredForProduct", $langs->transnoentitiesnoconv("Batch"), $tmpproduct->ref), null, 'errors'); $error++; } } $idstockmove = 0; if (!$error && GETPOST('idwarehouse-'.$line->id.'-'.$i) > 0) { // Record stock movement $id_product_batch = 0; $stockmove->setOrigin($object->element, $object->id); $stockmove->context['mrp_role'] = 'toconsume'; if ($qtytoprocess >= 0) { $idstockmove = $stockmove->livraison($user, $line->fk_product, GETPOSTINT('idwarehouse-'.$line->id.'-'.$i), $qtytoprocess, 0, $labelmovement, dol_now(), '', '', GETPOST('batch-'.$line->id.'-'.$i), $id_product_batch, $codemovement); } else { $idstockmove = $stockmove->reception($user, $line->fk_product, GETPOSTINT('idwarehouse-'.$line->id.'-'.$i), $qtytoprocess * -1, 0, $labelmovement, dol_now(), '', '', GETPOST('batch-'.$line->id.'-'.$i), $id_product_batch, $codemovement); } if ($idstockmove < 0) { $error++; setEventMessages($stockmove->error, $stockmove->errors, 'errors'); } } if (!$error) { // Record consumption $moline = new MoLine($db); $moline->fk_mo = $object->id; $moline->position = $pos; $moline->fk_product = $line->fk_product; $moline->fk_warehouse = GETPOSTINT('idwarehouse-'.$line->id.'-'.$i); $moline->qty = $qtytoprocess; $moline->batch = GETPOST('batch-'.$line->id.'-'.$i); $moline->role = 'consumed'; $moline->fk_mrp_production = $line->id; $moline->fk_stock_movement = $idstockmove == 0 ? null : $idstockmove; $moline->fk_user_creat = $user->id; $resultmoline = $moline->create($user); if ($resultmoline <= 0) { $error++; setEventMessages($moline->error, $moline->errors, 'errors'); } $pos++; } } $i++; } } } // Process line to produce $pos = 0; foreach ($object->lines as $line) { if ($line->role == 'toproduce') { $tmpproduct = new Product($db); $tmpproduct->fetch($line->fk_product); $i = 1; while (GETPOSTISSET('qtytoproduce-'.$line->id.'-'.$i)) { $qtytoprocess = (float) price2num(GETPOST('qtytoproduce-'.$line->id.'-'.$i)); $pricetoprocess = GETPOST('pricetoproduce-'.$line->id.'-'.$i) ? price2num(GETPOST('pricetoproduce-'.$line->id.'-'.$i)) : 0; if ($qtytoprocess != 0) { // Check warehouse is set if we should have to if (GETPOSTISSET('idwarehousetoproduce-'.$line->id.'-'.$i)) { // If there is a warehouse to set if (!(GETPOST('idwarehousetoproduce-'.$line->id.'-'.$i) > 0)) { // If there is no warehouse set. $langs->load("errors"); setEventMessages($langs->trans("ErrorFieldRequiredForProduct", $langs->transnoentitiesnoconv("Warehouse"), $tmpproduct->ref), null, 'errors'); $error++; } if (isModEnabled('productbatch') && $tmpproduct->status_batch && (!GETPOST('batchtoproduce-'.$line->id.'-'.$i))) { $langs->load("errors"); setEventMessages($langs->trans("ErrorFieldRequiredForProduct", $langs->transnoentitiesnoconv("Batch"), $tmpproduct->ref), null, 'errors'); $error++; } } $idstockmove = 0; if (!$error && GETPOST('idwarehousetoproduce-'.$line->id.'-'.$i) > 0) { // Record stock movement $id_product_batch = 0; $stockmove->origin_type = $object->element; $stockmove->origin_id = $object->id; $stockmove->context['mrp_role'] = 'toproduce'; $idstockmove = $stockmove->reception($user, $line->fk_product, GETPOSTINT('idwarehousetoproduce-'.$line->id.'-'.$i), $qtytoprocess, $pricetoprocess, $labelmovement, GETPOSTDATE('eatby-'.$line->id.'-'.$i), GETPOSTDATE('sellby-'.$line->id.'-'.$i), GETPOST('batchtoproduce-'.$line->id.'-'.$i), dol_now(), $id_product_batch, $codemovement); if ($idstockmove < 0) { $error++; setEventMessages($stockmove->error, $stockmove->errors, 'errors'); } } if (!$error) { // Record production $moline = new MoLine($db); $moline->fk_mo = $object->id; $moline->position = $pos; $moline->fk_product = $line->fk_product; $moline->fk_warehouse = GETPOSTINT('idwarehousetoproduce-'.$line->id.'-'.$i); $moline->qty = $qtytoprocess; $moline->batch = GETPOST('batchtoproduce-'.$line->id.'-'.$i); $moline->role = 'produced'; $moline->fk_mrp_production = $line->id; $moline->fk_stock_movement = (($idstockmove == 0) ? null : $idstockmove); $moline->fk_user_creat = $user->id; $resultmoline = $moline->create($user); if ($resultmoline <= 0) { $error++; setEventMessages($moline->error, $moline->errors, 'errors'); } $pos++; } } $i++; } } } if (!$error) { $consumptioncomplete = true; $productioncomplete = true; if (GETPOSTINT('autoclose')) { foreach ($object->lines as $line) { $tmpproduct = new Product($db); $tmpproduct->fetch($line->fk_product); if ((int) $tmpproduct->stockable_product > 0) { if ($line->role == 'toconsume') { $arrayoflines = $object->fetchLinesLinked('consumed', $line->id); $alreadyconsumed = 0; foreach ($arrayoflines as $line2) { $alreadyconsumed += $line2['qty']; } if ($alreadyconsumed < $line->qty) { $consumptioncomplete = false; } } if ($line->role == 'toproduce') { $arrayoflines = $object->fetchLinesLinked('produced', $line->id); $alreadyproduced = 0; foreach ($arrayoflines as $line2) { $alreadyproduced += $line2['qty']; } if ($alreadyproduced < $line->qty) { $productioncomplete = false; } } } } } else { $consumptioncomplete = false; $productioncomplete = false; } // Update status of MO dol_syslog("consumptioncomplete = ".json_encode($consumptioncomplete)." productioncomplete = ".json_encode($productioncomplete)); if ($consumptioncomplete && $productioncomplete) { $result = $object->setStatut($object::STATUS_PRODUCED, 0, '', 'MRP_MO_PRODUCED'); } else { $result = $object->setStatut($object::STATUS_INPROGRESS, 0, '', 'MRP_MO_PRODUCED'); } if ($result <= 0) { $error++; setEventMessages($object->error, $object->errors, 'errors'); } } if ($error) { $action = str_replace('confirm_', '', $action); $db->rollback(); } else { $db->commit(); // Auto-sync FORMING (rank-0 operation) cumulative to actual FG produced qty. $_syncTracker = new MoOpTracking($db); $_syncOps = MoOpTracking::getOperationsFromMo($object); if (!empty($_syncOps)) { $_syncFormingWs = $_syncOps[0]; // rank 0 is always FORMING $_syncCumuls = $_syncTracker->getCumulByMo($object->id); $_syncCurrentCumul = (float) ($_syncCumuls[$_syncFormingWs['fk_ws']] ?? 0); // Sum all produced lines from DB (fresh after commit) $_syncSql = "SELECT SUM(qty) as total FROM ".MAIN_DB_PREFIX."mrp_production WHERE fk_mo = ".(int)$object->id." AND role = 'produced'"; $_syncRes = $db->query($_syncSql); $_syncProducedQty = 0.0; if ($_syncRes) { $_syncObj = $db->fetch_object($_syncRes); if ($_syncObj && $_syncObj->total !== null) { $_syncProducedQty = (float) $_syncObj->total; } $db->free($_syncRes); } $_syncDelta = $_syncProducedQty - $_syncCurrentCumul; if (abs($_syncDelta) > 1e-9) { $_syncTracker->addDelta($object->id, $_syncFormingWs['fk_ws'], $_syncFormingWs['ws_label'], 0, $_syncDelta, $user->id, 'auto-sync: FG produced'); } } // Redirect to avoid to action done a second time if we make a back from browser header("Location: ".$_SERVER["PHP_SELF"].'?id='.$object->id); exit; } } // Action close produced if ($action == 'confirm_produced' && $confirm == 'yes' && $permissiontoadd) { // Backend guard: block if routing operations are not yet complete. if (!$opsComplete) { setEventMessages($langs->trans('MoOpTrackingNotComplete'), null, 'errors'); $action = ''; } else { $result = $object->setStatut($object::STATUS_PRODUCED, 0, '', 'MRP_MO_PRODUCED'); $result = $object->setStatut($object::STATUS_PRODUCED, 0, '', 'MRP_MO_PRODUCED'); if ($result >= 0) { // Define output language if (!getDolGlobalString('MAIN_DISABLE_PDF_AUTOUPDATE')) { $outputlangs = $langs; $newlang = ''; if (getDolGlobalInt('MAIN_MULTILANGS') /* && empty($newlang) */ && GETPOST('lang_id', 'aZ09')) { $newlang = GETPOST('lang_id', 'aZ09'); } if (getDolGlobalInt('MAIN_MULTILANGS') && empty($newlang)) { $newlang = $object->thirdparty->default_lang; } if (!empty($newlang)) { $outputlangs = new Translate("", $conf); $outputlangs->setDefaultLang($newlang); } $model = $object->model_pdf; $ret = $object->fetch($id); // Reload to get new records $object->generateDocument($model, $outputlangs, 0, 0, 0); } } else { setEventMessages($object->error, $object->errors, 'errors'); } } // end opsComplete check } // Action: save per-operation progress delta if ($action == 'confirm_op_progress' && $permissiontoproduce) { // Backend guard: reject save until MO has been formally started (status INPROGRESS). if ($object->status != Mo::STATUS_INPROGRESS) { setEventMessages($langs->trans('MoOpTrackingLockedUntilStart'), null, 'errors'); $action = ''; } else { $tracker = new MoOpTracking($db); $operations = MoOpTracking::getOperationsFromMo($object); $cumuls = $tracker->getCumulByMo($object->id); $db->begin(); $error = 0; foreach ($operations as $rank => $ws) { $delta = (float) price2num(GETPOST('op_delta_' . $ws['fk_ws'], 'alpha')); if ($delta == 0) { continue; } $currentCumul = (float) ($cumuls[$ws['fk_ws']] ?? 0); $newCumul = $currentCumul + $delta; // Enforce: cumulative cannot go below zero if ($newCumul < 0) { $error++; setEventMessages($langs->trans('MoOpTrackingNegativeCumul', $ws['ws_label']), null, 'errors'); break; } // Enforce: cumulative cannot exceed MO planned qty if ($newCumul > (float) $object->qty) { $error++; setEventMessages($langs->trans('MoOpTrackingExceedsMoQty', $ws['ws_label'], $object->qty), null, 'errors'); break; } // Enforce routing order: step N cumul must not exceed step N-1 cumul if ($rank > 0) { $prevWs = $operations[$rank - 1]; $prevCumul = (float) ($cumuls[$prevWs['fk_ws']] ?? 0); if ($newCumul > $prevCumul) { $error++; setEventMessages($langs->trans('MoOpTrackingExceedsPrev', $ws['ws_label'], $prevWs['ws_label']), null, 'errors'); break; } } if (!$error) { $note = GETPOST('op_note_' . $ws['fk_ws'], 'alphanohtml'); $res = $tracker->addDelta($object->id, $ws['fk_ws'], $ws['ws_label'], $rank, $delta, $user->id, $note); if ($res < 0) { $error++; setEventMessages($tracker->error, $tracker->errors, 'errors'); break; } // Update in-memory cumul for subsequent rank checks in this loop $cumuls[$ws['fk_ws']] = $newCumul; } } if ($error) { $db->rollback(); } else { $db->commit(); // Auto-update MO extrafield "working_station" to the first incomplete operation. // Re-fetch cumuls from DB after commit so negative corrections are reflected correctly. $newWsId = null; if (!empty($operations) && $object->qty > 0) { $cumulsFresh = $tracker->getCumulByMo($object->id); foreach ($operations as $ws) { $c = (float) ($cumulsFresh[$ws['fk_ws']] ?? 0); if ($c < (float) $object->qty) { $newWsId = (int) $ws['fk_ws']; break; } } // If $newWsId is still null all operations are complete — leave the field unchanged. if ($newWsId !== null) { $object->fetch_optionals(); $currentWsId = isset($object->array_options['options_working_station']) ? (int) $object->array_options['options_working_station'] : 0; if ($currentWsId !== $newWsId) { $object->array_options['options_working_station'] = $newWsId; $object->insertExtraFields(); } } } header('Location: ' . $_SERVER['PHP_SELF'] . '?id=' . $object->id); exit; } } // end STATUS_INPROGRESS guard } if ($action == 'confirm_editline' && $permissiontoadd) { $moline = new MoLine($db); $res = $moline->fetch(GETPOSTINT('lineid')); if ($result > 0) { $extrafields->fetch_name_optionals_label($moline->table_element); if (!empty($extrafields->attributes[$moline->table_element]['label'])) { foreach ($extrafields->attributes[$moline->table_element]['label'] as $key => $label) { $value = GETPOST('options_'.$key, 'alphanohtml'); $moline->array_options["options_".$key] = $value; } } $moline->qty = GETPOSTFLOAT('qty_lineProduce'); if (GETPOSTISSET('warehouse_lineProduce')) { $moline->fk_warehouse = (GETPOSTINT('warehouse_lineProduce') > 0 ? GETPOSTINT('warehouse_lineProduce') : 0); } if (GETPOSTISSET('workstation_lineProduce')) { $moline->fk_default_workstation = (GETPOSTINT('workstation_lineProduce') > 0 ? GETPOSTINT('workstation_lineProduce') : 0); } $res = $moline->update($user); if ($res < 0) { setEventMessages($moline->error, $moline->errors, 'errors'); header("Location: ".$_SERVER["PHP_SELF"].'?id='.$object->id); exit; } header("Location: ".$_SERVER["PHP_SELF"].'?id='.$object->id); exit; } } } /* * View */ $form = new Form($db); $formproject = new FormProjets($db); $formproduct = new FormProduct($db); $tmpwarehouse = new Entrepot($db); $tmpbatch = new Productlot($db); $tmpstockmovement = new MouvementStock($db); $title = $langs->trans('Mo'); $help_url = 'EN:Module_Manufacturing_Orders|FR:Module_Ordres_de_Fabrication|DE:Modul_Fertigungsauftrag'; $morejs = array('/mrp/js/lib_dispatch.js.php'); llxHeader('', $title, $help_url, '', 0, 0, $morejs, '', '', 'mod-mrp page-card_production'); $newToken = newToken(); // Part to show record if ($object->id > 0 && (empty($action) || ($action != 'edit' && $action != 'create' && $action != 'reload'))) { $res = $object->fetch_thirdparty(); $res = $object->fetch_optionals(); // $opsComplete, $_tracker, $_operations, $_cumuls were already computed in the // Actions section above and remain in scope here. if (getDolGlobalString('STOCK_CONSUMPTION_FROM_MANUFACTURING_WAREHOUSE') && $object->fk_warehouse > 0) { $tmpwarehouse->fetch((int) $object->fk_warehouse); $fk_default_warehouse = (int) $object->fk_warehouse; } $head = moPrepareHead($object); print dol_get_fiche_head($head, 'production', $langs->trans("ManufacturingOrder"), -1, $object->picto); $formconfirm = ''; // Confirmation to delete if ($action == 'delete') { $formquestion = array( array( 'label' => $langs->trans('MoCancelConsumedAndProducedLines'), 'name' => 'alsoCancelConsumedAndProducedLines', 'type' => 'checkbox', 'value' => 0 ), ); $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id, $langs->trans('DeleteMo'), $langs->trans('ConfirmDeleteMo'), 'confirm_delete', $formquestion, 0, 1); } // Confirmation to delete line if ($action == 'deleteline') { $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id.'&lineid='.$lineid.'&fk_movement='.$fk_movement, $langs->trans('DeleteLine'), $langs->trans('ConfirmDeleteLine'), 'confirm_deleteline', '', 0, 1); } // Clone confirmation if ($action == 'clone') { // Create an array for form $formquestion = array(); $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id, $langs->trans('ToClone'), $langs->trans('ConfirmCloneMo', $object->ref), 'confirm_clone', $formquestion, 'yes', 1); } // Confirmation of validation if ($action == 'validate') { // We check that object has a temporary ref $ref = substr($object->ref, 1, 4); if ($ref == 'PROV') { $object->fetch_product(); $numref = $object->getNextNumRef($object->product); } else { $numref = (string) $object->ref; } $text = $langs->trans('ConfirmValidateMo', $numref); /*if (isModEnabled('notification')) { require_once DOL_DOCUMENT_ROOT . '/core/class/notify.class.php'; $notify = new Notify($db); $text .= '
'; $text .= $notify->confirmMessage('BOM_VALIDATE', $object->socid, $object); }*/ $formquestion = array(); if (isModEnabled('mrp')) { $langs->load("mrp"); require_once DOL_DOCUMENT_ROOT.'/product/class/html.formproduct.class.php'; $formproduct = new FormProduct($db); $forcecombo = 0; if ($conf->browser->name == 'ie') { $forcecombo = 1; // There is a bug in IE10 that make combo inside popup crazy } $formquestion = array( // 'text' => $langs->trans("ConfirmClone"), // array('type' => 'checkbox', 'name' => 'clone_content', 'label' => $langs->trans("CloneMainAttributes"), 'value' => 1), // array('type' => 'checkbox', 'name' => 'update_prices', 'label' => $langs->trans("PuttingPricesUpToDate"), 'value' => 1), ); } $formconfirm = $form->formconfirm($_SERVER["PHP_SELF"].'?id='.$object->id, $langs->trans('Validate'), $text, 'confirm_validate', $formquestion, 0, 1, 220); } // Confirmation to cancel if ($action == 'cancel') { $formquestion = array( array( 'label' => $langs->trans('MoCancelConsumedAndProducedLines'), 'name' => 'alsoCancelConsumedAndProducedLines', 'type' => 'checkbox', 'value' => !getDolGlobalString('MO_ALSO_CANCEL_CONSUMED_AND_PRODUCED_LINES_BY_DEFAULT') ? 0 : 1 ), ); $formconfirm = $form->formconfirm(dolBuildUrl($_SERVER["PHP_SELF"], ['id' => $object->id]), $langs->trans('CancelMo'), $langs->trans('ConfirmCancelMo'), 'confirm_cancel', $formquestion, 0, 1); } // Call Hook formConfirm $parameters = array('formConfirm' => $formconfirm, 'lineid' => $lineid); $reshook = $hookmanager->executeHooks('formConfirm', $parameters, $object, $action); // Note that $action and $object may have been modified by hook if (empty($reshook)) { $formconfirm .= $hookmanager->resPrint; } elseif ($reshook > 0) { $formconfirm = $hookmanager->resPrint; } // Print form confirm print $formconfirm; // MO file // ------------------------------------------------------------ $linkback = ''.$langs->trans("BackToList").''; $morehtmlref = '
'; /* // Ref bis $morehtmlref.=$form->editfieldkey("RefBis", 'ref_client', $object->ref_client, $object, $user->rights->mrp->creer, 'string', '', 0, 1); $morehtmlref.=$form->editfieldval("RefBis", 'ref_client', $object->ref_client, $object, $user->rights->mrp->creer, 'string', '', null, null, '', 1); */ // Thirdparty if (is_object($object->thirdparty)) { $morehtmlref .= $object->thirdparty->getNomUrl(1, 'customer'); if (!getDolGlobalString('MAIN_DISABLE_OTHER_LINK') && $object->thirdparty->id > 0) { $morehtmlref .= ' ('.$langs->trans("OtherOrders").')'; } } // Project if (isModEnabled('project')) { $langs->load("projects"); if (is_object($object->thirdparty)) { $morehtmlref .= '
'; } if ($permissiontoadd) { $morehtmlref .= img_picto($langs->trans("Project"), 'project', 'class="pictofixedwidth"'); if ($action != 'classify') { $morehtmlref .= ''.img_edit($langs->transnoentitiesnoconv('SetProject')).' '; } $morehtmlref .= $form->form_project($_SERVER['PHP_SELF'].'?id='.$object->id, $object->socid, (string) $object->fk_project, ($action == 'classify' ? 'projectid' : 'none'), 0, 0, 0, 1, '', 'maxwidth300'); } else { if (!empty($object->fk_project)) { $proj = new Project($db); $proj->fetch($object->fk_project); $morehtmlref .= $proj->getNomUrl(1); if ($proj->title) { $morehtmlref .= ' - '.dol_escape_htmltag($proj->title).''; } } } } $morehtmlref .= '
'; dol_banner_tab($object, 'ref', $linkback, 1, 'ref', 'ref', $morehtmlref); // Compact operation routing status — visible in header without scrolling. if (!empty($_operations)) { print '
'; print ''.$langs->trans('MoOpTrackingStatus').':  '; foreach ($_operations as $_si => $_sop) { $_scumul = (float) ($_cumuls[$_sop['fk_ws']] ?? 0); $_smoqty = (float) $object->qty; if ($_smoqty > 0 && $_scumul >= $_smoqty) { $_sbadge = ''; } elseif ($_scumul > 0) { $_sbadge = ''.round($_scumul, 2).'/'.$_smoqty.''; } else { $_sbadge = 'waiting'; } if ($_si > 0) { print '  ›  '; } print ''.dol_escape_htmltag($_sop['ws_label']).' '.$_sbadge; } print '
'; } print '
'; print '
'; print '
'; print ''."\n"; // Common attributes $keyforbreak = 'fk_warehouse'; unset($object->fields['fk_project']); unset($object->fields['fk_soc']); include DOL_DOCUMENT_ROOT.'/core/tpl/commonfields_view.tpl.php'; // Other attributes include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_view.tpl.php'; print '
'; print '
'; print '
'; print '
'; print dol_get_fiche_end(); if (!in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print '
'; $parameters = array(); // Note that $action and $object may be modified by hook $reshook = $hookmanager->executeHooks('addMoreActionsButtons', $parameters, $object, $action); if (empty($reshook)) { // Validate if ($object->status == $object::STATUS_DRAFT) { if ($permissiontoadd) { if (empty($object->table_element_line) || (is_array($object->lines) && count($object->lines) > 0)) { print ''.$langs->trans("Validate").''; } else { $langs->load("errors"); print ''.$langs->trans("Validate").''; } } } // Consume or produce if ($object->status == Mo::STATUS_VALIDATED || $object->status == Mo::STATUS_INPROGRESS) { if ($permissiontoproduce) { print ''.$langs->trans('ConsumeOrProduce').''; } else { print ''.$langs->trans('ConsumeOrProduce').''; } } elseif ($object->status == Mo::STATUS_DRAFT) { print ''.$langs->trans('ConsumeOrProduce').''; } // ConsumeAndProduceAll if ($object->status == Mo::STATUS_VALIDATED || $object->status == Mo::STATUS_INPROGRESS) { if (!$opsComplete) { print ''.$langs->trans('ConsumeAndProduceAll').''; } elseif ($permissiontoproduce) { print ''.$langs->trans('ConsumeAndProduceAll').''; } else { print ''.$langs->trans('ConsumeAndProduceAll').''; } } elseif ($object->status == Mo::STATUS_DRAFT) { print ''.$langs->trans('ConsumeAndProduceAll').''; } // Cancel - Reopen if ($permissiontoadd) { if ($object->status == $object::STATUS_VALIDATED || $object->status == $object::STATUS_INPROGRESS) { $arrayproduced = $object->fetchLinesLinked('produced', 0); $nbProduced = 0; foreach ($arrayproduced as $lineproduced) { $nbProduced += $lineproduced['qty']; } if (!$opsComplete) { print ''.$langs->trans("Close").''."\n"; } elseif ($nbProduced > 0) { // If production has started, we can close it print ''.$langs->trans("Close").''."\n"; } else { print 'transnoentitiesnoconv("Production")).'">'.$langs->trans("Close").''."\n"; } print ''.$langs->trans("Cancel").''."\n"; } if ($object->status == $object::STATUS_CANCELED) { print ''.$langs->trans("ReOpen").''."\n"; } if ($object->status == $object::STATUS_PRODUCED) { if ($permissiontoproduce) { print ''.$langs->trans('ReOpen').''; } else { print ''.$langs->trans('ReOpen').''; } } } } print '
'; } if (in_array($action, array('consumeorproduce', 'consumeandproduceall', 'addconsumeline', 'addproduceline', 'editline'))) { print ''; print '
'; print ''; print ''; print ''; print ''; // Note: closing form is add end of page if (in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { $defaultstockmovementlabel = GETPOST('inventorylabel', 'alphanohtml') ? GETPOST('inventorylabel', 'alphanohtml') : $langs->trans("ProductionForRef", $object->ref); $defaultstockmovementcode = GETPOST('inventorycode', 'alphanohtml') ? GETPOST('inventorycode', 'alphanohtml') : dol_print_date(dol_now(), 'dayhourlog'); print '
'; print '
'.$langs->trans("ConfirmProductionDesc", $langs->transnoentitiesnoconv("Confirm")).'
'; print ''.$langs->trans("InventoryCode").':'; print ''; print ''; print '   '; print ''; print ''.$langs->trans("MovementLabel").':'; print ''; print '

'; print '
'; print ''; print '   '; print ''; print '

'; print '
'; print '
'; } } /* * Lines */ $collapse = 1; if (!empty($object->table_element_line)) { // Show object lines $object->fetchLines(); $bomcost = 0; if ($object->fk_bom > 0) { $bom = new BOM($db); $res = $bom->fetch($object->fk_bom); if ($res > 0) { $bom->calculateCosts(); $bomcost = $bom->unit_cost; } } // Lines to consume print ''."\n"; print '
'; print '
'; print '
'; $url = $_SERVER["PHP_SELF"].'?id='.$object->id.'&action=addconsumeline&token='.newToken(); $permissiontoaddaconsumeline = ($object->status != $object::STATUS_PRODUCED && $object->status != $object::STATUS_CANCELED) ? 1 : -2; $parameters = array('morecss' => 'reposition'); $helpText = ''; if ($permissiontoaddaconsumeline == -2) { $helpText = $langs->trans('MOIsClosed'); } $newcardbutton = ''; if ($action != 'consumeorproduce' && $action != 'consumeandproduceall') { $newcardbutton = dolGetButtonTitle($langs->trans('AddNewConsumeLines'), $helpText, 'fa fa-plus-circle size15x', $url, '', $permissiontoaddaconsumeline, $parameters); } print load_fiche_titre($langs->trans('Consumption'), $newcardbutton, '', 0, '', '', ''); print '
'; print ''; print ''."\n"; print ''; // Product print ''; // Qty print ''; // Unit print ''; // Cost price if ($permissiontoupdatecost && getDolGlobalString('MRP_SHOW_COST_FOR_CONSUMPTION')) { print ''; } // Qty already consumed print ''; // Warehouse print ''; if (isModEnabled('stock')) { // Available print ''; } // Lot - serial if (isModEnabled('productbatch')) { print ''; } // Split print ''; // SplitAll print ''; // Edit Line if ($object->status == Mo::STATUS_DRAFT) { print ''; } // Action if ($permissiontodelete) { print ''; } print ''; if ($action == 'addconsumeline') { print ''."\n"; print ''; // Product print ''; // Qty print ''; // Unit print ''; // Cost price if ($permissiontoupdatecost && getDolGlobalString('MRP_SHOW_COST_FOR_CONSUMPTION')) { print ''; } $colspan = 3; if (isModEnabled('stock')) { $colspan++; } if (isModEnabled('productbatch')) { $colspan++; } // Qty already consumed + Warehouse print ''; // Split All print ''; // Edit Line if ($object->status == Mo::STATUS_DRAFT) { print ''; } // Action if ($permissiontodelete) { print ''; } print ''; // Extrafields Line if (is_object($objectline)) { $extrafields->fetch_name_optionals_label($object->table_element_line); $temps = $objectline->showOptionals($extrafields, 'edit', array(), '', '', '1', 'line'); if (!empty($temps)) { print ''; } } } // Lines to consume $bomcostupdated = 0; // We will recalculate the unitary cost to produce a product using the real "products to consume into MO" if (!empty($object->lines)) { $nblinetoconsume = 0; foreach ($object->lines as $line) { if ($line->role == 'toconsume') { $nblinetoconsume++; } } $nblinetoconsumecursor = 0; foreach ($object->lines as $line) { if ($line->role == 'toconsume') { $nblinetoconsumecursor++; $tmpproduct = new Product($db); $tmpproduct->fetch($line->fk_product); $linecost = price2num($tmpproduct->pmp, 'MT'); if ($object->qty > 0) { // add free consume line cost to $bomcostupdated $costprice = price2num((!empty($tmpproduct->cost_price)) ? $tmpproduct->cost_price : $tmpproduct->pmp); if (empty($costprice)) { require_once DOL_DOCUMENT_ROOT.'/fourn/class/fournisseur.product.class.php'; $productFournisseur = new ProductFournisseur($db); if ($productFournisseur->find_min_price_product_fournisseur($line->fk_product, $line->qty) > 0) { $costprice = $productFournisseur->fourn_unitprice; } else { $costprice = 0; } } $useunit = (($tmpproduct->type == Product::TYPE_PRODUCT && getDolGlobalInt('PRODUCT_USE_UNITS')) || (($tmpproduct->type == Product::TYPE_SERVICE) && ($line->fk_unit))); if ($useunit && $line->fk_unit > 0) { $reg = []; $qtyhourservice = 0; if (preg_match('/^(\d+)([a-z]+)$/', $tmpproduct->duration, $reg)) { $qtyhourservice = convertDurationtoHour((float) $reg[1], (string) $reg[2]); } $qtyhourforline = 0; if ($line->fk_unit) { $unitforline = measuringUnitString($line->fk_unit, '', null, 1); $qtyhourforline = convertDurationtoHour($line->qty, $unitforline); } if ($qtyhourservice && $qtyhourforline) { $linecost = price2num(($qtyhourforline / $qtyhourservice * $costprice) / $object->qty, 'MT'); // price for line for all quantities $bomcostupdated += price2num(($qtyhourforline / $qtyhourservice * $costprice) / $object->qty, 'MU'); // same but with full accuracy } else { $linecost = price2num(($line->qty * $costprice) / $object->qty, 'MT'); // price for line for all quantities $bomcostupdated += price2num(($line->qty * $costprice) / $object->qty, 'MU'); // same but with full accuracy } } else { $linecost = price2num(($line->qty * $costprice) / $object->qty, 'MT'); // price for line for all quantities $bomcostupdated += price2num(($line->qty * $costprice) / $object->qty, 'MU'); // same but with full accuracy } } $bomcostupdated = price2num($bomcostupdated, 'MU'); $arrayoflines = $object->fetchLinesLinked('consumed', $line->id); $alreadyconsumed = 0; foreach ($arrayoflines as $line2) { $alreadyconsumed += $line2['qty']; } if ($action == 'editline' && $lineid == $line->id) { $linecost = price2num($tmpproduct->pmp, 'MT'); $arrayoflines = $object->fetchLinesLinked('consumed', $line->id); $alreadyconsumed = 0; if (is_array($arrayoflines) && !empty($arrayoflines)) { foreach ($arrayoflines as $line2) { $alreadyconsumed += $line2['qty']; } } $suffix = '_' . $line->id; print '' . "\n"; // hidden fields for js function print ''; // Duration - Time spent print ''; print ''; print ''; // Product print ''; // Qty print ''; // Unit print ''; // Cost price if ($permissiontoupdatecost && getDolGlobalString('MRP_SHOW_COST_FOR_CONSUMPTION')) { print ''; } // Qty consumed print ''; // Warehouse / Workstation print ''; // Stock if (isModEnabled('stock')) { print ''; } // Lot - serial if (isModEnabled('productbatch')) { print ''; } // Split + SplitAll + Edit line + Delete print ''; print ''; // Extrafields Line if (!empty($extrafields)) { $line->fetch_optionals(); $temps = $line->showOptionals($extrafields, 'edit', array(), '', '', '1', 'line'); if (!empty($temps)) { $colspan = 10; print ''; } } } else { $suffix = '_' . $line->id; print '' . "\n"; // hidden fields for js function print ''; print ''; print ''; // Product print ''; // Qty print ''; // Unit print ''; // Cost price if ($permissiontoupdatecost && getDolGlobalString('MRP_SHOW_COST_FOR_CONSUMPTION')) { print ''; } // Already consumed print ''; // Warehouse and/or workstation print ''; // Stock if (isModEnabled('stock')) { print ''; } // Lot if (isModEnabled('productbatch')) { print ''; } // Split print ''; // Split All print ''; // Action Edit line if ($object->status == Mo::STATUS_DRAFT) { $href = $_SERVER["PHP_SELF"] . '?id=' . ((int) $object->id) . '&action=editline&token=' . newToken() . '&lineid=' . ((int) $line->id); print ''; } // Action delete line, if no consumption has occurred for this product if ($permissiontodelete && empty($arrayoflines)) { $href = $_SERVER["PHP_SELF"] . '?id=' . ((int) $object->id) . '&action=deleteline&token=' . newToken() . '&lineid=' . ((int) $line->id); print ''; } print ''; // Extrafields Line if (!empty($extrafields)) { $line->fetch_optionals(); $temps = $line->showOptionals($extrafields, 'view', array(), '', '', '1', 'line'); if (!empty($temps)) { $colspan = 10; print ''; } } } // Show detailed of already consumed with js code to collapse foreach ($arrayoflines as $line2) { print ''; // Date print ''; // Qty print ''; // Unit print ''; // Cost price if ($permissiontoupdatecost && getDolGlobalString('MRP_SHOW_COST_FOR_CONSUMPTION')) { print ''; } //Already consumed print ''; // Warehouse print ''; // Stock if (isModEnabled('stock')) { print ''; } // Lot Batch if (isModEnabled('productbatch')) { print ''; } // Split print ''; // Split All print ''; // Action Edit line if ($object->status == Mo::STATUS_DRAFT) { $href = $_SERVER["PHP_SELF"] . '?id=' . ((int) $object->id) . '&action=editline&token=' . newToken() . '&lineid=' . ((int) $line2['rowid']); print ''; } // Action delete line if ($permissiontodelete) { $href = $_SERVER["PHP_SELF"].'?id='.((int) $object->id).'&action=deleteline&token='.newToken().'&lineid='.((int) $line2['rowid']).'&fk_movement='.((int) $line2['fk_stock_movement']); print ''; } print ''; } if (in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { $i = 1; print ''."\n"; $maxQty = 1; print ''; // Ref print ''; $preselected = (GETPOSTISSET('qty-'.$line->id.'-'.$i) ? GETPOST('qty-'.$line->id.'-'.$i) : max(0, $line->qty - $alreadyconsumed)); $disable = ''; if (getDolGlobalString('MRP_NEVER_CONSUME_MORE_THAN_EXPECTED') && ($line->qty - $alreadyconsumed) <= 0) { $disable = 'disabled'; } // input hidden with fk_product of line print ''; // Qty print ''; // Unit print ''; // Cost if ($permissiontoupdatecost && getDolGlobalString('MRP_SHOW_COST_FOR_CONSUMPTION')) { print ''; } // Already consumed print ''; // Warehouse print ''; // Stock if (isModEnabled('stock')) { print ''; } // Lot / Batch if (isModEnabled('productbatch')) { print ''; } // Split $type = 'batch'; print ''; // Split All print ''; // Edit Line if ($object->status == Mo::STATUS_DRAFT) { print ''; } // Action delete line if ($permissiontodelete) { print ''; } print ''; } } } } print '
'.$langs->trans("Product").''.$langs->trans("Qty").''.$langs->trans("UnitCost").''; print $langs->trans("QtyAlreadyConsumedShort"); print ''; if ($collapse || in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print $langs->trans("Warehouse"); if (isModEnabled('workstation')) { print ' '.$langs->trans("or").' '.$langs->trans("Workstation"); } // Select warehouse to force it everywhere if (in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { $listwarehouses = $tmpwarehouse->list_array(1); if (count($listwarehouses) > 1) { print '
'.$form->selectarray('fk_default_warehouse', $listwarehouses, $fk_default_warehouse, $langs->trans("ForceTo"), 0, 0, '', 0, 0, 0, '', 'minwidth100 maxwidth200', 1); } elseif (count($listwarehouses) == 1) { print '
'.$form->selectarray('fk_default_warehouse', $listwarehouses, $fk_default_warehouse, 0, 0, 0, '', 0, 0, 0, '', 'minwidth100 maxwidth200', 1); } } } print '
'; if ($collapse || in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print $langs->trans("Stock"); } print ''; if ($collapse || in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print $langs->trans("Batch"); } print '
'; print $form->select_produits(0, 'productidtoadd', '', 0, 0, -1, 2, '', 1, array(), 0, '1', 0, 'maxwidth150'); print ''; //if (getDolGlobalInt('PRODUCT_USE_UNITS')) { //... //} print ''; print ''; print ''; print '
'; print $temps; print '
' . $tmpproduct->getNomUrl(1); print '
' . $tmpproduct->label . ''; print '
'; print ''; print ''; $useunit = (($tmpproduct->type == Product::TYPE_PRODUCT && getDolGlobalInt('PRODUCT_USE_UNITS')) || (($tmpproduct->type == Product::TYPE_SERVICE) && ($line->fk_unit))); if ($useunit) { print measuringUnitString($line->fk_unit, '', null, 2); } print ''; print ' ' . price2num($alreadyconsumed, 'MS'); print ''; if ($tmpproduct->type == Product::TYPE_PRODUCT) { print $formproduct->selectWarehouses($line->fk_warehouse, 'warehouse_lineProduce', 'warehouseopen', 1); } elseif (isModEnabled('workstation')) { print $formproduct->selectWorkstations($line->fk_default_workstation, 'workstation_lineProduce', 1); } print ''; if ($tmpproduct->isStockManaged()) { if ($tmpproduct->stock_reel < ($line->qty - $alreadyconsumed)) { print img_warning($langs->trans('StockTooLow')).' '; } print ''. $tmpproduct->stock_reel .' '; } print ''; print ''; print ''; print '
'; print $temps; print '
' . $tmpproduct->getNomUrl(1); print '
' . $tmpproduct->label . '
'; print '
'; $help = ''; if ($line->qty_frozen) { $help = ($help ? '
' : '') . '' . $langs->trans("QuantityFrozen") . ': ' . yn(1) . ' (' . $langs->trans("QuantityConsumedInvariable") . ')'; print $form->textwithpicto('', $help, -1, 'lock') . ' '; } if ($line->disable_stock_change) { $help = ($help ? '
' : '') . '' . $langs->trans("DisableStockChange") . ': ' . yn(1) . ' (' . (($tmpproduct->type == Product::TYPE_SERVICE && !getDolGlobalString('STOCK_SUPPORTS_SERVICES')) ? $langs->trans("NoStockChangeOnServices") : $langs->trans("DisableStockChangeHelp")) . ')'; print $form->textwithpicto('', $help, -1, 'help') . ' '; } print price2num($line->qty, 'MS'); print '
'; $useunit = (($tmpproduct->type == Product::TYPE_PRODUCT && getDolGlobalInt('PRODUCT_USE_UNITS')) || (($tmpproduct->type == Product::TYPE_SERVICE) && ($line->fk_unit))); if ($useunit) { print measuringUnitString($line->fk_unit, '', null, 2); } print ''; print price($linecost); print ''; if ($alreadyconsumed) { print ''; if (empty($conf->use_javascript_ajax)) { print 'id . '">'; } print img_picto($langs->trans("ShowDetails"), "chevron-down", 'id="expandtoproduce' . $line->id . '"'); if (empty($conf->use_javascript_ajax)) { print ''; } } else { if ($nblinetoconsume == $nblinetoconsumecursor) { // If it is the last line print ''; } } print ' ' . price2num($alreadyconsumed, 'MS'); print ''; if ($tmpproduct->isStockManaged()) { // When STOCK_CONSUMPTION_FROM_MANUFACTURING_WAREHOUSE is set, we always use the warehouse of the MO, the same than production. if (getDolGlobalString('STOCK_CONSUMPTION_FROM_MANUFACTURING_WAREHOUSE') && $tmpwarehouse->id > 0) { print img_picto('', $tmpwarehouse->picto) . " " . $tmpwarehouse->label; } else { if ($line->fk_warehouse > 0) { $warehouseline = new Entrepot($db); $warehouseline->fetch($line->fk_warehouse); print $warehouseline->getNomUrl(1); } } } if (isModEnabled('workstation') && $line->fk_default_workstation > 0) { $tmpworkstation = new Workstation($db); $tmpworkstation->fetch($line->fk_default_workstation); print $tmpworkstation->getNomUrl(1); } print ''; if (!getDolGlobalString('STOCK_SUPPORTS_SERVICES') && $tmpproduct->type != Product::TYPE_SERVICE) { if (!$line->disable_stock_change && $tmpproduct->stock_reel < ($line->qty - $alreadyconsumed)) { print img_warning($langs->trans('StockTooLow')) . ' '; } if (!getDolGlobalString('STOCK_CONSUMPTION_FROM_MANUFACTURING_WAREHOUSE') || empty($tmpwarehouse->id)) { print price2num($tmpproduct->stock_reel, 'MS'); // Available } else { // Print only the stock in the selected warehouse $tmpproduct->load_stock(); $wh_stock = $tmpproduct->stock_warehouse[$tmpwarehouse->id]; if (!empty($wh_stock)) { print price2num($wh_stock->real, 'MS'); } else { print "0"; } } } print ''; print ''; print img_picto($langs->trans('TooltipEditAndRevertStockMovement'), 'edit'); print ''; print ''; print ''; print img_picto($langs->trans('TooltipDeleteAndRevertStockMovement'), 'delete'); print ''; print '
'; print $temps; print '
'; $tmpstockmovement->id = $line2['fk_stock_movement']; print ''.img_picto($langs->trans("StockMovement"), 'movement', 'class="paddingright"').''; print dol_print_date($line2['date'], 'dayhour', 'tzuserrel'); print ''.$line2['qty'].''; if ($line2['fk_warehouse'] > 0) { $result = $tmpwarehouse->fetch($line2['fk_warehouse']); if ($result > 0) { print $tmpwarehouse->getNomUrl(1); } } print ''; if ($line2['batch'] != '') { $tmpbatch->fetch(0, $line2['fk_product'], $line2['batch']); print $tmpbatch->getNomUrl(1); } print ''; print ''; print img_picto($langs->trans('TooltipEditAndRevertStockMovement'), 'edit'); print ''; print ''; print ''; print img_picto($langs->trans('TooltipDeleteAndRevertStockMovement'), 'delete'); print ''; print '
'.$langs->trans("ToConsume").''; if ((int) $tmpproduct->stockable_product > 0) { print ''; } else { print ''; print '' . $langs->trans("StockDisabled") . ''; } print ''; if (($tmpproduct->type == Product::TYPE_PRODUCT || getDolGlobalString('STOCK_SUPPORTS_SERVICES')) && ((int) $tmpproduct->stockable_product > 0)) { if (empty($line->disable_stock_change)) { $preselected = (GETPOSTISSET('idwarehouse-'.$line->id.'-'.$i) ? GETPOST('idwarehouse-'.$line->id.'-'.$i) : ($tmpproduct->fk_default_warehouse > 0 ? $tmpproduct->fk_default_warehouse : ($object->fk_warehouse > 0 ? $object->fk_warehouse : 'ifone'))); print $formproduct->selectWarehouses($preselected, 'idwarehouse-'.$line->id.'-'.$i, '', 1, 0, $line->fk_product, '', 1, 0, array(), 'maxwidth200 csswarehouse_'.$line->id.'_'.$i); } else { print ''.$langs->trans("DisableStockChange").''; } } else { if ((int) $tmpproduct->stockable_product > 0) { print '' . $langs->trans("StockDisabled") . ''; } else { print '' . $langs->trans("NoStockChangeOnServices") . ''; } } print ''; if ($tmpproduct->status_batch) { $preselected = (GETPOSTISSET('batch-'.$line->id.'-'.$i) ? GETPOST('batch-'.$line->id.'-'.$i) : ''); print ''; print $formproduct->selectLotDataList('batch-'.$line->id.'-'.$i, 0, $line->fk_product, 0, array()); } print ''; print ' '.img_picto($langs->trans('AddStockLocationLine'), 'split', 'class="splitbutton" onClick="addDispatchLine('.((int) $line->id).', \''.dol_escape_js($type).'\', \'qtymissingconsume\')"'); print ''; if (($action == 'consumeorproduce' || $action == 'consumeandproduceall') && $tmpproduct->status_batch == 2) { print img_picto($langs->trans('SplitAllQuantity'), 'split', 'class="splitbutton splitallbutton field-error-icon" data-max-qty="1" onClick="addDispatchLine('.$line->id.', \'batch\', \'allmissingconsume\')"'); } print '
'; print '
'; // default warehouse processing print ''; if (in_array($action, array('consumeorproduce', 'consumeandproduceall')) && getDolGlobalString('STOCK_CONSUMPTION_FROM_MANUFACTURING_WAREHOUSE')) { print ''; } // Lines to produce print '
'; print '
'; print '
'; $nblinetoproduce = 0; $atLeastOneEatBy = false; $atLeastOneSellBy = false; foreach ($object->lines as $line) { if ($line->role == 'toproduce') { $tmpproduct = new Product($db); $tmpproduct->fetch($line->fk_product); if ( $tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_EAT_BY || $tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_SELL_AND_EAT ) { $atLeastOneEatBy = true; } if ( $tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_SELL_BY || $tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_SELL_AND_EAT ) { $atLeastOneSellBy = true; } $nblinetoproduce++; } } $newcardbutton = ''; $url = $_SERVER["PHP_SELF"].'?id='.$object->id.'&action=addproduceline&token='.newToken(); $permissiontoaddaproductline = $object->status != $object::STATUS_PRODUCED && $object->status != $object::STATUS_CANCELED; $parameters = array('morecss' => 'reposition'); if ($action != 'consumeorproduce' && $action != 'consumeandproduceall') { if ($nblinetoproduce == 0 || $object->mrptype == 1) { $newcardbutton = dolGetButtonTitle($langs->trans('AddNewProduceLines'), '', 'fa fa-plus-circle size15x', $url, '', (int) $permissiontoaddaproductline, $parameters); } } print load_fiche_titre($langs->trans('Production'), $newcardbutton, '', 0, '', ''); print '
'; print ''; print ''; // Product print ''; // Qty print ''; /// Unit if (getDolGlobalInt('PRODUCT_USE_UNITS')) { print ''; } // Cost price if ($permissiontoupdatecost) { if (empty($bomcostupdated)) { print ''; } else { print ''; } } // Already produced print ''; // Warehouse print ''; // Lot if (isModEnabled('productbatch')) { print ''; // Split print ''; // Split All print ''; // sell by if ($atLeastOneEatBy) { print ''; } // eat by if ($atLeastOneSellBy) { print ''; } } // Action delete if ($permissiontodelete) { print ''; } print ''; if ($action == 'addproduceline') { print ''."\n"; print ''; // Product print ''; // Qty print ''; //Unit if (getDolGlobalInt('PRODUCT_USE_UNITS')) { print ''; } // Cost price if ($permissiontoupdatecost) { print ''; } // Action (cost price + already produced) print ''; // Lot - serial if (isModEnabled('productbatch')) { print ''; // Split print ''; // Split All print ''; } // Action delete if ($permissiontodelete) { print ''; } print ''; } if (!empty($object->lines)) { $nblinetoproduce = 0; foreach ($object->lines as $line) { if ($line->role == 'toproduce') { $nblinetoproduce++; } } $nblinetoproducecursor = 0; foreach ($object->lines as $line) { if ($line->role == 'toproduce') { $i = 1; $nblinetoproducecursor++; $tmpproduct = new Product($db); $tmpproduct->fetch($line->fk_product); $arrayoflines = $object->fetchLinesLinked('produced', $line->id); $alreadyproduced = 0; foreach ($arrayoflines as $line2) { $alreadyproduced += $line2['qty']; } $suffix = '_'.$line->id; print ''."\n"; // hidden fields for js function print ''; print ''; print ''; // Product print ''; // Qty print ''; // Unit if (getDolGlobalInt('PRODUCT_USE_UNITS')) { print ''; } // Cost price if ($permissiontoupdatecost) { // Defined $manufacturingcost $manufacturingcost = 0; $manufacturingcostsrc = ''; if ($object->mrptype == 0) { // If MO is a "Manufacture" type (and not "Disassemble") $manufacturingcost = $bomcostupdated; $manufacturingcostsrc = $langs->trans("CalculatedFromProductsToConsume"); if (empty($manufacturingcost)) { $manufacturingcost = $bomcost; $manufacturingcostsrc = $langs->trans("ValueFromBom"); } if (empty($manufacturingcost)) { $manufacturingcost = price2num($tmpproduct->cost_price, 'MU'); $manufacturingcostsrc = $langs->trans("CostPrice"); } if (empty($manufacturingcost)) { $manufacturingcost = price2num($tmpproduct->pmp, 'MU'); $manufacturingcostsrc = $langs->trans("PMPValue"); } } print ''; } // Already produced print ''; // Warehouse print ''; // Lot if (isModEnabled('productbatch')) { print ''; // Split print ''; // Split All print ''; } // Delete if ($permissiontodelete) { if ($line->origin_type == 'free') { $href = $_SERVER["PHP_SELF"]; $href .= '?id='.$object->id; $href .= '&action=deleteline'; $href .= '&token='.newToken(); $href .= '&lineid='.$line->id; print ''; } else { print ''; } } print ''; // Show detailed of already consumed with js code to collapse foreach ($arrayoflines as $line2) { print ''; // Product print ''; // Qty print ''; // Unit if (getDolGlobalInt('PRODUCT_USE_UNITS')) { print ''; } // Cost price if ($permissiontoupdatecost) { print ''; } // Already produced print ''; // Warehouse print ''; // Lot if (isModEnabled('productbatch')) { print ''; // Split print ''; // Split All print ''; } // Action delete if ($permissiontodelete) { print ''; } print ''; } if (in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print ''."\n"; $maxQty = 1; print ''; // Product print ''; $preselected = (GETPOSTISSET('qtytoproduce-'.$line->id.'-'.$i) ? GETPOST('qtytoproduce-'.$line->id.'-'.$i) : max(0, $object->qty - $alreadyproduced)); // Qty print ''; //Unit if (getDolGlobalInt('PRODUCT_USE_UNITS')) { print ''; } // Cost if ($permissiontoupdatecost) { // Defined $manufacturingcost $manufacturingcost = 0; $manufacturingcostsrc = ''; if ($object->mrptype == 0) { // If MO is a "Manufacture" type (and not "Disassemble") $manufacturingcost = $bomcostupdated; $manufacturingcostsrc = $langs->trans("CalculatedFromProductsToConsume"); if (empty($manufacturingcost)) { $manufacturingcost = $bomcost; $manufacturingcostsrc = $langs->trans("ValueFromBom"); } if (empty($manufacturingcost)) { $manufacturingcost = price2num($tmpproduct->cost_price, 'MU'); $manufacturingcostsrc = $langs->trans("CostPrice"); } if (empty($manufacturingcost)) { $manufacturingcost = price2num($tmpproduct->pmp, 'MU'); $manufacturingcostsrc = $langs->trans("PMPValue"); } } if ($tmpproduct->type == Product::TYPE_PRODUCT || getDolGlobalString('STOCK_SUPPORTS_SERVICES')) { $preselected = (GETPOSTISSET('pricetoproduce-'.$line->id.'-'.$i) ? GETPOST('pricetoproduce-'.$line->id.'-'.$i) : ($manufacturingcost ? price($manufacturingcost) : '')); print ''; } else { print ''; } } // Already produced print ''; // Warehouse print ''; // Lot if (isModEnabled('productbatch')) { print ''; // Batch number in same column than the stock movement picto if ($tmpproduct->status_batch) { $type = 'batch'; print ''; print ''; } else { print ''; print ''; } } // sell by mandatory if ($tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_SELL_BY || $tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_SELL_AND_EAT) { print ''; } else { if ($atLeastOneSellBy) { print ''; } } // eat by mandatory if ($tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_EAT_BY || $tmpproduct->sell_or_eat_by_mandatory == $tmpproduct::SELL_OR_EAT_BY_MANDATORY_ID_SELL_AND_EAT) { print ''; } else { if ($atLeastOneEatBy) { print ''; } } // Action delete print ''; print ''; } } } } print '
'.$langs->trans("Product").''.$langs->trans("Qty").''.$langs->trans("Unit").''; print $langs->trans("UnitCost"); print ''; print $langs->trans("ManufacturingPrice"); print ''; print $langs->trans("QtyAlreadyProducedShort"); print ''; if ($collapse || in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print $langs->trans("Warehouse"); } print ''; if ($collapse || in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { print $langs->trans("Batch"); } print ''.$langs->trans("EatByDate").''.$langs->trans("SellByDate").'
'; print $form->select_produits(0, 'productidtoadd', '', 0, 0, -1, 2, '', 1, array(), 0, '1', 0, 'maxwidth300'); print ''; print ''; print ''; print '
'.$tmpproduct->getNomUrl(1); print '
'.$tmpproduct->label.''; print '
'.$line->qty.''.measuringUnitString($line->fk_unit, '', null, 1).''; if ($manufacturingcost) { print price($manufacturingcost); } print ''; if ($alreadyproduced) { print ''; if (empty($conf->use_javascript_ajax)) { print 'id.'">'; } print img_picto($langs->trans("ShowDetails"), "chevron-down", 'id="expandtoproduce'.$line->id.'"'); if (empty($conf->use_javascript_ajax)) { print ''; } } print ' '.$alreadyproduced; print ''; print ''; print ''; print img_picto($langs->trans('TooltipDeleteAndRevertStockMovement'), "delete"); print ''; print '
'; $tmpstockmovement->id = $line2['fk_stock_movement']; print ''.img_picto($langs->trans("StockMovement"), 'movement', 'class="paddingright"').''; print dol_print_date($line2['date'], 'dayhour', 'tzuserrel'); print ''.$line2['qty'].''; if ($line2['fk_warehouse'] > 0) { $result = $tmpwarehouse->fetch($line2['fk_warehouse']); if ($result > 0) { print $tmpwarehouse->getNomUrl(1); } } print ''; if ($line2['batch'] != '') { $tmpbatch->fetch(0, $line2['fk_product'], $line2['batch']); print $tmpbatch->getNomUrl(1); } print '
'.$langs->trans("ToProduce").''; if ($tmpproduct->type == Product::TYPE_PRODUCT || getDolGlobalString('STOCK_SUPPORTS_SERVICES')) { $preselected = (GETPOSTISSET('idwarehousetoproduce-'.$line->id.'-'.$i) ? GETPOST('idwarehousetoproduce-'.$line->id.'-'.$i) : ($object->fk_warehouse > 0 ? $object->fk_warehouse : 'ifone')); print $formproduct->selectWarehouses($preselected, 'idwarehousetoproduce-'.$line->id.'-'.$i, '', 1, 0, $line->fk_product, '', 1, 0, array(), 'maxwidth200 csswarehouse_'.$line->id.'_'.$i); } else { print ''.$langs->trans("NoStockChangeOnServices").''; } print ''; if ($tmpproduct->status_batch) { $preselected = (GETPOSTISSET('batchtoproduce-'.$line->id.'-'.$i) ? GETPOST('batchtoproduce-'.$line->id.'-'.$i) : ''); print ''; } print ''; print img_picto($langs->trans('AddStockLocationLine'), 'split', 'class="splitbutton" onClick="addDispatchLine('.$line->id.', \''.$type.'\', \'qtymissing\')"'); print ''; if (($action == 'consumeorproduce' || $action == 'consumeandproduceall') && $tmpproduct->status_batch == 2) { print img_picto($langs->trans('SplitAllQuantity'), 'split', 'class="splitbutton splitallbutton field-error-icon" onClick="addDispatchLine('.$line->id.', \'batch\', \'alltoproduce\')"'); } // print ''; $preselectedSellBy = (GETPOSTISSET('sellby-' . $line->id . '-' . $i) ? GETPOSTDATE('sellby-' . $line->id . '-' . $i) : ''); print $form->selectDate($preselectedSellBy, 'sellby-' . $line->id . '-' . $i, 0, 0, 1, '', 1, 0); print ''; $preselectedEatBy = (GETPOSTISSET('eatby-' . $line->id . '-' . $i) ? GETPOSTDATE('eatby-' . $line->id . '-' . $i) : ''); print $form->selectDate($preselectedEatBy, 'eatby-' . $line->id . '-' . $i, 0, 0, 1, '', 1, 0); print '
'; print '
'; print '
'; print '
'; } if (in_array($action, array('consumeorproduce', 'consumeandproduceall', 'addconsumeline'))) { print "
\n"; } /* ------------------------------------------------------------------- * Operation Progress section (per-routing-step tracking, no stock ops) * ------------------------------------------------------------------- */ if ($object->id > 0 && !in_array($action, array('consumeorproduce', 'consumeandproduceall'))) { $object->fetchLines(); $tracker = new MoOpTracking($db); $operations = MoOpTracking::getOperationsFromMo($object); $cumuls = $tracker->getCumulByMo($object->id); // Pre-fetch all log entries (reused for the accordion below and for // building the "By" initials column). getLogByMo() returns rows ordered // DESC by rowid, so the first entry seen per workstation is the latest. $logs = $tracker->getLogByMo($object->id); // Build [fk_ws => initials] for the last user who saved each operation. $lastUserByWs = array(); if (!empty($logs)) { $_wsUidMap = array(); // [fk_ws => fk_user] — latest entry per ws foreach ($logs as $_le) { $_lws = (int) $_le['fk_ws']; if (!array_key_exists($_lws, $_wsUidMap)) { $_wsUidMap[$_lws] = (int) $_le['fk_user']; } } if (!empty($_wsUidMap)) { $_uids = array_unique(array_values($_wsUidMap)); $_sqlU = "SELECT rowid, firstname, lastname, login FROM " . MAIN_DB_PREFIX . "user"; $_sqlU .= " WHERE rowid IN (" . implode(',', array_map('intval', $_uids)) . ")"; $_resU = $db->query($_sqlU); $_uiCache = array(); if ($_resU) { while ($_uobj = $db->fetch_object($_resU)) { $_f = trim((string) ($_uobj->firstname ?? '')); $_l = trim((string) ($_uobj->lastname ?? '')); $_uiCache[(int) $_uobj->rowid] = ($_f !== '' || $_l !== '') ? strtoupper(substr($_f, 0, 1) . substr($_l, 0, 1)) : strtoupper(substr((string) ($_uobj->login ?? ''), 0, 2)); } $db->free($_resU); } foreach ($_wsUidMap as $_lws => $_uid) { $lastUserByWs[$_lws] = $_uiCache[$_uid] ?? '-'; } } } print '
'; print load_fiche_titre($langs->trans('MoOpTrackingTitle'), '', 'object_mrp'); if (empty($operations)) { print '

' . $langs->trans('MoOpTrackingNoOps') . '

'; } else { // MO is considered "started" only once it has reached INPROGRESS status // (i.e. the first CONSUME OR PRODUCE action has been confirmed). $moStarted = ($object->status == Mo::STATUS_INPROGRESS); $canEdit = $moStarted && $permissiontoproduce; if (!$moStarted) { print '

' . $langs->trans('MoOpTrackingLockedUntilStart') . '

'; } if ($canEdit) { print '
'; print ''; print ''; print ''; } print '
'; print ''; print ''; print ''; print ''; if ($canEdit) { print ''; print ''; } elseif ($moStarted) { // started but no permission — show columns anyway (future-proof) print ''; print ''; } print ''; print ''; foreach ($operations as $rank => $ws) { $cumul = (float) ($cumuls[$ws['fk_ws']] ?? 0); $pct = ($object->qty > 0) ? min(100, round($cumul / $object->qty * 100)) : 0; $csscolor = ($pct >= 100) ? ' style="color:green"' : ''; print ''; print ''; print ''; if ($canEdit) { if ($rank === 0) { // Rank-0 (FORMING) is auto-synced from FG produced — not manually editable. print ''; print ''; } else { print ''; print ''; } } elseif ($moStarted) { print ''; print ''; } print ''; print ''; } print '
' . $langs->trans('Operation') . '' . $langs->trans('MoOpTrackingCumul') . '' . $langs->trans('MoOpTrackingDelta') . '' . $langs->trans('Note') . '' . $langs->trans('MoOpTrackingDelta') . '' . $langs->trans('Note') . '' . $langs->trans('MoOpTrackingBy') . '
' . ($rank + 1) . ' '; print dol_escape_htmltag($ws['ws_label']) . ''; print price2num($cumul, 'MS'); if ($object->qty > 0) { print ' / ' . price2num($object->qty, 'MS') . ' (' . $pct . '%)'; } print '' . $langs->trans('MoOpTrackingAutoSync') . '' . dol_escape_htmltag($lastUserByWs[$ws['fk_ws']] ?? '-') . '
'; print '
'; if ($canEdit) { print '
'; print ''; print '
'; print '
'; } // Collapsible log of previous delta entries (last 10) // $logs was already fetched before the table above. if (!empty($logs)) { $logsToShow = array_slice($logs, 0, 10); print '
'; print '' . $langs->trans('MoOpTrackingLog') . ' (' . count($logs) . ')'; print '
'; print ''; print ''; print ''; print ''; print ''; print ''; foreach ($logsToShow as $entry) { $dcolor = ($entry['qty_delta'] >= 0) ? 'style="color:green"' : 'style="color:#c00"'; $_logInitials = isset($_uiCache[(int) $entry['fk_user']]) ? $_uiCache[(int) $entry['fk_user']] : '-'; print ''; print ''; print ''; print ''; print ''; print ''; print ''; } print '
' . $langs->trans('Date') . '' . $langs->trans('Operation') . '' . $langs->trans('MoOpTrackingDelta') . '' . $langs->trans('Note') . '' . $langs->trans('MoOpTrackingBy') . '
' . dol_print_date($db->jdate($entry['date_entry']), 'dayhour', 'tzuserrel') . '' . dol_escape_htmltag($entry['ws_label']) . '' . ($entry['qty_delta'] >= 0 ? '+' : '') . price2num($entry['qty_delta'], 'MS') . '' . dol_escape_htmltag((string) $entry['note']) . '' . dol_escape_htmltag($_logInitials) . '
'; } } print '
'; // fichecenter #mo-optracking } ?> close();