* Copyright (C) 2017 Laurent Destailleur * Copyright (C) 2018-2024 Frédéric France * Copyright (C) 2022 Open-Dsi * Copyright (C) 2024 MDW * * 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 . */ // Load Dolibarr environment require '../main.inc.php'; require_once DOL_DOCUMENT_ROOT.'/core/lib/product.lib.php'; require_once DOL_DOCUMENT_ROOT.'/product/class/product.class.php'; require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductAttribute.class.php'; require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductAttributeValue.class.php'; require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductCombination.class.php'; require_once DOL_DOCUMENT_ROOT.'/variants/class/ProductCombination2ValuePair.class.php'; $langs->loadLangs(array("products", "other")); $id = GETPOSTINT('id'); // ID of the parent Product $ref = GETPOST('ref', 'alpha'); $combination_id = GETPOSTINT('combination_id'); // ID of the combination $reference = GETPOST('reference', 'alpha'); // Reference of the variant Product $weight_impact = GETPOSTFLOAT('weight_impact', 2); $price_impact_percent = (bool) GETPOST('price_impact_percent'); $price_impact = $price_impact_percent ? GETPOSTFLOAT('price_impact', 2) : GETPOSTFLOAT('price_impact', 'MU'); // for PRODUIT_MULTIPRICES $level_price_impact = GETPOST('level_price_impact', 'array'); $level_price_impact = array_map('price2num', $level_price_impact); $level_price_impact = array_map('floatval', $level_price_impact); $level_price_impact_percent = GETPOST('level_price_impact_percent', 'array'); $level_price_impact_percent = array_map('boolval', $level_price_impact_percent); $form = new Form($db); $action = GETPOST('action', 'aZ09'); $massaction = GETPOST('massaction', 'alpha'); $show_files = GETPOSTINT('show_files'); $confirm = GETPOST('confirm', 'alpha'); $toselect = GETPOST('toselect', 'array'); $cancel = GETPOST('cancel', 'alpha'); $delete_product = GETPOST('delete_product', 'alpha'); $subaction = GETPOST('subaction', 'aZ09'); $backtopage = GETPOST('backtopage', 'alpha'); $sortfield = GETPOST('sortfield', 'aZ09comma'); $sortorder = GETPOST('sortorder', 'aZ09comma'); // Security check $fieldvalue = $id ?: $ref; $fieldtype = !empty($ref) ? 'ref' : 'rowid'; $prodstatic = new Product($db); $prodattr = new ProductAttribute($db); $prodattr_val = new ProductAttributeValue($db); $object = new Product($db); if ($id > 0 || $ref) { $object->fetch($id, $ref); } $selectedvariant = isset($_SESSION['addvariant_'.$object->id]) ? $_SESSION['addvariant_'.$object->id] : array(); $selected = ''; // Security check if (!isModEnabled('variants')) { accessforbidden('Module not enabled'); } if ($user->socid > 0) { // Protection if external user accessforbidden(); } if ($object->id > 0) { if ($object->isProduct()) { restrictedArea($user, 'produit', $object->id, 'product&product', '', ''); } if ($object->isService()) { restrictedArea($user, 'service', $object->id, 'product&product', '', ''); } } else { restrictedArea($user, 'produit|service', $fieldvalue, 'product&product', '', '', $fieldtype); } $usercanread = (($object->isProduct() && $user->hasRight('produit', 'lire')) || ($object->isService() && $user->hasRight('service', 'lire'))); $usercancreate = (($object->isProduct() && $user->hasRight('produit', 'creer')) || ($object->isService() && $user->hasRight('service', 'creer'))); $usercandelete = (($object->isProduct() && $user->hasRight('produit', 'supprimer')) || ($object->isService() && $user->hasRight('service', 'supprimer'))); /* * Actions */ if ($cancel) { $action = ''; $massaction = ''; unset($_SESSION['addvariant_'.$object->id]); } if (!$object->isProduct() && !$object->isService()) { header('Location: '.dol_buildpath('/product/card.php?id='.$object->id, 2)); exit(); } if ($action == 'add') { // Test on permission not required unset($selectedvariant); unset($_SESSION['addvariant_'.$object->id]); } if ($action == 'create' && GETPOST('selectvariant', 'alpha') && $usercancreate) { // We click on select combination $action = 'add'; $attribute_id = GETPOSTINT('attribute'); $attribute_value_id = GETPOSTINT('value'); if ($attribute_id > 0 && $attribute_value_id > 0) { $feature = $attribute_id . '-' . $attribute_value_id; $selectedvariant[$feature] = $feature; $_SESSION['addvariant_'.$object->id] = $selectedvariant; } } if ($action == 'create' && $subaction == 'delete' && $usercancreate) { // We click on select combination $action = 'add'; $feature = GETPOST('feature', 'intcomma'); if (isset($selectedvariant[$feature])) { unset($selectedvariant[$feature]); $_SESSION['addvariant_'.$object->id] = $selectedvariant; } } $prodcomb = new ProductCombination($db); $prodcomb2val = new ProductCombination2ValuePair($db); $productCombination2ValuePairs1 = array(); if (($action == 'add' || $action == 'create') && $usercancreate && empty($massaction) && !GETPOST('selectvariant', 'alpha') && empty($subaction)) { // We click on Create all defined combinations //$features = GETPOST('features', 'array'); $features = !empty($_SESSION['addvariant_'.$object->id]) ? $_SESSION['addvariant_'.$object->id] : array(); if (!$features) { if ($action == 'create') { // Test on permission already done setEventMessages($langs->trans('ErrorFieldsRequired'), null, 'errors'); } } else { $reference = trim($reference); if (empty($reference)) { $reference = false; } $weight_impact = price2num($weight_impact); $price_impact = price2num($price_impact); if (!getDolGlobalString('PRODUIT_MULTIPRICES')) { $level_price_impact = array(1 => $price_impact); $level_price_impact_percent = array(1 => $price_impact_percent); } $sanit_features = array(); //First, sanitize foreach ($features as $feature) { $explode = explode('-', $feature); if ($prodattr->fetch($explode[0]) <= 0 || $prodattr_val->fetch($explode[1]) <= 0) { continue; } // Valuepair $sanit_features[(int) $explode[0]] = (int) $explode[1]; $tmp = new ProductCombination2ValuePair($db); $tmp->fk_prod_attr = (int) $explode[0]; $tmp->fk_prod_attr_val = (int) $explode[1]; $productCombination2ValuePairs1[] = $tmp; } $db->begin(); // sanit_feature is an array with 1 (and only 1) value per attribute. // For example: Color->blue, Size->Small, Option->2 if (!$prodcomb->fetchByProductCombination2ValuePairs($id, $sanit_features)) { $result = $prodcomb->createProductCombination($user, $object, $sanit_features, array(), $level_price_impact_percent, $level_price_impact, $weight_impact, $reference); if ($result > 0) { setEventMessages($langs->trans('RecordSaved'), null, 'mesgs'); unset($_SESSION['addvariant_'.$object->id]); $db->commit(); header('Location: '.dol_buildpath('/variants/combinations.php?id='.$id, 2)); exit(); } else { $langs->load("errors"); setEventMessages($prodcomb->error, $prodcomb->errors, 'errors'); } } else { setEventMessages($langs->trans('ErrorRecordAlreadyExists'), null, 'errors'); } $db->rollback(); } } elseif (!empty($massaction)) { $bulkaction = $massaction; $error = 0; $db->begin(); foreach ($toselect as $prodid) { // need create new of Product to prevent rename dir behavior $prodstatic = new Product($db); if ($prodstatic->fetch($prodid) < 0) { continue; } if ($bulkaction == 'on_sell') { $prodstatic->status = 1; $res = $prodstatic->update($prodstatic->id, $user); if ($res <= 0) { setEventMessages($prodstatic->error, $prodstatic->errors, 'errors'); $error++; break; } } elseif ($bulkaction == 'on_buy') { $prodstatic->status_buy = 1; $res = $prodstatic->update($prodstatic->id, $user); if ($res <= 0) { setEventMessages($prodstatic->error, $prodstatic->errors, 'errors'); $error++; break; } } elseif ($bulkaction == 'not_sell') { $prodstatic->status = 0; $res = $prodstatic->update($prodstatic->id, $user); if ($res <= 0) { setEventMessages($prodstatic->error, $prodstatic->errors, 'errors'); $error++; break; } } elseif ($bulkaction == 'not_buy') { $prodstatic->status_buy = 0; $res = $prodstatic->update($prodstatic->id, $user); if ($res <= 0) { setEventMessages($prodstatic->error, $prodstatic->errors, 'errors'); $error++; break; } } elseif ($bulkaction == 'delete') { $res = $prodstatic->delete($user, $prodstatic->id); if ($res <= 0) { setEventMessages($prodstatic->error, $prodstatic->errors, 'errors'); $error++; break; } } else { break; } } if ($error) { $db->rollback(); if (empty($prodstatic->error)) { setEventMessages($langs->trans('CoreErrorMessage'), null, 'errors'); } } else { $db->commit(); setEventMessages($langs->trans('RecordSaved'), null, 'mesgs'); } } elseif ($action === 'update' && $combination_id > 0 && $usercancreate) { if ($prodcomb->fetch($combination_id) < 0) { dol_print_error($db, $langs->trans('ErrorRecordNotFound')); exit(); } $prodcomb->variation_weight = (float) price2num($weight_impact); // for conf PRODUIT_MULTIPRICES if (getDolGlobalString('PRODUIT_MULTIPRICES')) { $prodcomb->variation_price = $level_price_impact[1]; $prodcomb->variation_price_percentage = $level_price_impact_percent[1]; } else { $level_price_impact = array(1 => $price_impact); $level_price_impact_percent = array(1 => $price_impact_percent); $prodcomb->variation_price = $price_impact; $prodcomb->variation_price_percentage = $price_impact_percent; } if (getDolGlobalString('PRODUIT_MULTIPRICES')) { $prodcomb->combination_price_levels = array(); for ($i = 1; $i <= $conf->global->PRODUIT_MULTIPRICES_LIMIT; $i++) { $productCombinationLevel = new ProductCombinationLevel($db); $productCombinationLevel->fk_product_attribute_combination = $prodcomb->id; $productCombinationLevel->fk_price_level = $i; $productCombinationLevel->variation_price = (float) $level_price_impact[$i]; $productCombinationLevel->variation_price_percentage = (bool) $level_price_impact_percent[$i]; $prodcomb->combination_price_levels[$i] = $productCombinationLevel; } } $error = 0; $db->begin(); // Update product variant ref $product_child = new Product($db); $product_child->fetch($prodcomb->fk_product_child); $product_child->oldcopy = clone $product_child; // @phan-ignore-current-line PhanTypeMismatchProperty $product_child->ref = $reference; $result = $product_child->update($product_child->id, $user); if ($result < 0) { setEventMessages($product_child->error, $product_child->errors, 'errors'); $error++; } if (!$error) { // Update product variant infos $result = $prodcomb->update($user); if ($result < 0) { setEventMessages($prodcomb->error, $prodcomb->errors, 'errors'); $error++; } } if (!$error) { $db->commit(); setEventMessages($langs->trans('RecordSaved'), null, 'mesgs'); header('Location: ' . dol_buildpath('/variants/combinations.php?id=' . $id, 2)); exit(); } else { $db->rollback(); } } // Reload variants $productCombinations = $prodcomb->fetchAllByFkProductParent($object->id, true); if ($action === 'confirm_deletecombination' && $usercancreate) { if ($prodcomb->fetch($combination_id) > 0) { $db->begin(); if ($prodcomb->delete($user) > 0 && (empty($delete_product) || ($delete_product == 'on' && $prodstatic->fetch($prodcomb->fk_product_child) > 0 && $prodstatic->delete($user) > 0))) { $db->commit(); setEventMessages($langs->trans('RecordSaved'), null, 'mesgs'); header('Location: '.dol_buildpath('/variants/combinations.php?id='.$object->id, 2)); exit(); } $db->rollback(); setEventMessages($langs->trans('ProductCombinationAlreadyUsed'), null, 'errors'); $action = ''; } } elseif ($action === 'edit' && $usercancreate) { if ($prodcomb->fetch($combination_id) < 0) { dol_print_error($db, $langs->trans('ErrorRecordNotFound')); exit(); } $product_child = new Product($db); $product_child->fetch($prodcomb->fk_product_child); $reference = $product_child->ref; $weight_impact = $prodcomb->variation_weight; $price_impact = $prodcomb->variation_price; $price_impact_percent = $prodcomb->variation_price_percentage; $productCombination2ValuePairs1 = $prodcomb2val->fetchByFkCombination($combination_id); } elseif ($action === 'confirm_copycombination' && $usercancreate) { //Check destination product $dest_product = GETPOST('dest_product'); if ($prodstatic->fetch(0, $dest_product) > 0) { //To prevent from copying to the same product if ($prodstatic->ref != $object->ref) { if ($prodcomb->copyAll($user, $object->id, $prodstatic) > 0) { header('Location: '.dol_buildpath('/variants/combinations.php?id='.$prodstatic->id, 2)); exit(); } else { setEventMessages($langs->trans('ErrorCopyProductCombinations'), null, 'errors'); } } } else { setEventMessages($langs->trans('ErrorDestinationProductNotFound'), null, 'errors'); } } /* * View */ $form = new Form($db); $title = $langs->trans("Variant"); llxHeader("", $title); if (!empty($id) || !empty($ref)) { $showbarcode = isModEnabled('barcode'); if (getDolGlobalString('MAIN_USE_ADVANCED_PERMS') && !$user->hasRight('barcode', 'lire_advance')) { $showbarcode = 0; } $head = product_prepare_head($object); $titre = $langs->trans("CardProduct".$object->type); $picto = $object->isService() ? 'service' : 'product'; print dol_get_fiche_head($head, 'combinations', $titre, -1, $picto); $linkback = ''.$langs->trans("BackToList").''; $object->next_prev_filter = "fk_product_type = ".((int) $object->type); dol_banner_tab($object, 'ref', $linkback, ($user->socid ? 0 : 1), 'ref', '', '', '', 0, '', ''); print '
'; print '
'; print ''; // Type if (isModEnabled("product") && isModEnabled("service")) { $typeformat = 'select;0:'.$langs->trans("Product").',1:'.$langs->trans("Service"); print ''; } // TVA print ''; // Price print ''; // Price minimum print ''; // Weight print '\n"; print "
'; print (!getDolGlobalString('PRODUCT_DENY_CHANGE_PRODUCT_TYPE')) ? $form->editfieldkey("Type", 'fk_product_type', $object->type, $object, $usercancreate, $typeformat) : $langs->trans('Type'); print ''; print $form->editfieldval("Type", 'fk_product_type', $object->type, $object, $usercancreate, $typeformat); print '
'.$langs->trans("DefaultTaxRate").''; $positiverates = ''; if (price2num($object->tva_tx)) { $positiverates .= ($positiverates ? '/' : '').price2num($object->tva_tx); } if (price2num($object->localtax1_type)) { $positiverates .= ($positiverates ? '/' : '').price2num($object->localtax1_tx); } if (price2num($object->localtax2_type)) { $positiverates .= ($positiverates ? '/' : '').price2num($object->localtax2_tx); } if (empty($positiverates)) { $positiverates = '0'; } echo vatrate($positiverates.($object->default_vat_code ? ' ('.$object->default_vat_code.')' : ''), '%', $object->tva_npr); /* if ($object->default_vat_code) { print vatrate($object->tva_tx, true) . ' ('.$object->default_vat_code.')'; } else print vatrate($object->tva_tx, true, $object->tva_npr, true);*/ print '
'.$langs->trans("SellingPrice").''; if ($object->price_base_type == 'TTC') { print price($object->price_ttc).' '.$langs->trans($object->price_base_type); } else { print price($object->price).' '.$langs->trans($object->price_base_type); } print '
'.$langs->trans("MinPrice").''; if ($object->price_base_type == 'TTC') { print price($object->price_min_ttc).' '.$langs->trans($object->price_base_type); } else { print price($object->price_min).' '.$langs->trans($object->price_base_type); } print '
'.$langs->trans("Weight").''; if ($object->weight != '') { print $object->weight." ".measuringUnitString(0, "weight", $object->weight_units); } else { print ' '; } print "
\n"; print '
'; print '
'; print dol_get_fiche_end(); $listofvariantselected = ''; // Create or edit a variant if ($action == 'add' || ($action == 'edit')) { if ($action == 'add') { $title = $langs->trans('NewProductCombination'); // print dol_get_fiche_head(); $features = !empty($_SESSION['addvariant_'.$object->id]) ? $_SESSION['addvariant_'.$object->id] : array(); //First, sanitize $listofvariantselected = '
'; if (!empty($features)) { $toprint = array(); foreach ($features as $feature) { $explode = explode('-', $feature); if ($prodattr->fetch($explode[0]) <= 0 || $prodattr_val->fetch($explode[1]) <= 0) { continue; } $toprint[] = '
  • ' . $prodattr->label.' : '.$prodattr_val->value . ' ' . img_delete() . '
  • '; } $listofvariantselected .= '
      ' . implode(' ', $toprint) . '
    '; } $listofvariantselected .= '
    '; //print dol_get_fiche_end(); } else { $title = $langs->trans('EditProductCombination'); } if ($action == 'add') { $prodattr_all = $prodattr->fetchAll(); if (!$selected) { $selected = $prodattr_all[key($prodattr_all)]->id; } $prodattr_alljson = array(); foreach ($prodattr_all as $each) { $prodattr_alljson[$each->id] = $each; } ?> '; print load_fiche_titre($title); print '
    '."\n"; print ''; print ''."\n"; if ($combination_id > 0) { print ''."\n"; } print dol_get_fiche_head(); if ($action == 'add') { print ''; print "\n"; print ''; print ''; print ''; print ''; ?> '; print ''; print '
    '; if (is_array($prodattr_all)) { print ''; } $htmltext = $langs->trans("GoOnMenuToCreateVairants", $langs->transnoentities("Product"), $langs->transnoentities("VariantAttributes")); print $form->textwithpicto('', $htmltext); /*print '     id).'">'; print $langs->trans("Create"); print '';*/ print '
    trans("GoOnMenuToCreateVairants", $langs->transnoentities("Product"), $langs->transnoentities("VariantAttributes")); print $form->textwithpicto('', $htmltext); /* print '     id).'">'; print $langs->trans("Create"); print ''; */ ?>
    ">
    '; print $listofvariantselected; print '
    '; print '
    '; } if (is_array($productCombination2ValuePairs1)) { print ''; // When in edit mode if (is_array($productCombination2ValuePairs1) && count($productCombination2ValuePairs1)) { ?> fetchCombinationPriceLevels(); for ($i = 1; $i <= $conf->global->PRODUIT_MULTIPRICES_LIMIT; $i++) { $keyforlabel = 'PRODUIT_MULTIPRICES_LABEL'.$i; $text = $langs->trans('ImpactOnPriceLevel', $i).' - '.getDolGlobalString($keyforlabel); print ''; print ''; print ''; print ''; } } if ($object->isProduct()) { print ''; print ''; print ''; print ''; } print '
    fetch($pc2v->fk_prod_attr); $result2 = $prodattr_val->fetch($pc2v->fk_prod_attr_val); if ($result1 > 0 && $result2 > 0) { print $prodattr->label.' : '.$prodattr_val->value.'
    '; // TODO Add delete link } } ?>
    >
    '; if ($i === 1) { print '
    ('.$langs->trans('ApplyToAllPriceImpactLevel').')'; } print '
    '; print 'combination_price_levels[$i]->variation_price_percentage) ? ' checked' : '').'> '; print '
    '; } if (getDolGlobalString('PRODUIT_MULTIPRICES')) { ?>
    value="trans('Create') : $langs->trans("Save") ?>" class="button button-save">   " class="button button-cancel">
    '; } else { if ($action === 'delete') { if ($prodcomb->fetch($combination_id) > 0) { $prodstatic->fetch($prodcomb->fk_product_child); print $form->formconfirm( "combinations.php?id=".urlencode((string) ($id))."&combination_id=".urlencode((string) ($combination_id)), $langs->trans('Delete'), $langs->trans('ProductCombinationDeleteDialog', $prodstatic->ref), "confirm_deletecombination", array(array('label' => $langs->trans('DeleteLinkedProduct'), 'type' => 'checkbox', 'name' => 'delete_product', 'value' => false)), 0, 1 ); } } elseif ($action === 'copy') { 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); } $comb2val = new ProductCombination2ValuePair($db); if ($productCombinations) { ?> '; print '
    '; print ''.$langs->trans('NewProductCombination').''; // NewVariant if ($productCombinations) { print ''.$langs->trans('PropagateVariant').''; } print '
    '; print ''; $arrayofselected = is_array($toselect) ? $toselect : array(); // List of variants print ''; print ''; print ''; print ''; // List of mass actions available $aaa = ''; if (count($productCombinations)) { $aaa = ''; $aaa .= ajax_combobox("bulk_action"); $aaa .= ''; } $massactionbutton = $aaa; $title = $langs->trans("ProductCombinations"); print_barre_liste($title, 0, $_SERVER["PHP_SELF"], '', $sortfield, $sortorder, $aaa, 0); print '
    '; ?> '; $searchpicto = $form->showCheckAddButtons('checkforselect', 1); print $searchpicto; print ''; } ?> isProduct()) { print''; } ?> '; $searchpicto = $form->showCheckAddButtons('checkforselect', 1); print $searchpicto; print ''; } ?> fetch($currcomb->fk_product_child); print ''; // Action column if (getDolGlobalString('MAIN_CHECKBOX_LEFT_COLUMN')) { print ''; } print ''; print ''; print ''; if ($object->isProduct()) { print ''; } print ''; print ''; print ''; // Action column if (!getDolGlobalString('MAIN_CHECKBOX_LEFT_COLUMN')) { print ''; } print ''; } } else { print ''; } print '
    trans('Product') ?> trans('Attributes') ?> trans('PriceImpact') ?> '.$langs->trans('WeightImpact').'trans('OnSell') ?> trans('OnBuy') ?>
    '; 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 $selected = 0; if (in_array($prodstatic->id, $arrayofselected)) { $selected = 1; } print ''; } print ''.$prodstatic->getNomUrl(1).''; foreach ($comb2val->fetchByFkCombination($currcomb->id) as $pc2v) { print dol_htmlentities($pc2v).'
    '; } print '
    '.($currcomb->variation_price >= 0 ? '+' : '').price($currcomb->variation_price).($currcomb->variation_price_percentage ? ' %' : '').''.($currcomb->variation_weight >= 0 ? '+' : '').price($currcomb->variation_weight).' '.measuringUnitString(0, 'weight', $prodstatic->weight_units).''.$prodstatic->getLibStatut(2, 0).''.$prodstatic->getLibStatut(2, 1).''; print 'id.'">'.img_edit().''; print 'id.'">'.img_delete().''; print ''; 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 $selected = 0; if (in_array($prodstatic->id, $arrayofselected)) { $selected = 1; } print ''; } print '
    '.$langs->trans("None").'
    '; print '
    '; print '
    '; } } // End of page llxFooter(); $db->close();