* @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 Warehouse $object */ class AdminWarehousesControllerCore extends AdminController { public function __construct() { $this->bootstrap = true; $this->table = 'warehouse'; $this->className = 'Warehouse'; $this->deleted = true; $this->lang = false; $this->multishop_context = Shop::CONTEXT_ALL; $this->fields_list = array( 'id_warehouse' => array( 'title' => $this->l('ID'), 'width' => 50, ), 'reference' => array( 'title' => $this->l('Reference'), ), 'name' => array( 'title' => $this->l('Name'), ), 'management_type' => array( 'title' => $this->l('Management type'), ), 'employee' => array( 'title' => $this->l('Manager'), 'filter_key' => 'employee', 'havingFilter' => true ), 'location' => array( 'title' => $this->l('Location'), 'orderby' => false, 'filter' => false, 'search' => false, ), 'contact' => array( 'title' => $this->l('Phone Number'), 'orderby' => false, 'filter' => false, 'search' => false, ), ); $this->bulk_actions = array( 'delete' => array( 'text' => $this->l('Delete selected'), 'icon' => 'icon-trash', 'confirm' => $this->l('Delete selected items?') ) ); parent::__construct(); } public function initPageHeaderToolbar() { if (empty($this->display)) { $this->page_header_toolbar_btn['new_warehouse'] = array( 'href' => self::$currentIndex.'&addwarehouse&token='.$this->token, 'desc' => $this->l('Add new warehouse', null, null, false), 'icon' => 'process-icon-new' ); } parent::initPageHeaderToolbar(); } /** * AdminController::renderList() override * @see AdminController::renderList() */ public function renderList() { // removes links on rows $this->list_no_link = true; // adds actions on rows $this->addRowAction('edit'); $this->addRowAction('view'); $this->addRowAction('delete'); // query: select $this->_select = ' reference, name, management_type, CONCAT(e.lastname, \' \', e.firstname) as employee, ad.phone as contact, CONCAT(ad.city, \' - \', c.iso_code) as location'; // query: join $this->_join = ' LEFT JOIN `'._DB_PREFIX_.'employee` e ON (e.id_employee = a.id_employee) LEFT JOIN `'._DB_PREFIX_.'address` ad ON (ad.id_address = a.id_address) LEFT JOIN `'._DB_PREFIX_.'country` c ON (c.id_country = ad.id_country)'; $this->_use_found_rows = false; // display help informations $this->displayInformation($this->l('This interface allows you to manage your warehouses.').'
'); $this->displayInformation($this->l('Before adding stock in your warehouses, you should check the default currency used.').'
'); $this->displayInformation($this->l('You should also check the management type (according to the law in your country), the valuation currency and its associated carriers and shops.').'
'); $this->displayInformation($this->l('You can also see detailed information about your stock, such as its overall value, the number of products and quantities stored, etc.')); $this->displayInformation($this->l('Be careful! Products from different warehouses will need to be shipped in different packages.')); return parent::renderList(); } /** * AdminController::renderForm() override * @see AdminController::renderForm() */ public function renderForm() { /** @var Warehouse $obj */ // loads current warehouse if (!($obj = $this->loadObject(true))) { return; } // gets the manager of the warehouse $query = new DbQuery(); $query->select('id_employee, CONCAT(lastname," ",firstname) as name'); $query->from('employee'); $query->where('active = 1'); $employees_array = Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS($query); // sets the title of the toolbar if (Tools::isSubmit('add'.$this->table)) { $this->toolbar_title = $this->l('Stock: Create a warehouse'); } else { $this->toolbar_title = $this->l('Stock: Warehouse management'); } $tmp_addr = new Address(); $res = $tmp_addr->getFieldsRequiredDatabase(); $required_fields = array(); foreach ($res as $row) { $required_fields[(int)$row['id_required_field']] = $row['field_name']; } // sets the fields of the form $this->fields_form = array( 'legend' => array( 'title' => $this->l('Warehouse information'), 'icon' => 'icon-pencil' ), 'input' => array( array( 'type' => 'hidden', 'name' => 'id_address', ), array( 'type' => 'text', 'label' => $this->l('Reference'), 'name' => 'reference', 'maxlength' => 32, 'required' => true, 'hint' => $this->l('Reference for this warehouse.'), ), array( 'type' => 'text', 'label' => $this->l('Name'), 'name' => 'name', 'maxlength' => 45, 'required' => true, 'hint' => array( $this->l('Name of this warehouse.'), $this->l('Invalid characters:').' !<>,;?=+()@#"�{}_$%:', ) ), array( 'type' => 'text', 'label' => $this->l('Phone'), 'name' => 'phone', 'maxlength' => 16, 'hint' => $this->l('Phone number for this warehouse.'), 'required' => in_array('phone', $required_fields) ), array( 'type' => 'text', 'label' => $this->l('Mobile phone'), 'name' => 'phone_mobile', 'required' => in_array('phone_mobile', $required_fields), 'maxlength' => 16, 'hint' => $this->l('Mobile phone number for this supplier.') ), array( 'type' => 'text', 'label' => $this->l('Address'), 'name' => 'address', 'maxlength' => 128, 'required' => true ), array( 'type' => 'text', 'label' => $this->l('Address').' (2)', 'name' => 'address2', 'maxlength' => 128, 'hint' => $this->l('Complementary address (optional).'), 'required' => in_array('address2', $required_fields) ), array( 'type' => 'text', 'label' => $this->l('Zip/postal code'), 'name' => 'postcode', 'maxlength' => 12, 'required' => in_array('postcode', $required_fields) ), array( 'type' => 'text', 'label' => $this->l('City'), 'name' => 'city', 'maxlength' => 32, 'required' => true, ), array( 'type' => 'select', 'label' => $this->l('Country'), 'name' => 'id_country', 'required' => true, 'default_value' => (int)$this->context->country->id, 'options' => array( 'query' => Country::getCountries($this->context->language->id, false), 'id' => 'id_country', 'name' => 'name', ), 'hint' => $this->l('Country of location of the warehouse.') ), array( 'type' => 'select', 'label' => $this->l('State'), 'name' => 'id_state', 'required' => true, 'options' => array( 'query' => array(), 'id' => 'id_state', 'name' => 'name' ) ), array( 'type' => 'select', 'label' => $this->l('Manager'), 'name' => 'id_employee', 'required' => true, 'options' => array( 'query' => $employees_array, 'id' => 'id_employee', 'name' => 'name' ), ), array( 'type' => 'swap', 'label' => $this->l('Carriers'), 'name' => 'ids_carriers', 'required' => false, 'multiple' => true, 'options' => array( 'query' => Carrier::getCarriers($this->context->language->id, false, false, false, null, Carrier::ALL_CARRIERS), 'id' => 'id_reference', 'name' => 'name' ), 'hint' => array( $this->l('Associated carriers.'), $this->l('You can choose which carriers can ship orders from particular warehouses.'), $this->l('If you do not select any carrier, all the carriers will be able to ship from this warehouse.'), ), 'desc' => $this->l('If no carrier is selected, all the carriers will be allowed to ship from this warehouse. Use CTRL+Click to select more than one carrier.'), ), ), ); // Shop Association if (Shop::isFeatureActive()) { $this->fields_form['input'][] = array( 'type' => 'shop', 'label' => $this->l('Shop association'), 'name' => 'checkBoxShopAsso', 'disable_shared' => Shop::SHARE_STOCK ); } // if it is still possible to change currency valuation and management type if (Tools::isSubmit('addwarehouse') || Tools::isSubmit('submitAddwarehouse')) { // adds input management type $this->fields_form['input'][] = array( 'type' => 'select', 'label' => $this->l('Management type'), 'hint' => $this->l('Inventory valuation method. Be careful! You won\'t be able to change this value later!'), 'name' => 'management_type', 'required' => true, 'options' => array( 'query' => array( array( 'id' => 'WA', 'name' => $this->l('Weighted Average') ), array( 'id' => 'FIFO', 'name' => $this->l('First In, First Out') ), array( 'id' => 'LIFO', 'name' => $this->l('Last In, First Out') ), ), 'id' => 'id', 'name' => 'name' ), ); // adds input valuation currency $this->fields_form['input'][] = array( 'type' => 'select', 'label' => $this->l('Stock valuation currency'), 'hint' => $this->l('Be careful! You won\'t be able to change this value later!'), 'name' => 'id_currency', 'required' => true, 'options' => array( 'query' => Currency::getCurrencies(), 'id' => 'id_currency', 'name' => 'name' ) ); } else { // else hide input $this->fields_form['input'][] = array( 'type' => 'hidden', 'name' => 'management_type' ); $this->fields_form['input'][] = array( 'type' => 'hidden', 'name' => 'id_currency' ); } $this->fields_form['submit'] = array( 'title' => $this->l('Save'), ); $address = null; // loads current address for this warehouse - if possible if ($obj->id_address > 0) { $address = new Address($obj->id_address); } // loads current shops associated with this warehouse $shops = $obj->getShops(); $ids_shop = array(); foreach ($shops as $shop) { $ids_shop[] = $shop['id_shop']; } // loads current carriers associated with this warehouse $carriers = $obj->getCarriers(true); // if an address is available : force specific fields values if ($address != null) { $this->fields_value = array( 'id_address' => $address->id, 'phone' => $address->phone, 'address' => $address->address1, 'address2' => $address->address2, 'postcode' => $address->postcode, 'city' => $address->city, 'id_country' => $address->id_country, 'id_state' => $address->id_state, ); } else { // loads default country $this->fields_value = array( 'id_address' => 0, 'id_country' => Configuration::get('PS_COUNTRY_DEFAULT') ); } // loads shops and carriers $this->fields_value['ids_shops[]'] = $ids_shop; $this->fields_value['ids_carriers'] = $carriers; if (!Validate::isLoadedObject($obj)) { $this->fields_value['id_currency'] = (int)Configuration::get('PS_CURRENCY_DEFAULT'); } return parent::renderForm(); } /** * @see AdminController::renderView() */ public function renderView() { // gets necessary objects $id_warehouse = (int)Tools::getValue('id_warehouse'); $warehouse = new Warehouse($id_warehouse); $employee = new Employee($warehouse->id_employee); $currency = new Currency($warehouse->id_currency); $address = new Address($warehouse->id_address); $shops = $warehouse->getShops(); $this->toolbar_title = $warehouse->name; // checks objects if (!Validate::isLoadedObject($warehouse) || !Validate::isLoadedObject($employee) || !Validate::isLoadedObject($currency) || !Validate::isLoadedObject($address)) { return parent::renderView(); } // assigns to our view $this->tpl_view_vars = array( 'warehouse' => $warehouse, 'employee' => $employee, 'currency' => $currency, 'address' => $address, 'shops' => $shops, 'warehouse_num_products' => $warehouse->getNumberOfProducts(), 'warehouse_value' => Tools::displayPrice(Tools::ps_round($warehouse->getStockValue(), 2), $currency), 'warehouse_quantities' => $warehouse->getQuantitiesofProducts(), ); return parent::renderView(); } /** * Called once $object is set. * Used to process the associations with address/shops/carriers * @see AdminController::afterAdd() * * @param Warehouse $object * * @return bool */ protected function afterAdd($object) { // handles address association $address = new Address($object->id_address); if (Validate::isLoadedObject($address)) { $address->id_warehouse = (int)$object->id; $address->save(); } // handles carriers associations $ids_carriers_selected = Tools::getValue('ids_carriers_selected'); if (Tools::isSubmit('ids_carriers_selected') && !empty($ids_carriers_selected)) { $object->setCarriers($ids_carriers_selected); } else { $object->setCarriers(Tools::getValue('ids_carriers_available')); } return true; } /** * 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); // foreach item in the list to render $nb_items = count($this->_list); for ($i = 0; $i < $nb_items; ++$i) { // depending on the management type, translates the management type $item = &$this->_list[$i]; switch ($item['management_type']) {// management type can be either WA/FIFO/LIFO case 'WA': $item['management_type'] = $this->l('WA: Weighted Average'); break; case 'FIFO': $item['management_type'] = $this->l('FIFO: First In, First Out'); break; case 'LIFO': $item['management_type'] = $this->l('LIFO: Last In, First Out'); break; } } } /** * @return bool */ public function initContent() { if ($this->isAdvancedStockManagementActive()) { return parent::initContent(); } return false; } /** * @return bool */ public function initProcess() { if ($this->isAdvancedStockManagementActive()) { return parent::initProcess(); } return false; } /** * @return bool */ protected function isAdvancedStockManagementActive() { if (Configuration::get('PS_ADVANCED_STOCK_MANAGEMENT')) { return true; } $this->warnings[md5('PS_ADVANCED_STOCK_MANAGEMENT')] = $this->l('You need to activate advanced stock management before using this feature.'); return false; } /** * @see AdminController::processAdd(); */ public function processAdd() { if (Tools::isSubmit('submitAdd'.$this->table)) { if (!($obj = $this->loadObject(true))) { return; } $this->updateAddress(); // hack for enable the possibility to update a warehouse without recreate new id $this->deleted = false; return parent::processAdd(); } } protected function updateAddress() { /** @var AddressCore $address */ $address = new Address(); if (Tools::isSubmit('id_address') && (int)Tools::getValue('id_address') > 0) { $address = new Address((int)Tools::getValue('id_address')); } $address->alias = Tools::getValue('reference', null); $address->lastname = 'warehouse'; // skip problem with numeric characters $address->firstname = 'warehouse'; // in warehouse name $address->address1 = Tools::getValue('address', null); $address->address2 = Tools::getValue('address2', null); $address->postcode = Tools::getValue('postcode', null); $address->phone = Tools::getValue('phone', null); $address->id_country = Tools::getValue('id_country', null); $address->id_state = Tools::getValue('id_state', null); $address->city = Tools::getValue('city', null); if ( !($country = new Country($address->id_country, Configuration::get('PS_LANG_DEFAULT'))) || !Validate::isLoadedObject($country) ) { $this->errors[] = Tools::displayError('Country is invalid'); } $contains_state = isset($country) && is_object($country) ? (int)$country->contains_states: 0; $id_state = isset($address) && is_object($address) ? (int)$address->id_state: 0; if ($contains_state && !$id_state) { $this->errors[] = Tools::displayError('This country requires you to choose a State.'); } // validates the address $validation = $address->validateController(); // checks address validity if (count($validation) > 0) { // if not valid foreach ($validation as $item) { $this->errors[] = $item; } $this->errors[] = Tools::displayError( 'The address is not correct. Please make sure all of the required fields are completed.' ); } else { // valid if (Tools::isSubmit('id_address') && Tools::getValue('id_address') > 0) { $address->update(); } else { $address->save(); $_POST['id_address'] = $address->id; } } } /** * When submitting a warehouse deletion request, * make an attempt to load a warehouse instance from an identifier, * ensure the warehouse to be deleted, * - does not contain any products, * - nor it has some pending supply orders * before actual deletion * * @return bool|mixed */ public function processDelete() { if (!Tools::isSubmit('delete'.$this->table)) { return false; } /** @var Warehouse $warehouse */ $warehouse = $this->loadObject(true); if ($this->shouldForbidWarehouseDeletion($warehouse)) { return false; } return $this->deleteWarehouse($warehouse); } /** * @param $warehouse * @return bool */ protected function shouldForbidWarehouseDeletion($warehouse) { if (!$warehouse) { return true; } if ($warehouse->getQuantitiesOfProducts() > 0) { $this->errors[] = $this->l('It is not possible to delete a warehouse when there are products in it.'); return true; } if (SupplyOrder::warehouseHasPendingOrders($warehouse->id)) { $this->errors[] = $this->l('It is not possible to delete a Warehouse if it has pending supply orders.'); return true; } return false; } /** * @param Warehouse $warehouse * @return mixed */ protected function deleteWarehouse(Warehouse $warehouse) { $address = new Address($warehouse->id_address); $this->markAddressAsDeleted($address); /** @var WarehouseCore $warehouse */ $warehouse->setCarriers(array()); $warehouse->resetProductsLocations(); return parent::processDelete(); } /** * @param Address $address */ protected function markAddressAsDeleted(Address $address) { /** @var AddressCore $address */ $address->deleted = 1; $address->save(); } /** * @see AdminController::processUpdate(); */ public function processUpdate() { /** @var WarehouseCore $warehouse */ if (!($warehouse = $this->loadObject(true))) { return false; } $this->updateAddress(); // handles carriers associations $ids_carriers_selected = Tools::getValue('ids_carriers_selected'); if (Tools::isSubmit('ids_carriers_selected') && !empty($ids_carriers_selected)) { $warehouse->setCarriers($ids_carriers_selected); } else { $warehouse->setCarriers(Tools::getValue('ids_carriers_available')); } return parent::processUpdate(); } protected function updateAssoShop($id_object) { parent::updateAssoShop($id_object); if (!($obj = $this->loadObject(true))) { return; } /** @var Warehouse $obj */ $obj->resetStockAvailable(); } }