* @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,
);
}
}