dolibarr 21.0.0-beta
bom_net_needs.php
Go to the documentation of this file.
1<?php
2/* Copyright (C) 2017-2020 Laurent Destailleur <eldy@users.sourceforge.net>
3 * Copyright (C) 2019-2024 Frédéric France <frederic.france@free.fr>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 3 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <https://www.gnu.org/licenses/>.
17 */
18
25// Load Dolibarr environment
26require '../main.inc.php';
27require_once DOL_DOCUMENT_ROOT.'/core/class/html.formcompany.class.php';
28require_once DOL_DOCUMENT_ROOT.'/core/class/html.formfile.class.php';
29require_once DOL_DOCUMENT_ROOT.'/bom/class/bom.class.php';
30require_once DOL_DOCUMENT_ROOT.'/bom/lib/bom.lib.php';
31
41// Load translation files required by the page
42$langs->loadLangs(array("mrp", "other", "stocks"));
43
44// Get parameters
45$id = GETPOSTINT('id');
46$lineid = GETPOSTINT('lineid');
47$ref = GETPOST('ref', 'alpha');
48$action = GETPOST('action', 'aZ09');
49$confirm = GETPOST('confirm', 'alpha');
50$cancel = GETPOST('cancel', 'aZ09');
51$contextpage = GETPOST('contextpage', 'aZ') ? GETPOST('contextpage', 'aZ') : 'bomnet_needs'; // To manage different context of search
52$backtopage = GETPOST('backtopage', 'alpha');
53
54
55// Initialize a technical objects
56$object = new BOM($db);
57$extrafields = new ExtraFields($db);
58
59// Initialize a technical objects for hooks
60$hookmanager->initHooks(array('bomnetneeds')); // Note that conf->hooks_modules contains array
61
62// Massaction
63$diroutputmassaction = $conf->bom->dir_output.'/temp/massgeneration/'.$user->id;
64
65// Fetch optionals attributes and labels
66$extrafields->fetch_name_optionals_label($object->table_element);
67$search_array_options = $extrafields->getOptionalsFromPost($object->table_element, '', 'search_');
68
69// Initialize array of search criteria
70$search_all = GETPOST("search_all", 'alpha');
71$search = array();
72foreach ($object->fields as $key => $val) {
73 if (GETPOST('search_'.$key, 'alpha')) {
74 $search[$key] = GETPOST('search_'.$key, 'alpha');
75 }
76}
77
78if (empty($action) && empty($id) && empty($ref)) {
79 $action = 'view';
80}
81
82// Load object
83include DOL_DOCUMENT_ROOT.'/core/actions_fetchobject.inc.php'; // Must be 'include', not 'include_once'.
84if ($object->id > 0) {
85 $object->calculateCosts();
86}
87
88
89// Security check - Protection if external user
90//if ($user->socid > 0) accessforbidden();
91//if ($user->socid > 0) $socid = $user->socid;
92$isdraft = (($object->status == $object::STATUS_DRAFT) ? 1 : 0);
93$result = restrictedArea($user, 'bom', $object->id, $object->table_element, '', '', 'rowid', $isdraft);
94
95// Permissions
96$permissionnote = $user->hasRight('bom', 'write'); // Used by the include of actions_setnotes.inc.php
97$permissiondellink = $user->hasRight('bom', 'write'); // Used by the include of actions_dellink.inc.php
98$permissiontoadd = $user->hasRight('bom', 'write'); // Used by the include of actions_addupdatedelete.inc.php and actions_lineupdown.inc.php
99$permissiontodelete = $user->hasRight('bom', 'delete') || ($permissiontoadd && isset($object->status) && $object->status == $object::STATUS_DRAFT);
100$upload_dir = $conf->bom->multidir_output[isset($object->entity) ? $object->entity : 1];
101
102
103/*
104 * Actions
105 */
106
107$parameters = array();
108$reshook = $hookmanager->executeHooks('doActions', $parameters, $object, $action); // Note that $action and $object may have been modified by some hooks
109if ($reshook < 0) {
110 setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
111}
112
113if (empty($reshook)) {
114 $error = 0;
115
116 $backurlforlist = DOL_URL_ROOT.'/bom/bom_list.php';
117
118 if (empty($backtopage) || ($cancel && empty($id))) {
119 if (empty($backtopage) || ($cancel && strpos($backtopage, '__ID__'))) {
120 if (empty($id) && (($action != 'add' && $action != 'create') || $cancel)) {
121 $backtopage = $backurlforlist;
122 } else {
123 $backtopage = DOL_URL_ROOT.'/bom/bom_net_needs.php?id='.($id > 0 ? $id : '__ID__');
124 }
125 }
126 }
127}
128
129
130/*
131 * View
132 */
133
134$form = new Form($db);
135$formfile = new FormFile($db);
136
137$title = $langs->trans('BOM');
138$help_url ='EN:Module_BOM';
139
140llxHeader('', $title, $help_url, '', 0, 0, '', '', '', 'mod-bom page-net_needs');
141
142
143$TChildBom = array();
144if ($action == 'treeview') {
145 $object->getNetNeedsTree($TChildBom, 1);
146} else {
147 $object->getNetNeeds($TChildBom, 1);
148}
149
150
151// Part to show record
152if ($object->id > 0 && (empty($action) || ($action != 'edit' && $action != 'create'))) {
153 $head = bomPrepareHead($object);
154 print dol_get_fiche_head($head, 'net_needs', $langs->trans("BillOfMaterials"), -1, 'bom');
155
156 $formconfirm = '';
157
158 // Call Hook formConfirm
159 $parameters = array('formConfirm' => $formconfirm, 'lineid' => $lineid);
160 $reshook = $hookmanager->executeHooks('formConfirm', $parameters, $object, $action); // Note that $action and $object may have been modified by hook
161 if (empty($reshook)) {
162 $formconfirm .= $hookmanager->resPrint;
163 } elseif ($reshook > 0) {
164 $formconfirm = $hookmanager->resPrint;
165 }
166
167 // Print form confirm
168 print $formconfirm;
169
170
171 // Object card
172 // ------------------------------------------------------------
173 $linkback = '<a href="'.DOL_URL_ROOT.'/bom/bom_list.php?restore_lastsearch_values=1'.(!empty($socid) ? '&socid='.$socid : '').'">'.$langs->trans("BackToList").'</a>';
174
175 $morehtmlref = '<div class="refidno">';
176
177 $morehtmlref .= '</div>';
178
179
180 dol_banner_tab($object, 'ref', $linkback, 1, 'ref', 'ref', $morehtmlref);
181
182
183 print '<div class="fichecenter">';
184 print '<div class="fichehalfleft">';
185 print '<div class="underbanner clearboth"></div>';
186 print '<table class="border centpercent tableforfield">'."\n";
187
188 // Common attributes
189 $keyforbreak = 'duration';
190 include DOL_DOCUMENT_ROOT.'/core/tpl/commonfields_view.tpl.php';
191
192 // Manufacturing cost
193 print '<tr><td>'.$form->textwithpicto($langs->trans("ManufacturingCost"), $langs->trans("BOMTotalCost")).'</td><td><span class="amount">';
194 print price($object->total_cost);
195 print '</span>';
196 if ($object->total_cost != $object->unit_cost) {
197 print '&nbsp; &nbsp; <span class="opacitymedium">('.$form->textwithpicto(price($object->unit_cost), $langs->trans("ManufacturingUnitCost"), 1, 'help', '').')</span>';
198 }
199 print '</td></tr>';
200
201 // Find sell price of generated product. We suppose we sell it to a company like ours (same country...).
202 $object->fetch_product();
203 $manufacturedvalued = '';
204 if (!empty($object->product)) {
205 global $mysoc;
206 $tmparray = $object->product->getSellPrice($mysoc, $mysoc);
207 $manufacturedvalued = $tmparray['pu_ht'] * $object->qty;
208 }
209 print '<tr><td>'.$langs->trans("ManufacturingGeneratedValue").'</td><td>'.price($manufacturedvalued).'</td></tr>';
210
211 // Other attributes
212 include DOL_DOCUMENT_ROOT.'/core/tpl/extrafields_view.tpl.php';
213
214 print '</table>';
215 print '</div>';
216 print '</div>';
217
218 print '<div class="clearboth"></div>';
219
220 print dol_get_fiche_end();
221
222 $viewlink = dolGetButtonTitle($langs->trans('GroupByX', $langs->transnoentitiesnoconv("Products")), '', 'fa fa-bars imgforviewmode', $_SERVER['PHP_SELF'].'?id='.$object->id.'&token='.newToken(), '', 1, array('morecss' => 'reposition '.($action !== 'treeview' ? 'btnTitleSelected' : '')));
223 $viewlink .= dolGetButtonTitle($langs->trans('TreeView'), '', 'fa fa-stream imgforviewmode', $_SERVER['PHP_SELF'].'?id='.$object->id.'&action=treeview&token='.newToken(), '', 1, array('morecss' => 'reposition marginleftonly '.($action == 'treeview' ? 'btnTitleSelected' : '')));
224
225 print load_fiche_titre($langs->trans("BOMNetNeeds"), $viewlink, 'product');
226
227 /*
228 * Lines
229 */
230 $text_stock_options = $langs->trans("RealStockDesc").'<br>';
231 $text_stock_options .= $langs->trans("RealStockWillAutomaticallyWhen").'<br>';
232 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_SHIPMENT') || getDolGlobalString('STOCK_CALCULATE_ON_SHIPMENT_CLOSE') ? '- '.$langs->trans("DeStockOnShipment").'<br>' : '');
233 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_VALIDATE_ORDER') ? '- '.$langs->trans("DeStockOnValidateOrder").'<br>' : '');
234 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_BILL') ? '- '.$langs->trans("DeStockOnBill").'<br>' : '');
235 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_SUPPLIER_BILL') ? '- '.$langs->trans("ReStockOnBill").'<br>' : '');
236 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_SUPPLIER_VALIDATE_ORDER') ? '- '.$langs->trans("ReStockOnValidateOrder").'<br>' : '');
237 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_SUPPLIER_DISPATCH_ORDER') ? '- '.$langs->trans("ReStockOnDispatchOrder").'<br>' : '');
238 $text_stock_options .= (getDolGlobalString('STOCK_CALCULATE_ON_RECEPTION') || getDolGlobalString('STOCK_CALCULATE_ON_RECEPTION_CLOSE') ? '- '.$langs->trans("StockOnReception").'<br>' : '');
239
240 print '<table id="tablelines" class="noborder noshadow" width="100%">';
241 print "<thead>\n";
242 print '<tr class="liste_titre nodrag nodrop">';
243 print '<td class="linecoldescription">'.$langs->trans('Product');
244 if (getDolGlobalString('BOM_SUB_BOM') && $action == 'treeview') {
245 print ' &nbsp; <a id="show_all" href="#">'.img_picto('', 'folder-open', 'class="paddingright"').$langs->trans("ExpandAll").'</a>&nbsp;&nbsp;';
246 print '<a id="hide_all" href="#">'.img_picto('', 'folder', 'class="paddingright"').$langs->trans("UndoExpandAll").'</a>&nbsp;';
247 }
248 print '</td>';
249 if ($action == 'treeview') {
250 print '<td class="left">'.$langs->trans('ProducedBy').'</td>';
251 }
252 print '<td class="linecolqty right">'.$langs->trans('Quantity').'</td>';
253 print '<td></td>'; // For unit
254 print '<td class="linecolstock right">'.$form->textwithpicto($langs->trans("PhysicalStock"), $text_stock_options, 1).'</td>';
255 print '<td class="linecoltheoricalstock right">'.$form->textwithpicto($langs->trans("VirtualStock"), $langs->trans("VirtualStockDesc")).'</td>';
256 print '</tr>';
257
258 print '</thead>';
259 print '<tbody>';
260 if (count($TChildBom) > 0) {
261 if ($action == 'treeview') {
262 foreach ($TChildBom as $fk_bom => $TProduct) {
263 $repeatChar = '&emsp;';
264 if (!empty($TProduct['bom'])) {
265 $prod = new Product($db);
266 $prod->fetch($TProduct['bom']->fk_product);
267 if ($TProduct['parentid'] != $object->id) {
268 print '<tr class="sub_bom_lines oddeven" parentid="'.$TProduct['parentid'].'">';
269 } else {
270 print '<tr class="oddeven">';
271 }
272 if ($action == 'treeview') {
273 print '<td class="linecoldescription">'.str_repeat($repeatChar, $TProduct['level']).$prod->getNomUrl(1);
274 } else {
275 print '<td class="linecoldescription">'.str_repeat($repeatChar, $TProduct['level']).$TProduct['bom']->getNomUrl(1);
276 }
277 print ' <a class="collapse_bom" id="collapse-'.$fk_bom.'" href="#">';
278 print img_picto('', 'folder-open');
279 print '</a>';
280 print '</td>';
281 if ($action == 'treeview') {
282 print '<td class="left">'.$TProduct['bom']->getNomUrl(1).'</td>';
283 }
284 print '<td class="linecolqty right">'.$TProduct['qty'].'</td>';
285 print '<td>';
286 print '</td>';
287 print '<td class="linecolstock right"></td>';
288 print '<td class="linecoltheoricalstock right"></td>';
289 print '</tr>';
290 }
291 if (!empty($TProduct['product'])) {
292 foreach ($TProduct['product'] as $fk_product => $TInfos) {
293 $prod = new Product($db);
294 $prod->fetch($fk_product);
295 $prod->load_virtual_stock();
296 if (empty($prod->stock_reel)) {
297 $prod->stock_reel = 0;
298 }
299 if ($fk_bom != $object->id) {
300 print '<tr class="sub_bom_lines oddeven" parentid="'.$fk_bom.'">';
301 } else {
302 print '<tr class="oddeven">';
303 }
304 print '<td class="linecoldescription">'.str_repeat($repeatChar, $TInfos['level']).$prod->getNomUrl(1).'</td>';
305 if ($action == 'treeview') {
306 print '<td></td>';
307 }
308 print '<td class="linecolqty right">'.$TInfos['qty'].'</td>';
309 print '<td>';
310 print '</td>';
311 print '<td class="linecolstock right">'.price2num($prod->stock_reel, 'MS').'</td>';
312 print '<td class="linecoltheoricalstock right">'.$prod->stock_theorique.'</td>';
313 print '</tr>';
314 }
315 }
316 }
317 } else {
318 foreach ($TChildBom as $fk_product => $elem) {
319 $prod = new Product($db);
320 $prod->fetch($fk_product);
321 $prod->load_virtual_stock();
322 if (empty($prod->stock_reel)) {
323 $prod->stock_reel = 0;
324 }
325 print '<tr class="oddeven">';
326 print '<td class="linecoldescription">'.$prod->getNomUrl(1).'</td>';
327 print '<td class="linecolqty right">'.$elem['qty'].'</td>';
328 print '<td>';
329 $useunit = (($prod->type == Product::TYPE_PRODUCT && getDolGlobalInt('PRODUCT_USE_UNITS')) || (($prod->type == Product::TYPE_SERVICE) && ($elem['fk_unit'])));
330 if ($useunit) {
331 require_once DOL_DOCUMENT_ROOT.'/core/class/cunits.class.php';
332 $unit = new CUnits($db);
333 $unit->fetch($elem['fk_unit']);
334 print(isset($unit->label) ? "&nbsp;".$langs->trans(ucwords($unit->label))."&nbsp;" : '');
335 }
336 print '</td>';
337 print '<td class="linecolstock right">'.price2num($prod->stock_reel, 'MS').'</td>';
338 print '<td class="linecoltheoricalstock right">'.$prod->stock_theorique.'</td>';
339 print '</tr>';
340 }
341 }
342 }
343 print '</tbody>';
344 print '</table>';
345
346
347
348 /*
349 * ButAction
350 */
351 print '<div class="tabsAction">'."\n";
352 $parameters = array();
353 $reshook = $hookmanager->executeHooks('addMoreActionsButtons', $parameters, $object, $action); // Note that $action and $object may have been modified by hook
354 if ($reshook < 0) {
355 setEventMessages($hookmanager->error, $hookmanager->errors, 'errors');
356 }
357 print '</div>'; ?>
358
359 <script type="text/javascript" language="javascript">
360 $(document).ready(function() {
361
362 function folderManage(element) {
363 var id_bom_line = element.attr('id').replace('collapse-', '');
364 let TSubLines = $('[parentid="'+ id_bom_line +'"]');
365
366 if(element.html().indexOf('folder-open') <= 0) {
367 $('[parentid="'+ id_bom_line +'"]').show();
368 element.html('<?php echo dol_escape_js(img_picto('', 'folder-open')); ?>');
369 }
370 else {
371 for (let i = 0; i < TSubLines.length; i++) {
372 let subBomFolder = $(TSubLines[i]).children('.linecoldescription').children('.collapse_bom');
373 if (subBomFolder.length > 0) {
374 folderManage(subBomFolder);
375 }
376 }
377 TSubLines.hide();
378 element.html('<?php echo dol_escape_js(img_picto('', 'folder')); ?>');
379 }
380 }
381
382 // When clicking on collapse
383 $(".collapse_bom").click(function() {
384 folderManage($(this));
385 return false;
386 });
387
388 // To Show all the sub bom lines
389 $("#show_all").click(function() {
390 console.log("We click on show all");
391 $("[class^=sub_bom_lines]").show();
392 $("[class^=collapse_bom]").html('<?php echo dol_escape_js(img_picto('', 'folder-open')); ?>');
393 return false;
394 });
395
396 // To Hide all the sub bom lines
397 $("#hide_all").click(function() {
398 console.log("We click on hide all");
399 $("[class^=sub_bom_lines]").hide();
400 $("[class^=collapse_bom]").html('<?php echo dol_escape_js(img_picto('', 'folder')); ?>');
401 return false;
402 });
403
404 });
405 </script>
406
407 <?php
408}
409
410// End of page
411llxFooter();
412$db->close();
$id
Definition account.php:48
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Definition card.php:66
llxFooter($comment='', $zone='private', $disabledoutputofmessages=0)
Empty footer.
Definition wrapper.php:87
if(!defined('NOREQUIRESOC')) if(!defined( 'NOREQUIRETRAN')) if(!defined('NOTOKENRENEWAL')) if(!defined( 'NOREQUIREMENU')) if(!defined('NOREQUIREHTML')) if(!defined( 'NOREQUIREAJAX')) llxHeader($head='', $title='', $help_url='', $target='', $disablejs=0, $disablehead=0, $arrayofjs='', $arrayofcss='', $morequerystring='', $morecssonbody='', $replacemainareaby='', $disablenofollow=0, $disablenoindex=0)
Empty header.
Definition wrapper.php:71
bomPrepareHead($object)
Prepare array of tabs for BillOfMaterials.
Definition bom.lib.php:79
Class for BOM.
Definition bom.class.php:42
Class of dictionary type of thirdparty (used by imports)
Class to manage standard extra fields.
Class to offer components to list and upload files.
Class to manage generation of HTML components Only common components must be here.
Class to manage products or services.
const TYPE_PRODUCT
Regular product.
const TYPE_SERVICE
Service.
load_fiche_titre($title, $morehtmlright='', $picto='generic', $pictoisfullpath=0, $id='', $morecssontable='', $morehtmlcenter='')
Load a title with picto.
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='', $noduplicate=0, $attop=0)
Set event messages in dol_events session object.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
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.
dol_get_fiche_head($links=array(), $active='', $title='', $notab=0, $picto='', $pictoisfullpath=0, $morehtmlright='', $morecss='', $limittoshow=0, $moretabssuffix='', $dragdropfile=0)
Show tabs of a record.
dolGetButtonTitle($label, $helpText='', $iconClass='fa fa-file', $url='', $id='', $status=1, $params=array())
Function dolGetButtonTitle : this kind of buttons are used in title in list.
dol_get_fiche_end($notab=0)
Return tab footer of a card.
price($amount, $form=0, $outlangs='', $trunc=1, $rounding=-1, $forcerounding=-1, $currency_code='')
Function to format a value into an amount for visual output Function used into PDF and HTML pages.
getDolGlobalInt($key, $default=0)
Return a Dolibarr global constant int value.
newToken()
Return the value of token currently saved into session with name 'newtoken'.
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
getDolGlobalString($key, $default='')
Return a Dolibarr global constant string value.
global $conf
The following vars must be defined: $type2label $form $conf, $lang, The following vars may also be de...
Definition member.php:79
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition repair.php:149
restrictedArea(User $user, $features, $object=0, $tableandshare='', $feature2='', $dbt_keyfield='fk_soc', $dbt_select='rowid', $isdraft=0, $mode=0)
Check permissions of a user to show a page and an object.