dolibarr 20.0.2
objectline_view.tpl.php
1<?php
2/* Copyright (C) 2010-2013 Regis Houssin <regis.houssin@inodbox.com>
3 * Copyright (C) 2010-2011 Laurent Destailleur <eldy@users.sourceforge.net>
4 * Copyright (C) 2012-2013 Christophe Battarel <christophe.battarel@altairis.fr>
5 * Copyright (C) 2012 Cédric Salvador <csalvador@gpcsolutions.fr>
6 * Copyright (C) 2012-2014 Raphaël Doursenaud <rdoursenaud@gpcsolutions.fr>
7 * Copyright (C) 2013 Florian Henry <florian.henry@open-concept.pro>
8 * Copyright (C) 2017 Juanjo Menent <jmenent@2byte.es>
9 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
10 *
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 3 of the License, or
14 * (at your option) any later version.
15 *
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program. If not, see <https://www.gnu.org/licenses/>.
23 *
24 * Need to have following variables defined:
25 * $object (invoice, order, ...)
26 * $conf
27 * $langs
28 * $forceall (0 by default, 1 for supplier invoices/orders)
29 * $element (used to test $user->hasRight($element, 'creer'))
30 * $permtoedit (used to replace test $user->hasRight($element, 'creer'))
31 * $inputalsopricewithtax (0 by default, 1 to also show column with unit price including tax)
32 * $object_rights->creer initialized from = $object->getRights()
33 * $disableedit, $disablemove, $disableremove
34 *
35 * $type, $text, $description, $line
36 */
37
42'@phan-var-force CommonObjectLine $line
43 @phan-var-force int $num
44 @phan-var-force CommonObject $this
45 @phan-var-force CommonObject $object';
46
47require_once DOL_DOCUMENT_ROOT.'/workstation/class/workstation.class.php';
48
49// Protection to avoid direct call of template
50if (empty($object) || !is_object($object)) {
51 print "Error, template page can't be called as URL";
52 exit(1);
53}
54
55
56global $filtertype;
57if (empty($filtertype)) {
58 $filtertype = 0;
59}
60
61
62global $forceall, $senderissupplier, $inputalsopricewithtax, $outputalsopricetotalwithtax, $langs;
63
64if (empty($dateSelector)) {
65 $dateSelector = 0;
66}
67if (empty($forceall)) {
68 $forceall = 0;
69}
70if (empty($senderissupplier)) {
71 $senderissupplier = 0;
72}
73if (empty($inputalsopricewithtax)) {
74 $inputalsopricewithtax = 0;
75}
76if (empty($outputalsopricetotalwithtax)) {
77 $outputalsopricetotalwithtax = 0;
78}
79
80// add html5 elements
81if ($filtertype == 1) {
82 $domData = ' data-element="'.$line->element.'service"';
83} else {
84 $domData = ' data-element="'.$line->element.'"';
85}
86
87$domData .= ' data-id="'.$line->id.'"';
88$domData .= ' data-qty="'.$line->qty.'"';
89$domData .= ' data-product_type="'.$line->product_type.'"';
90
91// Lines for extrafield
92$objectline = new BOMLine($object->db);
93
94$coldisplay = 0;
95print "<!-- BEGIN PHP TEMPLATE objectline_view.tpl.php -->\n";
96print '<tr id="row-'.$line->id.'" class="drag drop oddeven" '.$domData.' >';
97
98// Line nb
99if (getDolGlobalString('MAIN_VIEW_LINE_NUMBER')) {
100 print '<td class="linecolnum center">'.($i + 1).'</td>';
101 $coldisplay++;
102}
103
104// Product
105print '<td class="linecoldescription bomline minwidth300imp">';
106print '<div id="line_'.$line->id.'"></div>';
107$coldisplay++;
108$tmpproduct = new Product($object->db);
109$tmpproduct->fetch($line->fk_product);
110$tmpbom = new BOM($object->db);
111$res = $tmpbom->fetch($line->fk_bom_child);
112if ($tmpbom->id > 0) {
113 print $tmpproduct->getNomUrl(1);
114 print ' '.$langs->trans("or").' ';
115 print $tmpbom->getNomUrl(1);
116 print ' <a class="collapse_bom" id="collapse-'.$line->id.'" href="#">';
117 print(!getDolGlobalString('BOM_SHOW_ALL_BOM_BY_DEFAULT') ? img_picto('', 'folder') : img_picto('', 'folder-open'));
118 print '</a>';
119} else {
120 print $tmpproduct->getNomUrl(1);
121 print ' - '.$tmpproduct->label;
122}
123
124// Line extrafield
125if (!empty($extrafields)) {
126 $temps = $line->showOptionals($extrafields, 'view', array(), '', '', 1, 'line');
127 if (!empty($temps)) {
128 print '<div style="padding-top: 10px" id="extrafield_lines_area_'.$line->id.'" name="extrafield_lines_area_'.$line->id.'">';
129 print $temps;
130 print '</div>';
131 }
132}
133
134print '</td>';
135
136// Qty
137print '<td class="linecolqty nowrap right">';
138$coldisplay++;
139echo price($line->qty, 0, '', 0, 0); // Yes, it is a quantity, not a price, but we just want the formatting role of function price
140print '</td>';
141
142if ($filtertype != 1) {
143 if (getDolGlobalInt('PRODUCT_USE_UNITS')) { // For product, unit is shown only if option PRODUCT_USE_UNITS is on
144 print '<td class="linecoluseunit nowrap">';
145 $label = measuringUnitString($line->fk_unit, '', '', 1);
146 if ($label !== '') {
147 print $langs->trans($label);
148 }
149 print '</td>';
150 }
151
152 print '<td class="linecolqtyfrozen nowrap right">';
153 $coldisplay++;
154 echo $line->qty_frozen ? yn($line->qty_frozen) : '';
155 print '</td>';
156 print '<td class="linecoldisablestockchange nowrap right">';
157 $coldisplay++;
158 echo $line->disable_stock_change ? yn($line->disable_stock_change) : ''; // Yes, it is a quantity, not a price, but we just want the formatting role of function price
159 print '</td>';
160
161 print '<td class="linecolefficiency nowrap right">';
162 $coldisplay++;
163 echo $line->efficiency;
164 print '</td>';
165} else {
166 // Unit // For services, units are always enabled
167 print '<td class="linecolunit nowrap">';
168 $coldisplay++;
169
170 if (!empty($line->fk_unit)) {
171 require_once DOL_DOCUMENT_ROOT.'/core/class/cunits.class.php';
172 $unit = new CUnits($this->db);
173 $unit->fetch($line->fk_unit);
174 print(isset($unit->label) ? "&nbsp;".$langs->trans(ucwords($unit->label))."&nbsp;" : '');
175 }
176
177 print '</td>';
178
179 // Work station
180 if (isModEnabled('workstation')) {
181 $workstation = new Workstation($object->db);
182 $res = $workstation->fetch($line->fk_default_workstation);
183
184 print '<td class="linecolworkstation nowrap">';
185 $coldisplay++;
186 if ($res > 0) {
187 $unit = new CUnits($object->db);
188 $fk_defaultUnit = $unit->getUnitFromCode('h', 'short_label', 'time');
189 $nbPlannedHour = $unit->unitConverter($line->qty, $line->fk_unit, $fk_defaultUnit);
190 $line->total_cost = 0;
191 if ($workstation->thm_machine_estimated) $line->total_cost += $nbPlannedHour * $workstation->thm_machine_estimated;
192 if ($workstation->thm_operator_estimated) $line->total_cost += $nbPlannedHour * $workstation->thm_operator_estimated;
193 echo $workstation->getNomUrl(1);
194 }
195 print '</td>';
196 }
197}
198
199// Cost
200$total_cost = 0;
201
202$tmpbom->calculateCosts();
203print '<td id="costline_'.$line->id.'" class="linecolcost nowrap right">';
204$coldisplay++;
205if (!empty($line->fk_bom_child)) {
206 echo '<span class="amount">'.price($tmpbom->total_cost * (float) $line->qty).'</span>';
207} else {
208 echo '<span class="amount">'.price($line->total_cost).'</span>';
209}
210print '</td>';
211
212if ($this->status == 0 && ($object_rights->write) && $action != 'selectlines') {
213 print '<td class="linecoledit center">';
214 $coldisplay++;
215 if (($line->info_bits & 2) == 2 || !empty($disableedit)) {
216 } else {
217 print '<a class="editfielda reposition" href="'.$_SERVER["PHP_SELF"].'?id='.$this->id.'&action=editline&token='.newToken().'&lineid='.$line->id.'">'.img_edit().'</a>';
218 }
219 print '</td>';
220
221 print '<td class="linecoldelete center">';
222 $coldisplay++;
223 if (($line->fk_prev_id == null) && empty($disableremove)) {
224 //La suppression n'est autorisée que si il n'y a pas de ligne dans une précédente situation
225 print '<a class="reposition" href="'.$_SERVER["PHP_SELF"].'?id='.$this->id.'&action=deleteline&token='.newToken().'&lineid='.$line->id.'">';
226 print img_delete();
227 print '</a>';
228 }
229 print '</td>';
230
231 if ($num > 1 && $conf->browser->layout != 'phone' && empty($disablemove)) {
232 print '<td class="linecolmove tdlineupdown center">';
233 $coldisplay++;
234 if ($i > 0) {
235 print '<a class="lineupdown" href="'.$_SERVER["PHP_SELF"].'?id='.$this->id.'&action=up&token='.newToken().'&rowid='.$line->id.'">';
236 echo img_up('default', 0, 'imgupforline');
237 print '</a>';
238 }
239 if ($i < $num - 1) {
240 print '<a class="lineupdown" href="'.$_SERVER["PHP_SELF"].'?id='.$this->id.'&action=down&token='.newToken().'&rowid='.$line->id.'">';
241 echo img_down('default', 0, 'imgdownforline');
242 print '</a>';
243 }
244 print '</td>';
245 } else {
246 print '<td '.(($conf->browser->layout != 'phone' && empty($disablemove)) ? ' class="linecolmove tdlineupdown center"' : ' class="linecolmove center"').'></td>';
247 $coldisplay++;
248 }
249} else {
250 print '<td colspan="3"></td>';
251 $coldisplay = $coldisplay + 3;
252}
253
254if ($action == 'selectlines') {
255 print '<td class="linecolcheck center">';
256 print '<input type="checkbox" class="linecheckbox" name="line_checkbox['.($i + 1).']" value="'.$line->id.'" >';
257 print '</td>';
258}
259
260print '</tr>';
261
262// Select of all the sub-BOM lines
263// From this point to the end of the file, we only take care of sub-BOM lines
264$sql = 'SELECT rowid, fk_bom_child, fk_product, qty FROM '.MAIN_DB_PREFIX.'bom_bomline AS bl';
265$sql .= ' WHERE fk_bom ='. (int) $tmpbom->id;
266$resql = $object->db->query($sql);
267
268if ($resql) {
269 // Loop on all the sub-BOM lines if they exist
270 while ($obj = $object->db->fetch_object($resql)) {
271 $sub_bom_product = new Product($object->db);
272 $sub_bom_product->fetch($obj->fk_product);
273
274 $sub_bom = new BOM($object->db);
275 if (!empty($obj->fk_bom_child)) {
276 $sub_bom->fetch($obj->fk_bom_child);
277 }
278
279 $sub_bom_line = new BOMLine($object->db);
280 $sub_bom_line->fetch($obj->rowid);
281
282 //If hidden conf is set, we show directly all the sub-BOM lines
283 if (!getDolGlobalString('BOM_SHOW_ALL_BOM_BY_DEFAULT')) {
284 print '<tr style="display:none" class="sub_bom_lines" parentid="'.$line->id.'">';
285 } else {
286 print '<tr class="sub_bom_lines" parentid="'.$line->id.'">';
287 }
288
289 // Product OR BOM
290 print '<td style="padding-left: 5%" id="sub_bom_product_'.$sub_bom_line->id.'">';
291 if (!empty($obj->fk_bom_child)) {
292 print $sub_bom_product->getNomUrl(1);
293 print ' '.$langs->trans('or').' ';
294 print $sub_bom->getNomUrl(1);
295 } else {
296 print $sub_bom_product->getNomUrl(1);
297 print '</td>';
298 }
299
300 // Qty
301 $label = $sub_bom_product->getLabelOfUnit('long');
302 if ($sub_bom_line->qty_frozen > 0) {
303 print '<td class="linecolqty nowrap right" id="sub_bom_qty_'.$sub_bom_line->id.'">'.price($sub_bom_line->qty, 0, '', 0, 0).'</td>';
304 if (getDolGlobalString('PRODUCT_USE_UNITS')) {
305 print '<td class="linecoluseunit nowrap left">';
306 if ($label !== '') {
307 print $langs->trans($label);
308 }
309 print '</td>';
310 }
311 print '<td class="linecolqtyfrozen nowrap right" id="sub_bom_qty_frozen_'.$sub_bom_line->id.'">'.$langs->trans('Yes').'</td>';
312 } else {
313 print '<td class="linecolqty nowrap right" id="sub_bom_qty_'.$sub_bom_line->id.'">'.price($sub_bom_line->qty * (float) $line->qty, 0, '', 0, 0).'</td>';
314 if (getDolGlobalString('PRODUCT_USE_UNITS')) {
315 print '<td class="linecoluseunit nowrap left">';
316 if ($label !== '') {
317 print $langs->trans($label);
318 }
319 print '</td>';
320 }
321
322 print '<td class="linecolqtyfrozen nowrap right" id="sub_bom_qty_frozen_'.$sub_bom_line->id.'">&nbsp;</td>';
323 }
324
325 // Disable stock change
326 if ($sub_bom_line->disable_stock_change > 0) {
327 print '<td class="linecoldisablestockchange nowrap right" id="sub_bom_stock_change_'.$sub_bom_line->id.'">'.$sub_bom_line->disable_stock_change.'</td>';
328 } else {
329 print '<td class="linecoldisablestockchange nowrap right" id="sub_bom_stock_change_'.$sub_bom_line->id.'">&nbsp;</td>';
330 }
331
332 // Efficiency
333 print '<td class="linecolefficiency nowrap right" id="sub_bom_efficiency_'.$sub_bom_line->id.'">'.$sub_bom_line->efficiency.'</td>';
334
335 // Cost
336 if (!empty($sub_bom->id)) {
337 $sub_bom->calculateCosts();
338 print '<td class="linecolcost nowrap right" id="sub_bom_cost_'.$sub_bom_line->id.'"><span class="amount">'.price(price2num($sub_bom->total_cost * $sub_bom_line->qty * (float) $line->qty, 'MT')).'</span></td>';
339 $total_cost += $sub_bom->total_cost * $sub_bom_line->qty * (float) $line->qty;
340 } elseif ($sub_bom_product->type == Product::TYPE_SERVICE && isModEnabled('workstation') && !empty($sub_bom_product->fk_default_workstation)) {
341 //Convert qty to hour
342 $unit = measuringUnitString($sub_bom_line->fk_unit, '', '', 1);
343 $qty = convertDurationtoHour($sub_bom_line->qty, $unit);
344 $workstation = new Workstation($this->db);
345 $res = $workstation->fetch($sub_bom_product->fk_default_workstation);
346 if ($res > 0) {
347 $sub_bom_line->total_cost = (float) price2num($qty * ($workstation->thm_operator_estimated + $workstation->thm_machine_estimated), 'MT');
348 }
349
350 print '<td class="linecolcost nowrap right" id="sub_bom_cost_'.$sub_bom_line->id.'"><span class="amount">'.price(price2num($sub_bom_line->total_cost, 'MT')).'</span></td>';
351 $this->total_cost += $line->total_cost;
352 } elseif ($sub_bom_product->cost_price > 0) {
353 print '<td class="linecolcost nowrap right" id="sub_bom_cost_'.$sub_bom_line->id.'">';
354 print '<span class="amount">'.price(price2num($sub_bom_product->cost_price * $sub_bom_line->qty * (float) $line->qty, 'MT')).'</span></td>';
355 $total_cost += $sub_bom_product->cost_price * $sub_bom_line->qty * (float) $line->qty;
356 } elseif ($sub_bom_product->pmp > 0) { // PMP if cost price isn't defined
357 print '<td class="linecolcost nowrap right" id="sub_bom_cost_'.$sub_bom_line->id.'">';
358 print '<span class="amount">'.price(price2num($sub_bom_product->pmp * $sub_bom_line->qty * (float) $line->qty, 'MT')).'</span></td>';
359 $total_cost .= $sub_bom_product->pmp * $sub_bom_line->qty * (float) $line->qty;
360 } else { // Minimum purchase price if cost price and PMP aren't defined
361 $sql_supplier_price = "SELECT MIN(price) AS min_price, quantity AS qty FROM ".MAIN_DB_PREFIX."product_fournisseur_price";
362 $sql_supplier_price .= " WHERE fk_product = ". (int) $sub_bom_product->id;
363 $sql_supplier_price .= " GROUP BY quantity ORDER BY quantity ASC";
364 $resql_supplier_price = $object->db->query($sql_supplier_price);
365 if ($resql_supplier_price) {
366 $obj = $object->db->fetch_object($resql_supplier_price); // Take first value so the ref with the smaller minimum quantity
367 if (!empty($obj->qty) && !empty($sub_bom_line->qty) && !empty($line->qty)) {
368 $line_cost = $obj->min_price / $obj->qty * $sub_bom_line->qty * (float) $line->qty;
369 } else {
370 $line_cost = $obj->min_price;
371 }
372 print '<td class="linecolcost nowrap right" id="sub_bom_cost_'.$sub_bom_line->id.'"><span class="amount">'.price2num($line_cost, 'MT').'</span></td>';
373 $total_cost += $line_cost;
374 }
375 }
376
377 print '<td></td>';
378 print '<td></td>';
379 print '<td></td>';
380 }
381}
382
383
384print "<!-- END PHP TEMPLATE objectline_view.tpl.php -->\n";
if( $user->socid > 0) if(! $user->hasRight('accounting', 'chartofaccount')) $object
Definition card.php:58
print $langs trans("AuditedSecurityEvents").'</strong >< span class="opacitymedium"></span >< br > status
Or an array listing all the potential status of the object: array: int of the status => translated la...
Definition security.php:637
Class for BOM.
Definition bom.class.php:45
Class for BOMLine.
Class of dictionary type of thirdparty (used by imports)
Class to manage products or services.
const TYPE_SERVICE
Service.
Class for Workstation.
convertDurationtoHour($duration_value, $duration_unit)
Convert duration to hour.
Definition date.lib.php:334
img_delete($titlealt='default', $other='class="pictodelete"', $morecss='')
Show delete logo.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=0, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
yn($yesno, $case=1, $color=0)
Return yes or no in current language.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
img_down($titlealt='default', $selected=0, $moreclass='')
Show down arrow logo.
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'.
getDolGlobalString($key, $default='')
Return dolibarr global constant string value.
img_edit($titlealt='default', $float=0, $other='')
Show logo edit/modify fiche.
img_up($titlealt='default', $selected=0, $moreclass='')
Show top arrow logo.
measuringUnitString($unit, $measuring_style='', $scale='', $use_short_label=0, $outputlangs=null)
Return translation label of a unit key.