dolibarr  x.y.z
combinations.php
1 <?php
2 /* Copyright (C) 2016 Marcos García <marcosgdf@gmail.com>
3  * Copyright (C) 2017 Laurent Destailleur <eldy@users.sourceforge.net>
4  * Copyright (C) 2018-2019 Frédéric France <frederic.france@netlogic.fr>
5  * Copyright (C) 2022 Open-Dsi <support@open-dsi.fr>
6  *
7  * This program is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 3 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program. If not, see <https://www.gnu.org/licenses/>.
19  */
20 
21 // Load Dolibarr environment
22 require '../main.inc.php';
23 require_once DOL_DOCUMENT_ROOT.'/core/lib/product.lib.php';
24 require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php';
25 require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductAttribute.class.php';
26 require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductAttributeValue.class.php';
27 require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductCombination.class.php';
28 require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductCombination2ValuePair.class.php';
29 
30 $langs->loadLangs(array("products", "other"));
31 
32 $id = GETPOST('id', 'int');
33 $valueid = GETPOST('valueid', 'int');
34 $ref = GETPOST('ref', 'alpha');
35 $weight_impact = price2num(GETPOST('weight_impact', 'alpha'), 2);
36 $price_impact_percent = (bool) GETPOST('price_impact_percent');
37 if ($price_impact_percent) {
38  $price_impact = price2num(GETPOST('price_impact', 'alpha'), 2);
39 } else {
40  $price_impact = price2num(GETPOST('price_impact', 'alpha'), 'MU');
41 }
42 $level_price_impact = GETPOST('level_price_impact', 'array');
43 $level_price_impact_percent = GETPOST('level_price_impact_percent', 'array');
44 
45 $reference = GETPOST('reference', 'alpha');
46 $form = new Form($db);
47 
48 $action = GETPOST('action', 'aZ09');
49 $massaction = GETPOST('massaction', 'alpha');
50 $show_files = GETPOST('show_files', 'int');
51 $confirm = GETPOST('confirm', 'alpha');
52 $toselect = GETPOST('toselect', 'array');
53 $cancel = GETPOST('cancel', 'alpha');
54 $delete_product = GETPOST('delete_product', 'alpha');
55 $subaction = GETPOST('subaction', 'aZ09');
56 $backtopage = GETPOST('backtopage', 'alpha');
57 $sortfield = GETPOST('sortfield', 'aZ09comma');
58 $sortorder = GETPOST('sortorder', 'aZ09comma');
59 
60 // Security check
61 $fieldvalue = (!empty($id) ? $id : $ref);
62 $fieldtype = (!empty($ref) ? 'ref' : 'rowid');
63 
64 $prodstatic = new Product($db);
65 $prodattr = new ProductAttribute($db);
66 $prodattr_val = new ProductAttributeValue($db);
67 
68 $object = new Product($db);
69 if ($id > 0 || $ref) {
70  $object->fetch($id, $ref);
71 }
72 
73 $selectedvariant = !empty($_SESSION['addvariant_'.$object->id]) ? $_SESSION['addvariant_'.$object->id] : array();
74 $selected = "";
75 // Security check
76 if (!isModEnabled('variants')) {
77  accessforbidden('Module not enabled');
78 }
79 if ($user->socid > 0) { // Protection if external user
81 }
82 
83 if ($object->id > 0) {
84  if ($object->type == $object::TYPE_PRODUCT) {
85  restrictedArea($user, 'produit', $object->id, 'product&product', '', '');
86  }
87  if ($object->type == $object::TYPE_SERVICE) {
88  restrictedArea($user, 'service', $object->id, 'product&product', '', '');
89  }
90 } else {
91  restrictedArea($user, 'produit|service', $fieldvalue, 'product&product', '', '', $fieldtype);
92 }
93 $usercanread = (($object->type == Product::TYPE_PRODUCT && $user->rights->produit->lire) || ($object->type == Product::TYPE_SERVICE && $user->rights->service->lire));
94 $usercancreate = (($object->type == Product::TYPE_PRODUCT && $user->rights->produit->creer) || ($object->type == Product::TYPE_SERVICE && $user->rights->service->creer));
95 $usercandelete = (($object->type == Product::TYPE_PRODUCT && $user->rights->produit->supprimer) || ($object->type == Product::TYPE_SERVICE && $user->rights->service->supprimer));
96 
97 
98 /*
99  * Actions
100  */
101 
102 if ($cancel) {
103  $action = '';
104  $massaction = '';
105  unset($_SESSION['addvariant_'.$object->id]);
106 }
107 
108 if (!$object->isProduct() && !$object->isService()) {
109  header('Location: '.dol_buildpath('/product/card.php?id='.$object->id, 2));
110  exit();
111 }
112 if ($action == 'add') {
113  unset($selectedvariant);
114  unset($_SESSION['addvariant_'.$object->id]);
115 }
116 if ($action == 'create' && GETPOST('selectvariant', 'alpha')) { // We click on select combination
117  $action = 'add';
118  $attribute_id = GETPOST('attribute', 'int');
119  $attribute_value_id = GETPOST('value', 'int');
120  if ($attribute_id> 0 && $attribute_value_id > 0) {
121  $feature = $attribute_id . '-' . $attribute_value_id;
122  $selectedvariant[$feature] = $feature;
123  $_SESSION['addvariant_'.$object->id] = $selectedvariant;
124  }
125 }
126 if ($action == 'create' && $subaction == 'delete') { // We click on select combination
127  $action = 'add';
128  $feature = GETPOST('feature', 'intcomma');
129  if (isset($selectedvariant[$feature])) {
130  unset($selectedvariant[$feature]);
131  $_SESSION['addvariant_'.$object->id] = $selectedvariant;
132  }
133 }
134 
135 
136 $prodcomb = new ProductCombination($db);
137 $prodcomb2val = new ProductCombination2ValuePair($db);
138 
139 $productCombination2ValuePairs1 = array();
140 
141 if (($action == 'add' || $action == 'create') && empty($massaction) && !GETPOST('selectvariant', 'alpha') && empty($subaction)) { // We click on Create all defined combinations
142  //$features = GETPOST('features', 'array');
143  $features = !empty($_SESSION['addvariant_'.$object->id]) ? $_SESSION['addvariant_'.$object->id] : array();
144 
145  if (!$features) {
146  if ($action == 'create') {
147  setEventMessages($langs->trans('ErrorFieldsRequired'), null, 'errors');
148  }
149  } else {
150  $reference = trim($reference);
151  if (empty($reference)) {
152  $reference = false;
153  }
154  $weight_impact = price2num($weight_impact);
155  $price_impact = price2num($price_impact);
156 
157  // for conf PRODUIT_MULTIPRICES
158  if (!empty($conf->global->PRODUIT_MULTIPRICES)) {
159  $level_price_impact = array_map('price2num', $level_price_impact);
160  } else {
161  $level_price_impact = array(1 => $price_impact);
162  $level_price_impact_percent = array(1 => $price_impact_percent);
163  }
164 
165  $sanit_features = array();
166 
167  //First, sanitize
168  foreach ($features as $feature) {
169  $explode = explode('-', $feature);
170  if ($prodattr->fetch($explode[0]) <= 0 || $prodattr_val->fetch($explode[1]) <= 0) {
171  continue;
172  }
173 
174  // Valuepair
175  $sanit_features[$explode[0]] = $explode[1];
176 
177  $tmp = new ProductCombination2ValuePair($db);
178  $tmp->fk_prod_attr = $explode[0];
179  $tmp->fk_prod_attr_val = $explode[1];
180 
181  $productCombination2ValuePairs1[] = $tmp;
182  }
183 
184  $db->begin();
185 
186  // sanit_feature is an array with 1 (and only 1) value per attribute.
187  // For example: Color->blue, Size->Small, Option->2
188  //var_dump($sanit_features);
189  if (!$prodcomb->fetchByProductCombination2ValuePairs($id, $sanit_features)) {
190  $result = $prodcomb->createProductCombination($user, $object, $sanit_features, array(), $level_price_impact_percent, $level_price_impact, $weight_impact, $reference);
191  if ($result > 0) {
192  setEventMessages($langs->trans('RecordSaved'), null, 'mesgs');
193  unset($_SESSION['addvariant_'.$object->id]);
194 
195  $db->commit();
196  header('Location: '.dol_buildpath('/variants/combinations.php?id='.$id, 2));
197  exit();
198  } else {
199  $langs->load("errors");
200  setEventMessages($prodcomb->error, $prodcomb->errors, 'errors');
201  }
202  } else {
203  setEventMessages($langs->trans('ErrorRecordAlreadyExists'), null, 'errors');
204  }
205 
206  $db->rollback();
207  }
208 } elseif (!empty($massaction)) {
209  $bulkaction = $massaction;
210  $error = 0;
211 
212  $db->begin();
213 
214  foreach ($toselect as $prodid) {
215  // need create new of Product to prevent rename dir behavior
216  $prodstatic = new Product($db);
217  if ($prodstatic->fetch($prodid) < 0) {
218  continue;
219  }
220 
221  if ($bulkaction == 'on_sell') {
222  $prodstatic->status = 1;
223  $res = $prodstatic->update($prodstatic->id, $user);
224  if ($res <= 0) {
225  setEventMessages($prodstatic->error, $prodstatic->errors, 'errors');
226  $error++;
227  break;
228  }
229  } elseif ($bulkaction == 'on_buy') {
230  $prodstatic->status_buy = 1;
231  $res = $prodstatic->update($prodstatic->id, $user);
232  if ($res <= 0) {
233  setEventMessages($prodstatic->error, $prodstatic->errors, 'errors');
234  $error++;
235  break;
236  }
237  } elseif ($bulkaction == 'not_sell') {
238  $prodstatic->status = 0;
239  $res = $prodstatic->update($prodstatic->id, $user);
240  if ($res <= 0) {
241  setEventMessages($prodstatic->error, $prodstatic->errors, 'errors');
242  $error++;
243  break;
244  }
245  } elseif ($bulkaction == 'not_buy') {
246  $prodstatic->status_buy = 0;
247  $res = $prodstatic->update($prodstatic->id, $user);
248  if ($res <= 0) {
249  setEventMessages($prodstatic->error, $prodstatic->errors, 'errors');
250  $error++;
251  break;
252  }
253  } elseif ($bulkaction == 'delete') {
254  $res = $prodstatic->delete($user, $prodstatic->id);
255  if ($res <= 0) {
256  setEventMessages($prodstatic->error, $prodstatic->errors, 'errors');
257  $error++;
258  break;
259  }
260  } else {
261  break;
262  }
263  }
264 
265  if ($error) {
266  $db->rollback();
267  if (empty($prodstatic->error)) {
268  setEventMessages($langs->trans('CoreErrorMessage'), null, 'errors');
269  }
270  } else {
271  $db->commit();
272  setEventMessages($langs->trans('RecordSaved'), null, 'mesgs');
273  }
274 } elseif ($action === 'update' && $valueid > 0) {
275  if ($prodcomb->fetch($valueid) < 0) {
276  dol_print_error($db, $langs->trans('ErrorRecordNotFound'));
277  exit();
278  }
279 
280  $prodcomb->variation_weight = price2num($weight_impact);
281 
282  // for conf PRODUIT_MULTIPRICES
283  if (!empty($conf->global->PRODUIT_MULTIPRICES)) {
284  $level_price_impact = array_map('price2num', $level_price_impact);
285 
286  $prodcomb->variation_price = $level_price_impact[1];
287  $prodcomb->variation_price_percentage = (bool) $level_price_impact_percent[1];
288  } else {
289  $level_price_impact = array(1 => $price_impact);
290  $level_price_impact_percent = array(1 => $price_impact_percent);
291 
292  $prodcomb->variation_price = $price_impact;
293  $prodcomb->variation_price_percentage = $price_impact_percent;
294  }
295 
296  if (!empty($conf->global->PRODUIT_MULTIPRICES)) {
297  $prodcomb->combination_price_levels = array();
298  for ($i = 1; $i <= $conf->global->PRODUIT_MULTIPRICES_LIMIT; $i++) {
299  $productCombinationLevel = new ProductCombinationLevel($db);
300  $productCombinationLevel->fk_product_attribute_combination = $prodcomb->id;
301  $productCombinationLevel->fk_price_level = $i;
302  $productCombinationLevel->variation_price = $level_price_impact[$i];
303  $productCombinationLevel->variation_price_percentage = (bool) $level_price_impact_percent[$i];
304  $prodcomb->combination_price_levels[$i] = $productCombinationLevel;
305  }
306  }
307 
308  if ($prodcomb->update($user) > 0) {
309  setEventMessages($langs->trans('RecordSaved'), null, 'mesgs');
310  header('Location: '.dol_buildpath('/variants/combinations.php?id='.$id, 2));
311  exit();
312  } else {
313  setEventMessages($prodcomb->error, $prodcomb->errors, 'errors');
314  }
315 }
316 
317 
318 // Reload variants
319 $productCombinations = $prodcomb->fetchAllByFkProductParent($object->id);
320 
321 if ($action === 'confirm_deletecombination') {
322  if ($prodcomb->fetch($valueid) > 0) {
323  $db->begin();
324 
325  if ($prodcomb->delete($user) > 0 && (empty($delete_product) || ($delete_product == 'on' && $prodstatic->fetch($prodcomb->fk_product_child) > 0 && $prodstatic->delete($user) > 0))) {
326  $db->commit();
327  setEventMessages($langs->trans('RecordSaved'), null, 'mesgs');
328  header('Location: '.dol_buildpath('/variants/combinations.php?id='.$object->id, 2));
329  exit();
330  }
331 
332  $db->rollback();
333  setEventMessages($langs->trans('ProductCombinationAlreadyUsed'), null, 'errors');
334  $action = '';
335  }
336 } elseif ($action === 'edit') {
337  if ($prodcomb->fetch($valueid) < 0) {
338  dol_print_error($db, $langs->trans('ErrorRecordNotFound'));
339  exit();
340  }
341 
342  $weight_impact = $prodcomb->variation_weight;
343  $price_impact = $prodcomb->variation_price;
344  $price_impact_percent = $prodcomb->variation_price_percentage;
345 
346  $productCombination2ValuePairs1 = $prodcomb2val->fetchByFkCombination($valueid);
347 } elseif ($action === 'confirm_copycombination') {
348  //Check destination product
349  $dest_product = GETPOST('dest_product');
350 
351  if ($prodstatic->fetch('', $dest_product) > 0) {
352  //To prevent from copying to the same product
353  if ($prodstatic->ref != $object->ref) {
354  if ($prodcomb->copyAll($user, $object->id, $prodstatic) > 0) {
355  header('Location: '.dol_buildpath('/variants/combinations.php?id='.$prodstatic->id, 2));
356  exit();
357  } else {
358  setEventMessages($langs->trans('ErrorCopyProductCombinations'), null, 'errors');
359  }
360  }
361  } else {
362  setEventMessages($langs->trans('ErrorDestinationProductNotFound'), null, 'errors');
363  }
364 }
365 
366 
367 
368 /*
369  * View
370  */
371 
372 $form = new Form($db);
373 
374 $title = $langs->trans("Variant");
375 
376 llxHeader("", $title);
377 
378 
379 if (!empty($id) || !empty($ref)) {
380  $showbarcode = empty($conf->barcode->enabled) ? 0 : 1;
381  if (!empty($conf->global->MAIN_USE_ADVANCED_PERMS) && empty($user->rights->barcode->lire_advance)) {
382  $showbarcode = 0;
383  }
384 
385  $head = product_prepare_head($object);
386  $titre = $langs->trans("CardProduct".$object->type);
387  $picto = ($object->type == Product::TYPE_SERVICE ? 'service' : 'product');
388 
389  print dol_get_fiche_head($head, 'combinations', $titre, -1, $picto);
390 
391  $linkback = '<a href="'.DOL_URL_ROOT.'/product/list.php?type='.$object->type.'">'.$langs->trans("BackToList").'</a>';
392  $object->next_prev_filter = " fk_product_type = ".$object->type;
393 
394  dol_banner_tab($object, 'ref', $linkback, ($user->socid ? 0 : 1), 'ref', '', '', '', 0, '', '');
395 
396  print '<div class="fichecenter">';
397 
398  print '<div class="underbanner clearboth"></div>';
399  print '<table class="border centpercent tableforfield">';
400 
401  // Type
402  if (isModEnabled("product") && isModEnabled("service")) {
403  $typeformat = 'select;0:'.$langs->trans("Product").',1:'.$langs->trans("Service");
404  print '<tr><td class="titlefieldcreate">';
405  print (empty($conf->global->PRODUCT_DENY_CHANGE_PRODUCT_TYPE)) ? $form->editfieldkey("Type", 'fk_product_type', $object->type, $object, $usercancreate, $typeformat) : $langs->trans('Type');
406  print '</td><td>';
407  print $form->editfieldval("Type", 'fk_product_type', $object->type, $object, $usercancreate, $typeformat);
408  print '</td></tr>';
409  }
410 
411  // TVA
412  print '<tr><td class="titlefieldcreate">'.$langs->trans("DefaultTaxRate").'</td><td>';
413 
414  $positiverates = '';
415  if (price2num($object->tva_tx)) {
416  $positiverates .= ($positiverates ? '/' : '').price2num($object->tva_tx);
417  }
418  if (price2num($object->localtax1_type)) {
419  $positiverates .= ($positiverates ? '/' : '').price2num($object->localtax1_tx);
420  }
421  if (price2num($object->localtax2_type)) {
422  $positiverates .= ($positiverates ? '/' : '').price2num($object->localtax2_tx);
423  }
424  if (empty($positiverates)) {
425  $positiverates = '0';
426  }
427  echo vatrate($positiverates.($object->default_vat_code ? ' ('.$object->default_vat_code.')' : ''), '%', $object->tva_npr);
428  /*
429  if ($object->default_vat_code)
430  {
431  print vatrate($object->tva_tx, true) . ' ('.$object->default_vat_code.')';
432  }
433  else print vatrate($object->tva_tx, true, $object->tva_npr, true);*/
434  print '</td></tr>';
435 
436  // Price
437  print '<tr><td>'.$langs->trans("SellingPrice").'</td><td>';
438  if ($object->price_base_type == 'TTC') {
439  print price($object->price_ttc).' '.$langs->trans($object->price_base_type);
440  } else {
441  print price($object->price).' '.$langs->trans($object->price_base_type);
442  }
443  print '</td></tr>';
444 
445  // Price minimum
446  print '<tr><td>'.$langs->trans("MinPrice").'</td><td>';
447  if ($object->price_base_type == 'TTC') {
448  print price($object->price_min_ttc).' '.$langs->trans($object->price_base_type);
449  } else {
450  print price($object->price_min).' '.$langs->trans($object->price_base_type);
451  }
452  print '</td></tr>';
453 
454  // Weight
455  print '<tr><td>'.$langs->trans("Weight").'</td><td>';
456  if ($object->weight != '') {
457  print $object->weight." ".measuringUnitString(0, "weight", $object->weight_units);
458  } else {
459  print '&nbsp;';
460  }
461  print "</td></tr>\n";
462 
463  print "</table>\n";
464 
465  print '</div>';
466  print '<div style="clear:both"></div>';
467 
468  print dol_get_fiche_end();
469 
470  $listofvariantselected = '';
471 
472  // Create or edit a varian
473  if ($action == 'add' || ($action == 'edit')) {
474  if ($action == 'add') {
475  $title = $langs->trans('NewProductCombination');
476  // print dol_get_fiche_head();
477  $features = !empty($_SESSION['addvariant_'.$object->id]) ? $_SESSION['addvariant_'.$object->id] : array();
478  //First, sanitize
479  $listofvariantselected = '<div id="parttoaddvariant">';
480  if (!empty($features)) {
481  $toprint = array();
482  foreach ($features as $feature) {
483  $explode = explode('-', $feature);
484  if ($prodattr->fetch($explode[0]) <= 0 || $prodattr_val->fetch($explode[1]) <= 0) {
485  continue;
486  }
487  $toprint[] = '<li class="select2-search-choice-dolibarr noborderoncategories" style="background: #ddd;">' . $prodattr->label.' : '.$prodattr_val->value .
488  ' <a class="reposition" href="'.$_SERVER["PHP_SELF"].'?id='.$object->id.'&action=create&subaction=delete&feature='.urlencode($feature).'">' . img_delete() . '</a></li>';
489  }
490  $listofvariantselected .= '<div class="select2-container-multi-dolibarr" style="width: 90%;"><ul class="select2-choices-dolibarr">' . implode(' ', $toprint) . '</ul></div>';
491  }
492  $listofvariantselected .= '</div>';
493  //print dol_get_fiche_end();
494  } else {
495  $title = $langs->trans('EditProductCombination');
496  }
497 
498  if ($action == 'add') {
499  $prodattr_all = $prodattr->fetchAll();
500 
501  if (!$selected) {
502  $selected = $prodattr_all[key($prodattr_all)]->id;
503  }
504 
505  $prodattr_alljson = array();
506 
507  foreach ($prodattr_all as $each) {
508  $prodattr_alljson[$each->id] = $each;
509  }
510  ?>
511 
512  <script type="text/javascript">
513 
514  variants_available = <?php echo json_encode($prodattr_alljson); ?>;
515  variants_selected = {
516  index: [],
517  info: []
518  };
519 
520  <?php
521  foreach ($productCombination2ValuePairs1 as $pc2v) {
522  $prodattr_val->fetch($pc2v->fk_prod_attr_val);
523  ?>
524  variants_selected.index.push(<?php echo $pc2v->fk_prod_attr ?>);
525  variants_selected.info[<?php echo $pc2v->fk_prod_attr ?>] = {
526  attribute: variants_available[<?php echo $pc2v->fk_prod_attr ?>],
527  value: {
528  id: <?php echo $pc2v->fk_prod_attr_val ?>,
529  label: '<?php echo $prodattr_val->value ?>'
530  }
531  };
532  <?php
533  }
534  ?>
535 
536  restoreAttributes = function() {
537  jQuery("select[name=attribute]").empty().append('<option value="-1">&nbsp;</option>');
538 
539  jQuery.each(variants_available, function (key, val) {
540  if (jQuery.inArray(val.id, variants_selected.index) == -1) {
541  jQuery("select[name=attribute]").append('<option value="' + val.id + '">' + val.label + '</option>');
542  }
543  });
544  };
545 
546 
547  jQuery(document).ready(function() {
548  jQuery("select#attribute").change(function () {
549  console.log("Change of field variant attribute");
550  var select = jQuery("select#value");
551 
552  if (!jQuery(this).val().length || jQuery(this).val() == '-1') {
553  select.empty();
554  select.append('<option value="-1">&nbsp;</option>');
555  return;
556  }
557 
558  select.empty().append('<option value="">Loading...</option>');
559 
560  jQuery.getJSON("ajax/get_attribute_values.php", {
561  id: jQuery(this).val()
562  }, function(data) {
563  if (data.error) {
564  select.empty();
565  select.append('<option value="-1">&nbsp;</option>');
566  return alert(data.error);
567  }
568 
569  select.empty();
570  select.append('<option value="-1">&nbsp;</option>');
571 
572  jQuery(data).each(function (key, val) {
573  keyforoption = val.id
574  valforoption = val.value
575  select.append('<option value="' + keyforoption + '">' + valforoption + '</option>');
576  });
577  });
578  });
579  });
580  </script>
581 
582  <?php
583  }
584 
585  print '<br>';
586 
587  print load_fiche_titre($title);
588 
589  print '<form method="post" id="combinationform" action="'.$_SERVER["PHP_SELF"] .'?id='.$object->id.'">'."\n";
590  print '<input type="hidden" name="token" value="'.newToken().'">';
591  print '<input type="hidden" name="action" value="'.(($valueid > 0) ? "update" : "create").'">'."\n";
592  if ($valueid > 0) {
593  print '<input type="hidden" name="valueid" value="'.$valueid.'">'."\n";
594  }
595 
596  print dol_get_fiche_head();
597 
598 
599  if ($action == 'add') {
600  print '<table class="border" style="width: 100%">';
601  print "<!-- Variant -->\n";
602  print '<tr>';
603  print '<td class="titlefieldcreate fieldrequired"><label for="attribute">'.$langs->trans('ProductAttribute').'</label></td>';
604  print '<td>';
605  if (is_array($prodattr_all)) {
606  print '<select class="flat minwidth100" id="attribute" name="attribute">';
607  print '<option value="-1">&nbsp;</option>';
608  foreach ($prodattr_all as $attr) {
609  //print '<option value="'.$attr->id.'"'.($attr->id == GETPOST('attribute', 'int') ? ' selected="selected"' : '').'>'.$attr->label.'</option>';
610  print '<option value="'.$attr->id.'">'.$attr->label.'</option>';
611  }
612  print '</select>';
613  }
614 
615  $htmltext = $langs->trans("GoOnMenuToCreateVairants", $langs->transnoentities("Product"), $langs->transnoentities("VariantAttributes"));
616  print $form->textwithpicto('', $htmltext);
617  /*print ' &nbsp; &nbsp; <a href="'.DOL_URL_ROOT.'/variants/create.php?action=create&backtopage='.urlencode($_SERVER["PHP_SELF"].'?action=add&token='.newToken().'&id='.$object->id).'">';
618  print $langs->trans("Create");
619  print '</a>';*/
620 
621  print '</td>';
622  print '</tr>';
623  ?>
624  <!-- Value -->
625  <tr>
626  <td class="fieldrequired"><label for="value"><?php echo $langs->trans('Value') ?></label></td>
627  <td>
628  <select class="flat minwidth100" id="value" name="value">
629  <option value="-1">&nbsp;</option>
630  </select>
631  <?php
632  $htmltext = $langs->trans("GoOnMenuToCreateVairants", $langs->transnoentities("Product"), $langs->transnoentities("VariantAttributes"));
633  print $form->textwithpicto('', $htmltext);
634  /*
635  print ' &nbsp; &nbsp; <a href="'.DOL_URL_ROOT.'/variants/create.php?action=create&backtopage='.urlencode($_SERVER["PHP_SELF"].'?action=add&token='.newToken().'&id='.$object->id).'">';
636  print $langs->trans("Create");
637  print '</a>';
638  */
639  ?>
640  </td>
641  </tr>
642  <tr>
643  <td></td><td>
644  <input type="submit" class="button" name="selectvariant" id="selectvariant" value="<?php echo dol_escape_htmltag($langs->trans("SelectCombination")); ?>">
645  </td>
646  </tr>
647  <?php
648  print '<tr><td></td><td>';
649  print $listofvariantselected;
650  print '</td>';
651  print '</tr>';
652 
653  print '</table>';
654  print '<hr>';
655  }
656 
657  if (is_array($productCombination2ValuePairs1)) {
658  print '<table class="border" style="width: 100%">';
659 
660  // When in edit mode
661  if (is_array($productCombination2ValuePairs1) && count($productCombination2ValuePairs1)) {
662  ?>
663  <tr>
664  <td class="titlefieldcreate tdtop"><label for="features"><?php echo $langs->trans('Combination') ?></label></td>
665  <td class="tdtop">
666  <div class="inline-block valignmiddle quatrevingtpercent">
667  <?php
668  foreach ($productCombination2ValuePairs1 as $key => $val) {
669  $result1 = $prodattr->fetch($val->fk_prod_attr);
670  $result2 = $prodattr_val->fetch($val->fk_prod_attr_val);
671  //print 'rr'.$result1.' '.$result2;
672  if ($result1 > 0 && $result2 > 0) {
673  print $prodattr->label.' - '.$prodattr_val->value.'<br>';
674  // TODO Add delete link
675  }
676  }
677  ?>
678  </div>
679  <!-- <div class="inline-block valignmiddle">
680  <a href="#" class="inline-block valignmiddle button" id="delfeature"><?php echo img_edit_remove() ?></a>
681  </div>-->
682  </td>
683  <td>
684  </td>
685  </tr>
686  <?php
687  }
688  ?>
689  <tr>
690  <td><label for="reference"><?php echo $langs->trans('Reference') ?></label></td>
691  <td><input type="text" id="reference" name="reference" value="<?php echo trim($reference) ?>"></td>
692  </tr>
693  <?php
694  if (empty($conf->global->PRODUIT_MULTIPRICES)) {
695  ?>
696  <tr>
697  <td><label for="price_impact"><?php echo $langs->trans('PriceImpact') ?></label></td>
698  <td><input type="text" id="price_impact" name="price_impact" value="<?php echo price($price_impact) ?>">
699  <input type="checkbox" id="price_impact_percent" name="price_impact_percent" <?php echo $price_impact_percent ? ' checked' : '' ?>> <label for="price_impact_percent"><?php echo $langs->trans('PercentageVariation') ?></label>
700  </td>
701  </tr>
702  <?php
703  } else {
704  $prodcomb->fetchCombinationPriceLevels();
705 
706  for ($i = 1; $i <= $conf->global->PRODUIT_MULTIPRICES_LIMIT; $i++) {
707  print '<tr>';
708  print '<td><label for="level_price_impact_'.$i.'">'.$langs->trans('ImpactOnPriceLevel', $i).'</label>';
709  if ($i === 1) {
710  print ' <a id="apply-price-impact-to-all-level" class="classfortooltip" href="#" title="'.$langs->trans('ApplyToAllPriceImpactLevelHelp').'">('.$langs->trans('ApplyToAllPriceImpactLevel').')</a>';
711  }
712  print '</td>';
713  print '<td><input type="text" class="level_price_impact" id="level_price_impact_'.$i.'" name="level_price_impact['.$i.']" value="'.price($prodcomb->combination_price_levels[$i]->variation_price).'">';
714  print '<input type="checkbox" class="level_price_impact_percent" id="level_price_impact_percent_'.$i.'" name="level_price_impact_percent['.$i.']" '.(!empty($prodcomb->combination_price_levels[$i]->variation_price_percentage) ? ' checked' : '').'> <label for="level_price_impact_percent_'.$i.'">'.$langs->trans('PercentageVariation').'</label>';
715 
716  print '</td>';
717  print '</tr>';
718  }
719  }
720 
721  if ($object->isProduct()) {
722  print '<tr>';
723  print '<td><label for="weight_impact">'.$langs->trans('WeightImpact').'</label></td>';
724  print '<td><input type="text" id="weight_impact" name="weight_impact" value="'.price($weight_impact).'"></td>';
725  print '</tr>';
726  }
727 
728  print '</table>';
729  }
730 
731  if (!empty($conf->global->PRODUIT_MULTIPRICES)) {
732  ?>
733  <script>
734  $(document).ready(function() {
735  // Apply level 1 impact to all prices impact levels
736  $('body').on('click', '#apply-price-impact-to-all-level', function(e) {
737  e.preventDefault();
738  let priceImpact = $( "#level_price_impact_1" ).val();
739  let priceImpactPrecent = $( "#level_price_impact_percent_1" ).prop("checked");
740 
741  var multipricelimit = <?php print intval($conf->global->PRODUIT_MULTIPRICES_LIMIT); ?>
742 
743  for (let i = 2; i <= multipricelimit; i++) {
744  $( "#level_price_impact_" + i ).val(priceImpact);
745  $( "#level_price_impact_percent_" + i ).prop("checked", priceImpactPrecent);
746  }
747  });
748  });
749  </script>
750  <?php
751  }
752 
753  print dol_get_fiche_end();
754  ?>
755 
756  <div style="text-align: center">
757  <input type="submit" name="create" <?php if (!is_array($productCombination2ValuePairs1)) {
758  print ' disabled="disabled"';
759  } ?> value="<?php echo $action == 'add' ? $langs->trans('Create') : $langs->trans("Save") ?>" class="button button-save">
760  &nbsp;
761  <input type="submit" name="cancel" value="<?php echo $langs->trans("Cancel"); ?>" class="button button-cancel">
762  </div>
763 
764  <?php
765 
766  print '</form>';
767  } else {
768  if ($action === 'delete') {
769  if ($prodcomb->fetch($valueid) > 0) {
770  $prodstatic->fetch($prodcomb->fk_product_child);
771 
772  print $form->formconfirm(
773  "combinations.php?id=".urlencode($id)."&valueid=".urlencode($valueid),
774  $langs->trans('Delete'),
775  $langs->trans('ProductCombinationDeleteDialog', $prodstatic->ref),
776  "confirm_deletecombination",
777  array(array('label'=> $langs->trans('DeleteLinkedProduct'), 'type'=> 'checkbox', 'name' => 'delete_product', 'value' => false)),
778  0,
779  1
780  );
781  }
782  } elseif ($action === 'copy') {
783  print $form->formconfirm('combinations.php?id='.$id, $langs->trans('ToClone'), $langs->trans('ConfirmCloneProductCombinations'), 'confirm_copycombination', array(array('type' => 'text', 'label' => $langs->trans('CloneDestinationReference'), 'name' => 'dest_product')), 0, 1);
784  }
785 
786  $comb2val = new ProductCombination2ValuePair($db);
787 
788  if ($productCombinations) {
789  ?>
790 
791  <script type="text/javascript">
792  jQuery(document).ready(function() {
793 
794  jQuery('input[name="select_all"]').click(function() {
795 
796  if (jQuery(this).prop('checked')) {
797  var checked = true;
798  } else {
799  var checked = false;
800  }
801 
802  jQuery('table.liste input[type="checkbox"]').prop('checked', checked);
803  });
804 
805  jQuery('input[name^="select["]').click(function() {
806  jQuery('input[name="select_all"]').prop('checked', false);
807  });
808 
809  });
810  </script>
811 
812  <?php
813  }
814 
815  // Buttons
816  print '<div class="tabsAction">';
817 
818  print ' <div class="inline-block divButAction">';
819 
820  print '<a href="combinations.php?id='.$object->id.'&action=add&token='.newToken().'" class="butAction">'.$langs->trans('NewProductCombination').'</a>'; // NewVariant
821 
822  if ($productCombinations) {
823  print '<a href="combinations.php?id='.$object->id.'&action=copy&token='.newToken().'" class="butAction">'.$langs->trans('PropagateVariant').'</a>';
824  }
825 
826  print ' </div>';
827 
828  print '</div>';
829 
830 
831 
832  $arrayofselected = is_array($toselect) ? $toselect : array();
833 
834 
835  // List of variants
836  print '<form method="POST" action="'.$_SERVER["PHP_SELF"] .'?id='.$object->id.'">';
837  print '<input type="hidden" name="token" value="'.newToken().'">';
838  print '<input type="hidden" name="action" value="massaction">';
839  print '<input type="hidden" name="backtopage" value="'.$backtopage.'">';
840 
841  // List of mass actions available
842  /*
843  $arrayofmassactions = array(
844  'presend'=>$langs->trans("SendByMail"),
845  'builddoc'=>$langs->trans("PDFMerge"),
846  );
847  if ($user->rights->product->supprimer) $arrayofmassactions['predelete']='<span class="fa fa-trash paddingrightonly"></span>'.$langs->trans("Delete");
848  if (in_array($massaction, array('presend','predelete'))) $arrayofmassactions=array();
849  $massactionbutton=$form->selectMassAction('', $arrayofmassactions);
850  */
851 
852  $aaa = '';
853  if (count($productCombinations)) {
854  $aaa = '<select id="bulk_action" name="massaction" class="flat">';
855  $aaa .= ' <option value="nothing">&nbsp;</option>';
856  $aaa .= ' <option value="not_buy" data-html="'.dol_escape_htmltag(img_picto($langs->trans("SetToStatus"), 'stop-circle', 'class="pictofixedwidth"').$langs->trans('SetToStatus', $langs->transnoentitiesnoconv('ProductStatusNotOnBuy'))).'">'.$langs->trans('ProductStatusNotOnBuy').'</option>';
857  $aaa .= ' <option value="not_sell" data-html="'.dol_escape_htmltag(img_picto($langs->trans("SetToStatus"), 'stop-circle', 'class="pictofixedwidth"').$langs->trans('SetToStatus', $langs->transnoentitiesnoconv('ProductStatusNotOnSell'))).'">'.$langs->trans('ProductStatusNotOnSell').'</option>';
858  $aaa .= ' <option value="on_buy" data-html="'.dol_escape_htmltag(img_picto($langs->trans("SetToStatus"), 'stop-circle', 'class="pictofixedwidth"').$langs->trans('SetToStatus', $langs->transnoentitiesnoconv('ProductStatusOnBuy'))).'">'.$langs->trans('ProductStatusOnBuy').'</option>';
859  $aaa .= ' <option value="on_sell" data-html="'.dol_escape_htmltag(img_picto($langs->trans("SetToStatus"), 'stop-circle', 'class="pictofixedwidth"').$langs->trans('SetToStatus', $langs->transnoentitiesnoconv('ProductStatusOnSell'))).'">'.$langs->trans('ProductStatusOnSell').'</option>';
860  $aaa .= ' <option value="delete" data-html="'.dol_escape_htmltag(img_picto($langs->trans("Delete"), 'delete', 'class="pictofixedwidth"').$langs->trans('Delete')).'">'.$langs->trans('Delete').'</option>';
861  $aaa .= '</select>';
862  $aaa .= ajax_combobox("bulk_action");
863  $aaa .= '<input type="submit" value="'.dol_escape_htmltag($langs->trans("Apply")).'" class="button small">';
864  }
865  $massactionbutton = $aaa;
866 
867  $title = $langs->trans("ProductCombinations");
868 
869  print_barre_liste($title, 0, $_SERVER["PHP_SELF"], '', $sortfield, $sortorder, $aaa, 0);
870 
871  print '<div class="div-table-responsive">';
872  ?>
873  <table class="liste">
874  <tr class="liste_titre">
875  <td class="liste_titre"><?php echo $langs->trans('Product') ?></td>
876  <td class="liste_titre"><?php echo $langs->trans('Combination') ?></td>
877  <td class="liste_titre right"><?php echo $langs->trans('PriceImpact') ?></td>
878  <?php if ($object->isProduct()) {
879  print'<td class="liste_titre right">'.$langs->trans('WeightImpact').'</td>';
880  } ?>
881  <td class="liste_titre center"><?php echo $langs->trans('OnSell') ?></td>
882  <td class="liste_titre center"><?php echo $langs->trans('OnBuy') ?></td>
883  <td class="liste_titre"></td>
884  <?php
885  print '<td class="liste_titre center">';
886  $searchpicto = $form->showCheckAddButtons('checkforselect', 1);
887  print $searchpicto;
888  print '</td>';
889  ?>
890  </tr>
891  <?php
892 
893  if (count($productCombinations)) {
894  foreach ($productCombinations as $currcomb) {
895  $prodstatic->fetch($currcomb->fk_product_child);
896  print '<tr class="oddeven">';
897  print '<td>'.$prodstatic->getNomUrl(1).'</td>';
898  print '<td>';
899 
900  $productCombination2ValuePairs = $comb2val->fetchByFkCombination($currcomb->id);
901  $iMax = count($productCombination2ValuePairs);
902 
903  for ($i = 0; $i < $iMax; $i++) {
904  echo dol_htmlentities($productCombination2ValuePairs[$i]);
905  if ($i !== ($iMax - 1)) {
906  echo ', ';
907  }
908  }
909  print '</td>';
910  print '<td class="right">'.($currcomb->variation_price >= 0 ? '+' : '').price($currcomb->variation_price).($currcomb->variation_price_percentage ? ' %' : '').'</td>';
911  if ($object->isProduct()) {
912  print '<td class="right">'.($currcomb->variation_weight >= 0 ? '+' : '').price($currcomb->variation_weight).' '.measuringUnitString(0, 'weight', $prodstatic->weight_units).'</td>';
913  }
914  print '<td class="center">'.$prodstatic->getLibStatut(2, 0).'</td>';
915  print '<td class="center">'.$prodstatic->getLibStatut(2, 1).'</td>';
916  print '<td class="right">';
917  print '<a class="paddingleft paddingright editfielda" href="'.$_SERVER["PHP_SELF"].'?id='.$id.'&action=edit&token='.newToken().'&valueid='.$currcomb->id.'">'.img_edit().'</a>';
918  print '<a class="paddingleft paddingright" href="'.$_SERVER["PHP_SELF"].'?id='.$id.'&action=delete&token='.newToken().'&valueid='.$currcomb->id.'">'.img_delete().'</a>';
919  print '</td>';
920  print '<td class="nowrap center">';
921  if (!empty($productCombinations) || $massactionbutton || $massaction) { // If we are in select mode (massactionbutton defined) or if we have already selected and sent an action ($massaction) defined
922  $selected = 0;
923  if (in_array($prodstatic->id, $arrayofselected)) {
924  $selected = 1;
925  }
926  print '<input id="cb'.$prodstatic->id.'" class="flat checkforselect" type="checkbox" name="toselect[]" value="'.$prodstatic->id.'"'.($selected ? ' checked="checked"' : '').'>';
927  }
928  print '</td>';
929  print '</tr>';
930  }
931  } else {
932  print '<tr><td colspan="8"><span class="opacitymedium">'.$langs->trans("None").'</span></td></tr>';
933  }
934  print '</table>';
935  print '</div>';
936  print '</form>';
937  }
938 }
939 
940 // End of page
941 llxFooter();
942 $db->close();
ajax_combobox($htmlname, $events=array(), $minLengthToAutocomplete=0, $forcefocus=0, $widthTypeOfAutocomplete='resolve', $idforemptyvalue='-1', $morecss='')
Convert a html select field into an ajax combobox.
Definition: ajax.lib.php:449
if(!defined('NOREQUIRESOC')) if(!defined('NOREQUIRETRAN')) if(!defined('NOTOKENRENEWAL')) if(!defined('NOREQUIREMENU')) if(!defined('NOREQUIREHTML')) if(!defined('NOREQUIREAJAX')) llxHeader()
Empty header.
Definition: wrapper.php:56
llxFooter()
Empty footer.
Definition: wrapper.php:70
Class to manage generation of HTML components Only common components must be here.
Class ProductAttribute Used to represent a product attribute.
Class ProductAttributeValue Used to represent a product attribute value.
Class ProductCombination2ValuePair Used to represent the relation between a product combination,...
Class ProductCombination Used to represent a product combination.
Class ProductCombinationLevel Used to represent a product combination Level.
Class to manage products or services.
const TYPE_PRODUCT
Regular product.
const TYPE_SERVICE
Service.
if($cancel &&! $id) if($action=='add' &&! $cancel) if($action=='delete') if($id) $form
Actions.
Definition: card.php:143
dol_banner_tab($object, $paramid, $morehtml='', $shownav=1, $fieldid='rowid', $fieldref='ref', $morehtmlref='', $moreparam='', $nodbprefix=0, $morehtmlleft='', $morehtmlstatus='', $onlybanner=0, $morehtmlright='')
Show tab footer of a card.
vatrate($rate, $addpercent=false, $info_bits=0, $usestarfornpr=0, $html=0)
Return a string with VAT rate label formated for view output Used into pdf and HTML pages.
load_fiche_titre($titre, $morehtmlright='', $picto='generic', $pictoisfullpath=0, $id='', $morecssontable='', $morehtmlcenter='')
Load a title with picto.
dol_get_fiche_head($links=array(), $active='', $title='', $notab=0, $picto='', $pictoisfullpath=0, $morehtmlright='', $morecss='', $limittoshow=0, $moretabssuffix='')
Show tabs of a record.
img_delete($titlealt='default', $other='class="pictodelete"', $morecss='')
Show delete logo.
price2num($amount, $rounding='', $option=0)
Function that return a number with universal decimal format (decimal separator is '.
dol_print_error($db='', $error='', $errors=null)
Displays error message system with all the information to facilitate the diagnosis and the escalation...
dol_get_fiche_end($notab=0)
Return tab footer of a card.
setEventMessages($mesg, $mesgs, $style='mesgs', $messagekey='')
Set event messages in dol_events session object.
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.
img_picto($titlealt, $picto, $moreatt='', $pictoisfullpath=false, $srconly=0, $notitle=0, $alt='', $morecss='', $marginleftonlyshort=2)
Show picto whatever it's its name (generic function)
newToken()
Return the value of token currently saved into session with name 'newtoken'.
dol_htmlentities($string, $flags=ENT_QUOTES|ENT_SUBSTITUTE, $encoding='UTF-8', $double_encode=false)
Replace htmlentities functions.
GETPOST($paramname, $check='alphanohtml', $method=0, $filter=null, $options=null, $noreplace=0)
Return value of a param into GET or POST supervariable.
print_barre_liste($titre, $page, $file, $options='', $sortfield='', $sortorder='', $morehtmlcenter='', $num=-1, $totalnboflines='', $picto='generic', $pictoisfullpath=0, $morehtmlright='', $morecss='', $limit=-1, $hideselectlimit=0, $hidenavigation=0, $pagenavastextinput=0, $morehtmlrightbeforearrow='')
Print a title with navigation controls for pagination.
dol_buildpath($path, $type=0, $returnemptyifnotfound=0)
Return path of url or filesystem.
isModEnabled($module)
Is Dolibarr module enabled.
img_edit($titlealt='default', $float=0, $other='')
Show logo editer/modifier fiche.
img_edit_remove($titlealt='default', $other='')
Show logo -.
product_prepare_head($object)
Prepare array with list of tabs.
Definition: product.lib.php:35
measuringUnitString($unit, $measuring_style='', $scale='', $use_short_label=0, $outputlangs=null)
Return translation label of a unit key.
if(preg_match('/crypted:/i', $dolibarr_main_db_pass)||!empty($dolibarr_main_db_encrypted_pass)) $conf db type
Definition: repair.php:119
$conf db name
Only used if Module[ID]Name translation string is not found.
Definition: repair.php:122
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.
accessforbidden($message='', $printheader=1, $printfooter=1, $showonlymessage=0, $params=null)
Show a message to say access is forbidden and stop program.