* @copyright 2007-2016 PrestaShop SA * @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0) * International Registered Trademark & Property of PrestaShop SA */ /** * @since 1.5.0 * @property Product $object */ class AdminStockManagementControllerCore extends AdminController { public function __construct() { $this->bootstrap = true; $this->context = Context::getContext(); $this->table = 'product'; $this->list_id = 'product'; $this->className = 'Product'; $this->lang = true; $this->multishop_context = Shop::CONTEXT_ALL; $this->fields_list = array( 'reference' => array( 'title' => $this->l('Product reference'), 'filter_key' => 'a!reference' ), 'ean13' => array( 'title' => $this->l('EAN-13 or JAN barcode'), 'filter_key' => 'a!ean13' ), 'upc' => array( 'title' => $this->l('UPC barcode'), 'filter_key' => 'a!upc' ), 'name' => array( 'title' => $this->l('Name'), 'filter_key' => 'b!name' ), 'physical_quantity' => array( 'title' => $this->l('Physical quantity'), 'class' => 'fixed-width-xs', 'align' => 'center', 'orderby' => true, 'search' => false ), 'usable_quantity' => array( 'title' => $this->l('Usable quantity'), 'class' => 'fixed-width-xs', 'align' => 'center', 'orderby' => true, 'search' => false, ), ); parent::__construct(); // overrides confirmation messages specifically for this controller $this->_conf = array( 1 => $this->l('The product was successfully added to your stock.'), 2 => $this->l('The product was successfully removed from your stock.'), 3 => $this->l('The transfer was successfully completed.'), ); } public function initPageHeaderToolbar() { if ($this->display == 'details') { $this->page_header_toolbar_btn['back_to_list'] = array( 'href' => Context::getContext()->link->getAdminLink('AdminStockManagement'), 'desc' => $this->l('Back to list', null, null, false), 'icon' => 'process-icon-back' ); } parent::initPageHeaderToolbar(); } /** * AdminController::renderList() override * @see AdminController::renderList() */ public function renderList() { $id_product = (int)Tools::getValue('id_product'); if (!empty($id_product)) { $id_product_attribute = (int)Tools::getValue('id_product_attribute'); $this->previousManagementStock($id_product, $id_product_attribute); } else { // sets actions $this->addRowAction('details'); $this->addRowAction('addstock'); $this->addRowAction('prepareRemovestock'); if (count(Warehouse::getWarehouses(true)) > 1) { $this->addRowAction('prepareTransferstock'); } // no link on list rows $this->list_no_link = true; // inits toolbar $this->toolbar_btn = array(); // overrides query $this->_select = 'IFNULL(pa.ean13, a.ean13) as ean13, IFNULL(pa.upc, a.upc) as upc, IFNULL(pa.reference, a.reference) as reference, (SELECT SUM(physical_quantity) FROM `'._DB_PREFIX_.'stock` WHERE id_product = a.id_product) as physical_quantity, (SELECT SUM(usable_quantity) FROM `'._DB_PREFIX_.'stock` WHERE id_product = a.id_product) as usable_quantity, a.id_product as id, COUNT(pa.id_product_attribute) as variations'; $this->_join = 'LEFT JOIN `'._DB_PREFIX_.'product_attribute` pa ON (pa.id_product = a.id_product)'.Shop::addSqlAssociation('product_attribute', 'pa', false); $this->_where = 'AND a.is_virtual = 0 AND a.advanced_stock_management = 1 '; $this->_group = 'GROUP BY a.id_product'; // displays informations $this->displayInformation($this->l('This interface allows you to manage product stock and their variations.').'
'); $this->displayInformation($this->l('Through this interface, you can increase and decrease product stock for an given warehouse.')); $this->displayInformation($this->l('Furthermore, you can move product quantities between warehouses, or within one warehouse.').'
'); $this->displayInformation($this->l('If you want to increase quantities of multiple products at once, you can use the "Supply orders" page under the "Stock" menu.').'
'); $this->displayInformation($this->l('Finally, you need to provide the quantity that you\'ll be adding: "Usable for sale" means that this quantity will be available in your shop(s), otherwise it will be considered reserved (i.e. for other purposes).')); } return parent::renderList(); } public function renderDetails() { if (Tools::isSubmit('id_product')) { // override attributes $this->identifier = 'id_product_attribute'; $this->list_id = 'product_attribute'; $this->lang = false; $this->addRowAction('addstock'); $this->addRowAction('prepareRemovestock'); if (count(Warehouse::getWarehouses(true)) > 1) { $this->addRowAction('prepareTransferstock'); } // no link on list rows $this->list_no_link = true; // inits toolbar $this->toolbar_btn = array(); // Get product id $product_id = (int)Tools::getValue('id_product'); // Load product attributes with sql override $this->table = 'product_attribute'; $this->list_id = 'product_attribute'; $this->_select = 'a.id_product_attribute as id, a.id_product, a.reference, a.ean13, a.upc, SUM(s.physical_quantity) as physical_quantity, SUM(s.usable_quantity) as usable_quantity'; $this->_join = 'INNER JOIN `'._DB_PREFIX_.'product` p ON (p.id_product = a.id_product AND p.advanced_stock_management = 1)'; $this->_join .= 'LEFT JOIN `'._DB_PREFIX_.'stock` s ON (s.id_product = a.id_product AND s.id_product_attribute = a.id_product_attribute )'; $this->_where = 'AND a.id_product = '.$product_id; $this->_group = 'GROUP BY a.id_product_attribute'; $this->fields_list['name'] = array( 'title' => $this->l('Name'), 'orderby' => false, 'filter' => false, 'search' => false ); if (Tools::getIsset('id_product_attribute')) { self::$currentIndex = self::$currentIndex.'&id_product='.(int)$product_id; } else { self::$currentIndex = self::$currentIndex.'&id_product='.(int)$product_id.'&detailsproduct'; } $this->processFilter(); return parent::renderList(); } return $this->renderList(); } /** * AdminController::getList() override * @see AdminController::getList() * * @param int $id_lang * @param string|null $order_by * @param string|null $order_way * @param int $start * @param int|null $limit * @param int|bool $id_lang_shop * * @throws PrestaShopException */ public function getList($id_lang, $order_by = null, $order_way = null, $start = 0, $limit = null, $id_lang_shop = false) { parent::getList($id_lang, $order_by, $order_way, $start, $limit, $id_lang_shop); // Check each row to see if there are combinations and get the correct action in consequence $nb_items = count($this->_list); for ($i = 0; $i < $nb_items; $i++) { $item = &$this->_list[$i]; // if it's an ajax request we have to consider manipulating a product variation if (Tools::isSubmit('id_product')) { $item['name'] = Product::getProductName($item['id_product'], empty($item['id_product_attribute']) ? null : $item['id_product_attribute']); // no details for this row $this->addRowActionSkipList('details', array($item['id'])); // skip actions if no quantities if ( ($item['physical_quantity'] <= 0 && $item['usable_quantity'] <= 0) || (empty($item['physical_quantity']) && empty($item['usable_quantity'])) ) { $this->addRowActionSkipList('prepareRemovestock', array($item['id'])); $this->addRowActionSkipList('prepareTransferstock', array($item['id'])); } } // If current product has variations elseif (array_key_exists('variations', $item) && (int)$item['variations'] > 0) { // we have to desactivate stock actions on current row $this->addRowActionSkipList('addstock', array($item['id'])); $this->addRowActionSkipList('prepareRemovestock', array($item['id'])); $this->addRowActionSkipList('prepareTransferstock', array($item['id'])); // does not display these informaions because this product has combinations $item['reference'] = '--'; $item['ean13'] = '--'; $item['upc'] = '--'; } else { //there are no variations of current product, so we don't want to show details action $this->addRowActionSkipList('details', array($item['id'])); // skip actions if no quantities if ($item['physical_quantity'] <= 0 && $item['usable_quantity'] <= 0) { $this->addRowActionSkipList('prepareRemovestock', array($item['id'])); $this->addRowActionSkipList('prepareTransferstock', array($item['id'])); } } // Checks access if (!($this->tabAccess['add'] === '1')) { $this->addRowActionSkipList('addstock', array($item['id'])); } if (!($this->tabAccess['delete'] === '1')) { $this->addRowActionSkipList('removestock', array($item['id'])); } if (!($this->tabAccess['edit'] === '1')) { $this->addRowActionSkipList('transferstock', array($item['id'])); } } } /** * AdminController::renderForm() override * @see AdminController::renderForm() */ public function renderForm() { // gets the product $id_product = (int)Tools::getValue('id_product'); $id_product_attribute = (int)Tools::getValue('id_product_attribute'); // gets warehouses $warehouses_add = Warehouse::getWarehouses(true); $warehouses_remove = Warehouse::getWarehousesByProductId($id_product, $id_product_attribute); // displays warning if no warehouses if (!$warehouses_add) { $this->displayWarning($this->l('You must choose a warehouses before adding stock. See Stock/Warehouses.')); } // switch, in order to display the form corresponding to the current action switch ($this->display) { case 'addstock' : $this->displayInformation($this->l('Moving the mouse cursor over the quantity and price fields will give you the details about the last stock movement.')); // fields in the form $this->prepareAddStockForm($id_product, $id_product_attribute, $warehouses_add); break; case 'removestock' : $id_stock = (int)Tools::getValue('id_stock'); if (!empty($id_stock)) { $this->prepareRemoveStockForm($id_stock); } else { $this->renderList(); } break; case 'transferstock' : $id_stock = (int)Tools::getValue('id_stock'); if (!empty($id_stock)) { $this->prepareTransferStockForm($warehouses_add); } else { $this->renderList(); } break; } $this->initToolbar(); } /** * Prepare add stock form * @param $warehouses_remove */ public function prepareAddStockForm($id_product, $id_product_attribute, $warehouses_add) { // gets the last stock mvt for this product, so we can display the last unit price te and the last quantity added $last_sm_unit_price_te = $this->l('N/A'); $last_sm_quantity = 0; $last_sm_quantity_is_usable = -1; $last_sm = StockMvt::getLastPositiveStockMvt($id_product, $id_product_attribute); // if there is a stock mvt if ($last_sm != false) { $last_sm_currency = new Currency((int)$last_sm['id_currency']); $last_sm_quantity = (int)$last_sm['physical_quantity']; $last_sm_quantity_is_usable = (int)$last_sm['is_usable']; if (Validate::isLoadedObject($last_sm_currency)) { $last_sm_unit_price_te = Tools::displayPrice((float)$last_sm['price_te'], $last_sm_currency); } } //get currencies list $currencies = Currency::getCurrencies(); if (1 < count($currencies)) { array_unshift($currencies, '-'); } $this->fields_form[]['form'] = array( 'legend' => array( 'title' => $this->l('Add a product to your stock.'), 'icon' => 'icon-long-arrow-up' ), 'input' => array( array( 'type' => 'hidden', 'name' => 'is_post', ), array( 'type' => 'hidden', 'name' => 'id_product', ), array( 'type' => 'hidden', 'name' => 'id_product_attribute', ), array( 'type' => 'hidden', 'name' => 'check', ), array( 'type' => 'text', 'label' => $this->l('Product reference'), 'name' => 'reference', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('EAN-13 or JAN barcode'), 'name' => 'ean13', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('UPC barcode'), 'name' => 'upc', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Name'), 'name' => 'name', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Quantity to add'), 'name' => 'quantity', 'maxlength' => 6, 'required' => true, 'hint' => array( $this->l('Indicate the physical quantity of this product that you want to add.'), $this->l('Last physical quantity added: %s items (usable for sale: %s).'), ($last_sm_quantity > 0 ? $last_sm_quantity : $this->l('N/A')), ($last_sm_quantity > 0 ? ($last_sm_quantity_is_usable >= 0 ? $this->l('Yes') : $this->l('No')) : $this->l('N/A'))), ), array( 'type' => 'switch', 'label' => $this->l('Usable for sale?'), 'name' => 'usable', 'required' => true, 'is_bool' => true, 'values' => array( array( 'id' => 'active_on', 'value' => 1, 'label' => $this->l('Enabled') ), array( 'id' => 'active_off', 'value' => 0, 'label' => $this->l('Disabled') ) ), 'hint' => $this->l('Is this quantity ready to be displayed in your shop, or is it reserved in the warehouse for other purposes?') ), array( 'type' => 'select', 'label' => $this->l('Warehouse'), 'name' => 'id_warehouse', 'required' => true, 'options' => array( 'query' => $warehouses_add, 'id' => 'id_warehouse', 'name' => 'name' ), 'hint' => $this->l('Please select the warehouse that you\'ll be adding products to.') ), array( 'type' => 'text', 'label' => $this->l('Unit price (tax excl.)'), 'name' => 'price', 'required' => true, 'size' => 10, 'maxlength' => 10, 'hint' => array( $this->l('Unit purchase price or unit manufacturing cost for this product (tax excl.).'), sprintf($this->l('Last unit price (tax excl.): %s.'), $last_sm_unit_price_te), ) ), array( 'type' => 'select', 'label' => $this->l('Currency'), 'name' => 'id_currency', 'required' => true, 'options' => array( 'query' => $currencies, 'id' => 'id_currency', 'name' => 'name' ), 'hint' => $this->l('The currency associated to the product unit price.'), ), array( 'type' => 'select', 'label' => $this->l('Label'), 'name' => 'id_stock_mvt_reason', 'required' => true, 'options' => array( 'query' => StockMvtReason::getStockMvtReasonsWithFilter($this->context->language->id, array(Configuration::get('PS_STOCK_MVT_TRANSFER_TO')), 1), 'id' => 'id_stock_mvt_reason', 'name' => 'name' ), 'hint' => $this->l('Label used in stock movements.'), ), ), 'submit' => array( 'title' => $this->l('Add to stock') ) ); $this->fields_value['usable'] = 1; } /** * Prepare remove stock form * @param $warehouses_remove */ public function prepareRemoveStockForm($id_stock) { $this->fields_form[]['form'] = array( 'legend' => array( 'title' => $this->l('Remove the product from your stock.'), 'icon' => 'icon-long-arrow-down' ), 'input' => array( array( 'type' => 'hidden', 'name' => 'is_post', ), array( 'type' => 'hidden', 'name' => 'id_product', ), array( 'type' => 'hidden', 'name' => 'id_product_attribute', ), array( 'type' => 'hidden', 'name' => 'id_stock', ), array( 'type' => 'hidden', 'name' => 'check', ), array( 'type' => 'text', 'label' => $this->l('Product reference'), 'name' => 'reference', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('EAN-13 or JAN barcode'), 'name' => 'ean13', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Name'), 'name' => 'name', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Warehouse'), 'name' => 'warehouse_name', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Physical quantity'), 'name' => 'physical_products_quantity', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Usable quantity'), 'name' => 'usable_products_quantity', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Quantity to remove'), 'name' => 'quantity', 'maxlength' => 6, 'required' => true, 'hint' => $this->l('Indicate the physical quantity of this product that you want to remove.'), ), array( 'type' => 'switch', 'label' => $this->l('Usable for sale'), 'name' => 'usable', 'required' => true, 'is_bool' => true, 'values' => array( array( 'id' => 'active_on', 'value' => 1, 'label' => $this->l('Enabled') ), array( 'id' => 'active_off', 'value' => 0, 'label' => $this->l('Disabled') ) ), 'hint' => $this->l('Do you want to remove this quantity from the usable quantity (yes) or the physical quantity (no)?') ), array( 'type' => 'select', 'label' => $this->l('Label'), 'name' => 'id_stock_mvt_reason', 'required' => true, 'options' => array( 'query' => StockMvtReason::getStockMvtReasonsWithFilter($this->context->language->id, array(Configuration::get('PS_STOCK_MVT_TRANSFER_FROM')), -1), 'id' => 'id_stock_mvt_reason', 'name' => 'name' ), 'hint' => $this->l('Label used in stock movements.'), ), ), 'submit' => array( 'title' => $this->l('Remove from stock') ) ); } /** * Prepare transfer stock form * @param $warehouses_add * @param $warehouses_remove */ public function prepareTransferStockForm($warehouses_add) { $this->fields_form[]['form'] = array( 'legend' => array( 'title' => $this->l('Transfer a product from one warehouse to another'), 'icon' => 'icon-share-alt' ), 'input' => array( array( 'type' => 'hidden', 'name' => 'is_post', ), array( 'type' => 'hidden', 'name' => 'id_product', ), array( 'type' => 'hidden', 'name' => 'id_product_attribute', ), array( 'type' => 'hidden', 'name' => 'id_stock', ), array( 'type' => 'hidden', 'name' => 'check', ), array( 'type' => 'text', 'label' => $this->l('Product reference'), 'name' => 'reference', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('EAN-13 or JAN barcode'), 'name' => 'ean13', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Name'), 'name' => 'name', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Warehouse'), 'name' => 'warehouse_name', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Physical quantity'), 'name' => 'physical_products_quantity', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Usable quantity'), 'name' => 'usable_products_quantity', 'disabled' => true, ), array( 'type' => 'text', 'label' => $this->l('Quantity to transfer'), 'name' => 'quantity', 'maxlength' => 6, 'required' => true, 'hint' => $this->l('Indicate the physical quantity of this product that you want to transfer.') ), array( 'type' => 'switch', 'label' => $this->l('Is this product usable for sale in your source warehouse?'), 'name' => 'usable_from', 'required' => true, 'is_bool' => true, 'values' => array( array( 'id' => 'active_on', 'value' => 1, 'label' => $this->l('Yes') ), array( 'id' => 'active_off', 'value' => 0, 'label' => $this->l('No') ) ), 'hint' => $this->l('Is this the usable quantity for sale?') ), array( 'type' => 'select', 'label' => $this->l('Destination warehouse'), 'name' => 'id_warehouse_to', 'required' => true, 'options' => array( 'query' => $warehouses_add, 'id' => 'id_warehouse', 'name' => 'name' ), 'hint' => $this->l('Select the warehouse you\'d like to transfer your product(s) to. ') ), array( 'type' => 'switch', 'label' => $this->l('Is this product usable for sale in your destination warehouse?'), 'name' => 'usable_to', 'required' => true, 'class' => 't', 'is_bool' => true, 'values' => array( array( 'id' => 'active_on', 'value' => 1, 'label' => $this->l('Yes') ), array( 'id' => 'active_off', 'value' => 0, 'label' => $this->l('No') ) ), 'hint' => $this->l('Do you want it to be for sale/usable?') ), ), 'submit' => array( 'title' => $this->l('Transfer') ) ); } /** * Call if no GET id_stock, display a detail stock for a product/product_attribute (various price) * @param $id_product * @param $id_product_attribute */ public function previousManagementStock($id_product, $id_product_attribute) { $this->fields_list = array( 'reference' => array( 'title' => $this->l('Reference'), 'align' => 'center', 'havingFilter' => true, ), 'warehouse' => array( 'title' => $this->l('Warehouse'), 'havingFilter' => true ), 'price_te' => array( 'title' => $this->l('Price (tax excl.)'), 'orderby' => true, 'search' => false, 'type' => 'price', 'currency' => true, ), 'valuation' => array( 'title' => $this->l('Valuation'), 'orderby' => false, 'search' => false, 'type' => 'price', 'currency' => true, 'hint' => $this->l('Total value of the physical quantity. The sum (for all prices) is not available for all warehouses, please filter by warehouse.') ), 'physical_quantity' => array( 'title' => $this->l('Physical quantity'), 'class' => 'fixed-width-xs', 'align' => 'center', 'orderby' => true, 'search' => false ), 'usable_quantity' => array( 'title' => $this->l('Usable quantity'), 'class' => 'fixed-width-xs', 'align' => 'center', 'orderby' => true, 'search' => false, ), ); $this->display = null; $this->identifier = 'id_stock'; $this->page_header_toolbar_btn['back_to_list'] = array( 'href' => Context::getContext()->link->getAdminLink('AdminStockManagement'), 'desc' => $this->l('Back to list', null, null, false), 'icon' => 'process-icon-back' ); // sets actions5 $this->addRowAction('removestock'); if (count(Warehouse::getWarehouses(true)) > 1) { $this->addRowAction('transferstock'); } // no link on list rows $this->list_no_link = true; $this->table = 'stock'; $this->list_id = 'stock'; $this->lang = false; $lang_id = (int)$this->context->language->id; $id_warehouse = Tools::getValue('id_warehouse', -1); $this->_select = 'w.id_currency, a.id_product as id, (a.price_te * a.physical_quantity) as valuation, w.name as warehouse'; $this->_join = 'INNER JOIN `'._DB_PREFIX_.'product` p ON (p.id_product = a.id_product AND p.advanced_stock_management = 1)'; $this->_join .= ' LEFT JOIN `'._DB_PREFIX_.'warehouse` AS w ON (w.id_warehouse = a.id_warehouse)'; $this->_join .= ' RIGHT JOIN `'._DB_PREFIX_.'product_lang` AS b ON (b.id_product = a.id_product)'; $this->_where = 'AND a.id_product = '.(int)$id_product.' AND a.id_product_attribute = '.(int)$id_product_attribute; $this->_where .= ' AND b.id_lang = '.(int)$lang_id.' AND b.id_shop = p.id_shop_default'; if ($id_warehouse != -1) { $this->_where .= ' AND a.id_warehouse = '.(int)$id_warehouse; } $this->_orderBy = 'b.name'; $this->_orderWay = 'ASC'; } /** * AdminController::postProcess() override * @see AdminController::postProcess() */ public function postProcess() { parent::postProcess(); // Checks access if (Tools::isSubmit('addStock') && !($this->tabAccess['add'] === '1')) { $this->errors[] = Tools::displayError('You do not have the required permission to add stock.'); } if (Tools::isSubmit('removeStock') && !($this->tabAccess['delete'] === '1')) { $this->errors[] = Tools::displayError('You do not have the required permission to delete stock'); } if (Tools::isSubmit('transferStock') && !($this->tabAccess['edit'] === '1')) { $this->errors[] = Tools::displayError('You do not have the required permission to transfer stock.'); } if (count($this->errors)) { return; } // Global checks when add / remove / transfer product if ( (Tools::isSubmit('addstock') || Tools::isSubmit('removestock') || Tools::isSubmit('transferstock')) && Tools::isSubmit('is_post') ) { // get product ID $id_product = (int)Tools::getValue('id_product', 0); if ($id_product <= 0) { $this->errors[] = Tools::displayError('The selected product is not valid.'); } // get product_attribute ID $id_product_attribute = (int)Tools::getValue('id_product_attribute', 0); // check the product hash $check = Tools::getValue('check', ''); $check_valid = md5(_COOKIE_KEY_.$id_product.$id_product_attribute); if ($check != $check_valid) { $this->errors[] = Tools::displayError('The selected product is not valid.'); } // get quantity and check that the post value is really an integer // If it's not, we have nothing to do $quantity = Tools::getValue('quantity', 0); if (!is_numeric($quantity) || (int)$quantity <= 0) { $this->errors[] = Tools::displayError('The quantity value is not valid.'); } $quantity = (int)$quantity; $id_stock = (int)Tools::getValue('id_stock', 0); $token = Tools::getValue('token') ? Tools::getValue('token') : $this->token; $redirect = self::$currentIndex.'&token='.$token; } if (Tools::isSubmit('submitFilter') && Tools::getIsset('detailsproduct')) { $id_product = (int)Tools::getValue('id_product', 0); $token = Tools::getValue('token') ? Tools::getValue('token') : $this->token; $redirect = self::$currentIndex.'&id_product='.$id_product.'&detailsproduct&token='.$token; Tools::redirectAdmin($redirect); } // Global checks when add / remove product if ((Tools::isSubmit('addstock') || Tools::isSubmit('removestock')) && Tools::isSubmit('is_post')) { $stockAttributes = $this->getStockAttributes(); $id_warehouse = $stockAttributes['warehouse_id']; if ($id_warehouse <= 0 || !Warehouse::exists($id_warehouse)) { $this->errors[] = Tools::displayError('The selected warehouse is not valid.'); } // get stock movement reason id $id_stock_mvt_reason = (int)Tools::getValue('id_stock_mvt_reason', 0); if ($id_stock_mvt_reason <= 0 || !StockMvtReason::exists($id_stock_mvt_reason)) { $this->errors[] = Tools::displayError('The reason is not valid.'); } // get usable flag $usable = Tools::getValue('usable', null); if (is_null($usable)) { $this->errors[] = Tools::displayError('You have to specify whether the product quantity is usable for sale on shops or not.'); } $usable = (bool)$usable; } if (Tools::isSubmit('addstock') && Tools::isSubmit('is_post')) { // get product unit price $price = str_replace(',', '.', Tools::getValue('price', 0)); if (!is_numeric($price)) { $this->errors[] = Tools::displayError('The product price is not valid.'); } $price = round(floatval($price), 6); // get product unit price currency id $id_currency = (int)Tools::getValue('id_currency', 0); if ($id_currency <= 0 || (!($result = Currency::getCurrency($id_currency)) || empty($result))) { $this->errors[] = Tools::displayError('The selected currency is not valid.'); } // if all is ok, add stock if (count($this->errors) == 0) { $warehouse = new Warehouse($id_warehouse); // convert price to warehouse currency if needed if ($id_currency != $warehouse->id_currency) { // First convert price to the default currency $price_converted_to_default_currency = Tools::convertPrice($price, $id_currency, false); // Convert the new price from default currency to needed currency $price = Tools::convertPrice($price_converted_to_default_currency, $warehouse->id_currency, true); } // add stock $stock_manager = StockManagerFactory::getManager(); if ($stock_manager->addProduct($id_product, $id_product_attribute, $warehouse, $quantity, $id_stock_mvt_reason, $price, $usable)) { // Create warehouse_product_location entry if we add stock to a new warehouse $id_wpl = (int)WarehouseProductLocation::getIdByProductAndWarehouse($id_product, $id_product_attribute, $id_warehouse); if (!$id_wpl) { $wpl = new WarehouseProductLocation(); $wpl->id_product = (int)$id_product; $wpl->id_product_attribute = (int)$id_product_attribute; $wpl->id_warehouse = (int)$id_warehouse; $wpl->save(); } StockAvailable::synchronize($id_product); if (Tools::isSubmit('addstockAndStay')) { $redirect = self::$currentIndex.'&id_product='.(int)$id_product; if ($id_product_attribute) { $redirect .= '&id_product_attribute='.(int)$id_product_attribute; } $redirect .= '&addstock&token='.$token; } Tools::redirectAdmin($redirect.'&conf=1'); } else { $this->errors[] = Tools::displayError('An error occurred. No stock was added.'); } } } if (Tools::isSubmit('removestock') && Tools::isSubmit('is_post')) { $stockAttributes = $this->getStockAttributes(); // if all is ok, remove stock if (count($this->errors) == 0) { $warehouse = new Warehouse($id_warehouse); // remove stock $stock_manager = StockManagerFactory::getManager(); $removed_products = $stock_manager->removeProduct( $stockAttributes['product_id'], $stockAttributes['product_attribute_id'], $stockAttributes['warehouse'], $quantity, $id_stock_mvt_reason, $usable, $id_order = null, $ignore_pack = 0, $employee = null, $stockAttributes['stock'] ); if (count($removed_products) > 0) { StockAvailable::synchronize($stockAttributes['product_id']); Tools::redirectAdmin($redirect.'&conf=2'); } else { $stock = $stockAttributes['stock']; $testedQuantity = (int)$stock->physical_quantity; $errorMessage = Tools::displayError('You don\'t have enough physical quantity. Cannot remove %d items out of %d.'); if ($usable) { $testedQuantity = (int)$stock->usable_quantity; $errorMessage = Tools::displayError('You don\'t have enough usable quantity. Cannot remove %d items out of %d.'); } if ($testedQuantity < $quantity) { $this->errors[] = sprintf( $errorMessage, (int)$quantity, (int)$testedQuantity ); } } } } if (Tools::isSubmit('transferstock') && Tools::isSubmit('is_post')) { $stockAttributes = $this->getStockAttributes(); // get destination warehouse id $id_warehouse_to = (int)Tools::getValue('id_warehouse_to', 0); if ($id_warehouse_to <= 0 || !Warehouse::exists($id_warehouse_to)) { $this->errors[] = Tools::displayError('The destination warehouse is not valid.'); } // get usable flag for source warehouse $usable_from = Tools::getValue('usable_from', null); if (is_null($usable_from)) { $this->errors[] = Tools::displayError('You have to specify whether the product quantity in your source warehouse(s) is ready for sale or not.'); } $usable_from = (bool)$usable_from; // get usable flag for destination warehouse $usable_to = Tools::getValue('usable_to', null); if (is_null($usable_to)) { $this->errors[] = Tools::displayError('You have to specify whether the product quantity in your destination warehouse(s) is ready for sale or not.'); } $usable_to = (bool)$usable_to; // if we can process stock transfers if (count($this->errors) == 0) { // transfer stock $stock_manager = StockManagerFactory::getManager(); $is_transfer = $stock_manager->transferBetweenWarehouses( $stockAttributes['product_id'], $stockAttributes['product_attribute_id'], $quantity, $stockAttributes['warehouse_id'], $id_warehouse_to, $usable_from, $usable_to ); StockAvailable::synchronize($stockAttributes['product_id']); if ($is_transfer) { Tools::redirectAdmin($redirect.'&conf=3'); } else { $this->errors[] = Tools::displayError('It is not possible to transfer the specified quantity. No stock was transferred.'); } } } } /** * assign default action in toolbar_btn smarty var, if they are not set. * uses override to specifically add, modify or remove items * */ public function initToolbar() { switch ($this->display) { case 'addstock': $this->toolbar_btn['save-and-stay'] = array( 'short' => 'SaveAndStay', 'href' => '#', 'desc' => $this->l('Save and stay'), ); case 'removestock': case 'transferstock': case 'previousManagement': $this->toolbar_btn['save'] = array( 'href' => '#', 'desc' => $this->l('Save') ); // Default cancel button - like old back link $back = Tools::safeOutput(Tools::getValue('back', '')); if (empty($back)) { $back = self::$currentIndex.'&token='.$this->token; } $this->toolbar_btn['cancel'] = array( 'href' => $back, 'desc' => $this->l('Cancel') ); break; default: parent::initToolbar(); } } /** * AdminController::init() override * @see AdminController::init() */ public function init() { parent::init(); $id_product = (int)Tools::getValue('id_product'); if (!empty($id_product)) { $id_product_attribute = (int)Tools::getValue('id_product_attribute'); $product_name = Product::getProductName($id_product, $id_product_attribute); } if (Tools::isSubmit('addstock')) { $this->display = 'addstock'; $this->toolbar_title = $this->l('Stock: Add a product'); } if (Tools::isSubmit('removestock')) { $this->display = 'removestock'; $this->toolbar_title = $this->l('Stock: Remove a product'); } if (Tools::isSubmit('transferstock')) { $this->display = 'transferstock'; $this->toolbar_title = $this->l('Stock: Transfer a product'); } if (!empty($product_name)) { $this->toolbar_title .= empty($this->toolbar_title) ? $product_name : ' - ' . $product_name; } } /** * AdminController::initContent() override * @see AdminController::initContent() */ public function initContent() { if (!Configuration::get('PS_ADVANCED_STOCK_MANAGEMENT')) { $this->warnings[md5('PS_ADVANCED_STOCK_MANAGEMENT')] = $this->l('You need to activate the Advanced Stock Management feature prior to using this feature.'); return false; } if ( ($this->display == 'removestock' || $this->display == 'transferstock') && !Tools::isSubmit('id_stock') ) { $this->errors[] = Tools::displayError('An error occurred while loading the object.'); return false; } // Manage the add stock form if ($this->display == 'addstock' || $this->display == 'removestock' || $this->display == 'transferstock') { if (Tools::isSubmit('id_product') || Tools::isSubmit('id_product_attribute') || Tools::isSubmit('id_stock')) { $stockAttributes = $this->getStockAttributes(); $id_product = $stockAttributes['product_id']; $id_product_attribute = $stockAttributes['product_attribute_id']; $id_warehouse = $stockAttributes['warehouse_id']; $id_stock = $stockAttributes['stock_id']; $stock = $stockAttributes['stock']; $warehouse = $stockAttributes['warehouse']; $product_is_valid = false; $is_virtual = false; $lang_id = $this->context->language->id; $default_wholesale_price = 0; // try to load product attribute first if ($id_product_attribute > 0) { // try to load product attribute $combination = new Combination($id_product_attribute); if (Validate::isLoadedObject($combination)) { $product_is_valid = true; $id_product = $combination->id_product; $reference = $combination->reference; $ean13 = $combination->ean13; $upc = $combination->upc; $manufacturer_reference = $combination->supplier_reference; // get the full name for this combination $query = new DbQuery(); $query->select('IFNULL(CONCAT(pl.`name`, \' : \', GROUP_CONCAT(agl.`name`, \' - \', al.`name` SEPARATOR \', \')),pl.`name`) as name'); $query->from('product_attribute', 'a'); $query->join('INNER JOIN '._DB_PREFIX_.'product_lang pl ON (pl.`id_product` = a.`id_product` AND pl.`id_lang` = '.(int)$lang_id.') LEFT JOIN '._DB_PREFIX_.'product_attribute_combination pac ON (pac.`id_product_attribute` = a.`id_product_attribute`) LEFT JOIN '._DB_PREFIX_.'attribute atr ON (atr.`id_attribute` = pac.`id_attribute`) LEFT JOIN '._DB_PREFIX_.'attribute_lang al ON (al.`id_attribute` = atr.`id_attribute` AND al.`id_lang` = '.(int)$lang_id.') LEFT JOIN '._DB_PREFIX_.'attribute_group_lang agl ON (agl.`id_attribute_group` = atr.`id_attribute_group` AND agl.`id_lang` = '.(int)$lang_id.')' ); $query->where('a.`id_product_attribute` = '.$id_product_attribute); $name = Db::getInstance()->getValue($query); $p = new Product($id_product, false, $lang_id); $default_wholesale_price = $combination->wholesale_price > 0 ? $combination->wholesale_price : $p->wholesale_price; } } // try to load a simple product else { $product = new Product($id_product, false, $lang_id); if (is_int($product->id)) { $product_is_valid = true; $reference = $product->reference; $ean13 = $product->ean13; $upc = $product->upc; $name = $product->name; $manufacturer_reference = $product->supplier_reference; $is_pack = $product->cache_is_pack; $is_virtual = $product->is_virtual; $default_wholesale_price = $product->wholesale_price; } } if ($product_is_valid === true && $is_virtual == false) { // init form $this->renderForm(); $this->getlanguages(); $helper = new HelperForm(); $this->initPageHeaderToolbar(); // Check if form template has been overriden if (file_exists($this->context->smarty->getTemplateDir(0).'/'.$this->tpl_folder.'form.tpl')) { $helper->tpl = $this->tpl_folder.'form.tpl'; } $this->setHelperDisplay($helper); $helper->submit_action = $this->display; $helper->id = null; // no display standard hidden field in the form $helper->languages = $this->_languages; $helper->default_form_language = $this->default_form_language; $helper->allow_employee_form_lang = $this->allow_employee_form_lang; $helper->show_cancel_button = true; $helper->back_url = $this->context->link->getAdminLink('AdminStockManagement'); $helper->fields_value = array( 'id_product' => $id_product, 'id_product_attribute' => $id_product_attribute, 'id_stock' => $id_stock, 'reference' => $reference, 'manufacturer_reference' => $manufacturer_reference, 'name' => $name, 'ean13' => $ean13, 'upc' => $upc, 'warehouse_name' => empty($warehouse) ? false: $warehouse->name, 'check' => md5(_COOKIE_KEY_.$id_product.$id_product_attribute), 'quantity' => Tools::getValue('quantity', ''), 'id_warehouse' => $id_warehouse, 'usable' => $this->fields_value['usable'] ? $this->fields_value['usable'] : Tools::getValue('usable', 1), 'price' => Tools::getValue('price', (float)Tools::convertPrice($default_wholesale_price, null)), 'id_currency' => Tools::getValue('id_currency', ''), 'id_stock_mvt_reason' => Tools::getValue('id_stock_mvt_reason', ''), 'is_post' => 1, ); if ($this->display == 'addstock') { $_POST['id_product'] = (int)$id_product; } if ($this->display == 'removestock' || $this->display == 'transferstock') { $helper->fields_value['id_stock'] = $id_stock; $helper->fields_value['physical_products_quantity'] = $stock->physical_quantity; $helper->fields_value['usable_products_quantity'] = $stock->usable_quantity; $helper->fields_value['warehouse_name'] = empty($warehouse) ? false: $warehouse->name; } if ($this->display == 'transferstock') { $helper->fields_value['id_warehouse_from'] = Tools::getValue('id_warehouse_from', ''); $helper->fields_value['id_warehouse_to'] = Tools::getValue('id_warehouse_to', ''); $helper->fields_value['usable_from'] = Tools::getValue('usable_from', '1'); $helper->fields_value['usable_to'] = Tools::getValue('usable_to', '1'); } $this->content .= $helper->generateForm($this->fields_form); $this->context->smarty->assign(array( 'content' => $this->content, 'show_page_header_toolbar' => $this->show_page_header_toolbar, 'page_header_toolbar_title' => $this->page_header_toolbar_title, 'page_header_toolbar_btn' => $this->page_header_toolbar_btn )); } else { $this->errors[] = Tools::displayError('The specified product is not valid.'); } } } else { parent::initContent(); } } /** * Display addstock action link * @param string $token the token to add to the link * @param int $id the identifier to add to the link * @return string */ public function displayAddstockLink($token = null, $id) { if (!array_key_exists('AddStock', self::$cache_lang)) { self::$cache_lang['AddStock'] = $this->l('Add stock'); } $this->context->smarty->assign(array( 'href' => self::$currentIndex. '&'.$this->identifier.'='.$id. '&addstock&token='.($token != null ? $token : $this->token), 'action' => self::$cache_lang['AddStock'], )); return $this->context->smarty->fetch('helpers/list/list_action_addstock.tpl'); } /** * Display removestock action link * @param string $token the token to add to the link * @param int $id the identifier to add to the link * @return string */ public function displayRemovestockLink($token = null, $id) { if (!array_key_exists('RemoveStock', self::$cache_lang)) { self::$cache_lang['RemoveStock'] = $this->l('Remove stock'); } $this->context->smarty->assign(array( 'href' => self::$currentIndex. '&'.$this->identifier.'='.$id. '&removestock&token='.($token != null ? $token : $this->token), 'action' => self::$cache_lang['RemoveStock'], )); return $this->context->smarty->fetch('helpers/list/list_action_removestock.tpl'); } /** * Display transferstock action link * @param string $token the token to add to the link * @param int $id the identifier to add to the link * @return string */ public function displayTransferstockLink($token = null, $id) { if (!array_key_exists('TransferStock', self::$cache_lang)) { self::$cache_lang['TransferStock'] = $this->l('Transfer stock'); } $this->context->smarty->assign(array( 'href' => self::$currentIndex. '&'.$this->identifier.'='.$id. '&transferstock&token='.($token != null ? $token : $this->token), 'action' => self::$cache_lang['TransferStock'], )); return $this->context->smarty->fetch('helpers/list/list_action_transferstock.tpl'); } /** * Display removestock action link (fake link because don't have id_stock) * @param string $token the token to add to the link * @param int $id the identifier to add to the link * @return string */ public function displayPrepareRemovestockLink($token = null, $id) { if (!array_key_exists('RemoveStock', self::$cache_lang)) { self::$cache_lang['RemoveStock'] = $this->l('Remove stock'); } if (Tools::getIsset('detailsproduct')) { self::$currentIndex = str_replace('&detailsproduct', '', self::$currentIndex); } $this->context->smarty->assign(array( 'href' => self::$currentIndex. '&'.$this->identifier.'='.$id. '&token='.($token != null ? $token : $this->token), 'action' => self::$cache_lang['RemoveStock'], )); return $this->context->smarty->fetch('helpers/list/list_action_removestock.tpl'); } /** * Display transferstock action link (fake link because don't have id_stock) * @param string $token the token to add to the link * @param int $id the identifier to add to the link * @return string */ public function displayPrepareTransferstockLink($token = null, $id) { if (!array_key_exists('TransferStock', self::$cache_lang)) { self::$cache_lang['TransferStock'] = $this->l('Transfer stock'); } $this->context->smarty->assign(array( 'href' => self::$currentIndex. '&'.$this->identifier.'='.$id. '&token='.($token != null ? $token : $this->token), 'action' => self::$cache_lang['TransferStock'], )); return $this->context->smarty->fetch('helpers/list/list_action_transferstock.tpl'); } public function initProcess() { if (!Configuration::get('PS_ADVANCED_STOCK_MANAGEMENT')) { $this->warnings[md5('PS_ADVANCED_STOCK_MANAGEMENT')] = $this->l('You need to activate advanced stock management prior to using this feature.'); return false; } if (Tools::getIsset('detailsproduct')) { $this->list_id = 'product_attribute'; if (isset($_POST['submitReset'.$this->list_id])) { $this->processResetFilters(); } } else { $this->list_id = 'product'; } parent::initProcess(); } /** * @return array * @throws Exception */ protected function getStockAttributes() { $product_id = (int)Tools::getValue('id_product', 0); $product_attribute_id = (int)Tools::getValue('id_product_attribute', 0); $stock_id = (int)Tools::getValue('id_stock', 0); $stock = null; $warehouse_id = Tools::getValue('id_warehouse', null); $warehouse = null; if ($stock_id > 0) { $stock = new Stock($stock_id); $product_id = (int)$stock->id_product; $product_attribute_id = (int)$stock->id_product_attribute; $warehouse = new Warehouse((int)$stock->id_warehouse); $warehouse_id = $warehouse->id; } return array( 'product_id' => $product_id, 'product_attribute_id' => $product_attribute_id, 'stock_id' => $stock_id, 'stock' => $stock, 'warehouse_id' => $warehouse_id, 'warehouse' => $warehouse, ); } }