* @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0) */ /** * @property Supplier $object */ class AdminSuppliersControllerCore extends AdminController { /** @var bool */ public $bootstrap = true; public function __construct() { $this->table = 'supplier'; $this->className = 'Supplier'; parent::__construct(); $this->addRowAction('view'); $this->addRowAction('edit'); $this->addRowAction('delete'); $this->allow_export = true; $this->_defaultOrderBy = 'name'; $this->_defaultOrderWay = 'ASC'; $this->bulk_actions = [ 'delete' => [ 'text' => $this->trans('Delete selected', [], 'Admin.Actions'), 'icon' => 'icon-trash', 'confirm' => $this->trans('Delete selected items?', [], 'Admin.Notifications.Warning'), ], ]; $this->_select = 'COUNT(DISTINCT ps.`id_product`) AS products'; $this->_join = 'LEFT JOIN `' . _DB_PREFIX_ . 'product_supplier` ps ON (a.`id_supplier` = ps.`id_supplier`)'; $this->_group = 'GROUP BY a.`id_supplier`'; $this->fieldImageSettings = ['name' => 'logo', 'dir' => 'su']; $this->fields_list = [ 'id_supplier' => ['title' => $this->trans('ID', [], 'Admin.Global'), 'align' => 'center', 'class' => 'fixed-width-xs'], 'logo' => ['title' => $this->trans('Logo', [], 'Admin.Global'), 'align' => 'center', 'image' => 'su', 'orderby' => false, 'search' => false], 'name' => ['title' => $this->trans('Name', [], 'Admin.Global')], 'products' => ['title' => $this->trans('Number of products', [], 'Admin.Catalog.Feature'), 'align' => 'right', 'filter_type' => 'int', 'tmpTableFilter' => true], 'active' => ['title' => $this->trans('Enabled', [], 'Admin.Global'), 'align' => 'center', 'active' => 'status', 'type' => 'bool', 'orderby' => false, 'class' => 'fixed-width-xs'], ]; } public function setMedia($isNewTheme = false) { parent::setMedia($isNewTheme); $this->addJqueryUi('ui.widget'); $this->addJqueryPlugin('tagify'); } public function initPageHeaderToolbar() { if (empty($this->display)) { $this->page_header_toolbar_btn['new_supplier'] = [ 'href' => self::$currentIndex . '&addsupplier&token=' . $this->token, 'desc' => $this->trans('Add new supplier', [], 'Admin.Catalog.Feature'), 'icon' => 'process-icon-new', ]; } parent::initPageHeaderToolbar(); } /** * @return string|void * * @throws PrestaShopDatabaseException * @throws SmartyException */ public function renderForm() { // loads current warehouse if (!($obj = $this->loadObject(true))) { return; } $image = _PS_SUPP_IMG_DIR_ . $obj->id . '.jpg'; $image_url = ImageManager::thumbnail( $image, $this->table . '_' . (int) $obj->id . '.' . $this->imageType, 350, $this->imageType, true, true ); $image_size = file_exists($image) ? filesize($image) / 1000 : false; $tmp_addr = new Address(); $res = $tmp_addr->getFieldsRequiredDatabase(); $required_fields = []; foreach ($res as $row) { $required_fields[(int) $row['id_required_field']] = $row['field_name']; } $this->fields_form = [ 'legend' => [ 'title' => $this->trans('Suppliers', [], 'Admin.Global'), 'icon' => 'icon-truck', ], 'input' => [ [ 'type' => 'hidden', 'name' => 'id_address', ], [ 'type' => 'text', 'label' => $this->trans('Name', [], 'Admin.Global'), 'name' => 'name', 'required' => true, 'col' => 4, 'hint' => $this->trans('Invalid characters:', [], 'Admin.Notifications.Info') . ' <>;=#{}', ], ( in_array('company', $required_fields) ? [ 'type' => 'text', 'label' => $this->trans('Company', [], 'Admin.Global'), 'name' => 'company', 'display' => in_array('company', $required_fields), 'required' => in_array('company', $required_fields), 'maxlength' => 16, 'col' => 4, 'hint' => $this->trans('Company name for this supplier', [], 'Admin.Catalog.Help'), ] : null ), [ 'type' => 'textarea', 'label' => $this->trans('Description', [], 'Admin.Global'), 'name' => 'description', 'lang' => true, 'hint' => [ $this->trans('Invalid characters:', [], 'Admin.Notifications.Info') . ' <>;=#{}', $this->trans('Will appear in the list of suppliers.', [], 'Admin.Catalog.Help'), ], 'autoload_rte' => 'rte', //Enable TinyMCE editor for short description ], [ 'type' => 'text', 'label' => $this->trans('Phone', [], 'Admin.Global'), 'name' => 'phone', 'required' => in_array('phone', $required_fields), 'maxlength' => 16, 'col' => 4, 'hint' => $this->trans('Phone number for this supplier', [], 'Admin.Catalog.Help'), ], [ 'type' => 'text', 'label' => $this->trans('Mobile phone', [], 'Admin.Global'), 'name' => 'phone_mobile', 'required' => in_array('phone_mobile', $required_fields), 'maxlength' => 16, 'col' => 4, 'hint' => $this->trans('Mobile phone number for this supplier.', [], 'Admin.Catalog.Help'), ], [ 'type' => 'text', 'label' => $this->trans('Address', [], 'Admin.Global'), 'name' => 'address', 'maxlength' => 128, 'col' => 6, 'required' => true, ], [ 'type' => 'text', 'label' => $this->trans('Address (2)', [], 'Admin.Global'), 'name' => 'address2', 'required' => in_array('address2', $required_fields), 'col' => 6, 'maxlength' => 128, ], [ 'type' => 'text', 'label' => $this->trans('Zip/Postal code', [], 'Admin.Global'), 'name' => 'postcode', 'required' => in_array('postcode', $required_fields), 'maxlength' => 12, 'col' => 2, ], [ 'type' => 'text', 'label' => $this->trans('City', [], 'Admin.Global'), 'name' => 'city', 'maxlength' => 32, 'col' => 4, 'required' => true, ], [ 'type' => 'select', 'label' => $this->trans('Country', [], 'Admin.Global'), 'name' => 'id_country', 'required' => true, 'col' => 4, 'default_value' => (int) $this->context->country->id, 'options' => [ 'query' => Country::getCountries($this->context->language->id, false), 'id' => 'id_country', 'name' => 'name', ], ], [ 'type' => 'select', 'label' => $this->trans('State', [], 'Admin.Global'), 'name' => 'id_state', 'col' => 4, 'options' => [ 'id' => 'id_state', 'query' => [], 'name' => 'name', ], ], [ 'type' => 'text', 'label' => $this->trans('DNI', [], 'Admin.Global'), 'name' => 'dni', 'maxlength' => 16, 'col' => 4, 'required' => true, // Only required in case of specifics countries ], [ 'type' => 'file', 'label' => $this->trans('Logo', [], 'Admin.Global'), 'name' => 'logo', 'display_image' => true, 'image' => $image_url ? $image_url : false, 'size' => $image_size, 'hint' => $this->trans('Upload a supplier logo from your computer.', [], 'Admin.Catalog.Help'), ], [ 'type' => 'text', 'label' => $this->trans('Meta title', [], 'Admin.Global'), 'name' => 'meta_title', 'lang' => true, 'col' => 4, 'hint' => $this->trans('Invalid characters:', [], 'Admin.Notifications.Info') . ' <>;=#{}', ], [ 'type' => 'text', 'label' => $this->trans('Meta description', [], 'Admin.Global'), 'name' => 'meta_description', 'lang' => true, 'col' => 6, 'hint' => $this->trans('Invalid characters:', [], 'Admin.Notifications.Info') . ' <>;=#{}', ], [ 'type' => 'tags', 'label' => $this->trans('Meta keywords', [], 'Admin.Global'), 'name' => 'meta_keywords', 'lang' => true, 'col' => 6, 'hint' => [ $this->trans('To add tags, click in the field, write something, and then press the "Enter" key.', [], 'Admin.Shopparameters.Help'), $this->trans('Invalid characters:', [], 'Admin.Notifications.Info') . ' <>;=#{}', ], ], [ 'type' => 'switch', 'label' => $this->trans('Enable', [], 'Admin.Actions'), 'name' => 'active', 'required' => false, 'class' => 't', 'is_bool' => true, 'values' => [ [ 'id' => 'active_on', 'value' => 1, 'label' => $this->trans('Yes', [], 'Admin.Global'), ], [ 'id' => 'active_off', 'value' => 0, 'label' => $this->trans('No', [], 'Admin.Global'), ], ], ], ], 'submit' => [ 'title' => $this->trans('Save', [], 'Admin.Actions'), ], ]; // loads current address for this supplier - if possible $address = null; if (isset($obj->id)) { $id_address = Address::getAddressIdBySupplierId($obj->id); if ($id_address > 0) { $address = new Address((int) $id_address); } } // force specific fields values (address) if ($address != null) { $this->fields_value = [ 'id_address' => $address->id, 'phone' => $address->phone, 'phone_mobile' => $address->phone_mobile, 'address' => $address->address1, 'address2' => $address->address2, 'postcode' => $address->postcode, 'city' => $address->city, 'id_country' => $address->id_country, 'id_state' => $address->id_state, 'dni' => $address->dni, ]; } else { $this->fields_value = [ 'id_address' => 0, 'id_country' => Configuration::get('PS_COUNTRY_DEFAULT'), ]; } if (Shop::isFeatureActive()) { $this->fields_form['input'][] = [ 'type' => 'shop', 'label' => $this->trans('Store association', [], 'Admin.Global'), 'name' => 'checkBoxShopAsso', ]; } return parent::renderForm(); } /** * AdminController::initToolbar() override. * * @see AdminController::initToolbar() */ public function initToolbar() { parent::initToolbar(); if (empty($this->display) && $this->can_import) { $this->toolbar_btn['import'] = [ 'href' => $this->context->link->getAdminLink('AdminImport', true) . '&import_type=suppliers', 'desc' => $this->trans('Import', [], 'Admin.Actions'), ]; } } public function renderView() { $this->toolbar_title = $this->object->name; $products = $this->object->getProductsLite($this->context->language->id); $total_product = count($products); for ($i = 0; $i < $total_product; ++$i) { $products[$i] = new Product($products[$i]['id_product'], false, $this->context->language->id); $products[$i]->loadStockData(); // Build attributes combinations $combinations = $products[$i]->getAttributeCombinations($this->context->language->id); foreach ($combinations as $combination) { $comb_infos = Supplier::getProductInformationsBySupplier( $this->object->id, $products[$i]->id, $combination['id_product_attribute'] ); $comb_array[$combination['id_product_attribute']]['product_supplier_reference'] = $comb_infos['product_supplier_reference']; $comb_array[$combination['id_product_attribute']]['product_supplier_price_te'] = $this->context->getCurrentLocale()->formatPrice($comb_infos['product_supplier_price_te'], Currency::getIsoCodeById((int) $comb_infos['id_currency'])); $comb_array[$combination['id_product_attribute']]['reference'] = $combination['reference']; $comb_array[$combination['id_product_attribute']]['ean13'] = $combination['ean13']; $comb_array[$combination['id_product_attribute']]['upc'] = $combination['upc']; $comb_array[$combination['id_product_attribute']]['mpn'] = $combination['mpn']; $comb_array[$combination['id_product_attribute']]['quantity'] = $combination['quantity']; $comb_array[$combination['id_product_attribute']]['attributes'][] = [ $combination['group_name'], $combination['attribute_name'], $combination['id_attribute'], ]; } if (isset($comb_array)) { foreach ($comb_array as $key => $product_attribute) { $list = ''; foreach ($product_attribute['attributes'] as $attribute) { $list .= $attribute[0] . ' - ' . $attribute[1] . ', '; } $comb_array[$key]['attributes'] = rtrim($list, ', '); } $products[$i]->combination = $comb_array; unset($comb_array); } else { $product_infos = Supplier::getProductInformationsBySupplier( $this->object->id, $products[$i]->id, 0 ); $products[$i]->product_supplier_reference = $product_infos['product_supplier_reference']; $currencyId = $product_infos['id_currency'] ?: Currency::getDefaultCurrency()->id; $products[$i]->product_supplier_price_te = $this->context->getCurrentLocale()->formatPrice($product_infos['product_supplier_price_te'], Currency::getIsoCodeById((int) $currencyId)); } } $this->tpl_view_vars = [ 'supplier' => $this->object, 'products' => $products, 'stock_management' => Configuration::get('PS_STOCK_MANAGEMENT'), 'shopContext' => Shop::getContext(), ]; return parent::renderView(); } protected function afterImageUpload() { $return = true; // Should we generate high DPI images? $generate_hight_dpi_images = (bool) Configuration::get('PS_HIGHT_DPI'); /* Generate image with differents size */ if (($id_supplier = (int) Tools::getValue('id_supplier')) && count($_FILES) && file_exists(_PS_SUPP_IMG_DIR_ . $id_supplier . '.jpg')) { $images_types = ImageType::getImagesTypes('suppliers'); foreach ($images_types as $image_type) { $file = _PS_SUPP_IMG_DIR_ . $id_supplier . '.jpg'; if (!ImageManager::resize($file, _PS_SUPP_IMG_DIR_ . $id_supplier . '-' . stripslashes($image_type['name']) . '.jpg', (int) $image_type['width'], (int) $image_type['height'])) { $return = false; } if ($generate_hight_dpi_images) { if (!ImageManager::resize($file, _PS_SUPP_IMG_DIR_ . $id_supplier . '-' . stripslashes($image_type['name']) . '2x.jpg', (int) $image_type['width'] * 2, (int) $image_type['height'] * 2)) { $return = false; } } } $current_logo_file = _PS_TMP_IMG_DIR_ . 'supplier_mini_' . $id_supplier . '_' . $this->context->shop->id . '.jpg'; if (file_exists($current_logo_file)) { unlink($current_logo_file); } } return $return; } /** * AdminController::postProcess() override. * * @see AdminController::postProcess() */ public function postProcess() { // checks access if (Tools::isSubmit('submitAdd' . $this->table) && !($this->access('add'))) { $this->errors[] = $this->trans('You do not have permission to add suppliers.', [], 'Admin.Catalog.Notification'); return parent::postProcess(); } if (Tools::isSubmit('submitAdd' . $this->table)) { if (Tools::isSubmit('id_supplier') && !($obj = $this->loadObject(true))) { return; } // updates/creates address if it does not exist if (Tools::isSubmit('id_address') && (int) Tools::getValue('id_address') > 0) { $address = new Address((int) Tools::getValue('id_address')); } // updates address else { $address = new Address(); } // creates address $address->alias = Tools::getValue('name', null); $address->lastname = 'supplier'; // skip problem with numeric characters in supplier name $address->firstname = 'supplier'; // skip problem with numeric characters in supplier 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->phone_mobile = Tools::getValue('phone_mobile', null); $address->id_country = Tools::getValue('id_country', null); $address->id_state = Tools::getValue('id_state', null); $address->city = Tools::getValue('city', null); $address->dni = Tools::getValue('dni', null); $validation = $address->validateController(); /* * Make sure dni is checked without raising an exception. * This field is mandatory for some countries. */ if ($address->validateField('dni', $address->dni) !== true) { $validation['dni'] = $this->trans( '%s is invalid.', [ 'dni', ], 'Admin.Notifications.Error' ); } // checks address validity if (count($validation) > 0) { foreach ($validation as $item) { $this->errors[] = $item; } $this->errors[] = $this->trans('The address is not correct. Please make sure all of the required fields are completed.', [], 'Admin.Catalog.Notification'); } else { if (Tools::isSubmit('id_address') && Tools::getValue('id_address') > 0) { $address->update(); } else { $address->save(); $_POST['id_address'] = $address->id; } } return parent::postProcess(); } elseif (Tools::isSubmit('delete' . $this->table)) { if (!($obj = $this->loadObject(true))) { return; } elseif (SupplyOrder::supplierHasPendingOrders($obj->id)) { $this->errors[] = $this->trans('It is not possible to delete a supplier if there are pending supplier orders.', [], 'Admin.Catalog.Notification'); } else { //delete all product_supplier linked to this supplier Db::getInstance()->execute('DELETE FROM `' . _DB_PREFIX_ . 'product_supplier` WHERE `id_supplier`=' . (int) $obj->id); $id_address = Address::getAddressIdBySupplierId($obj->id); $address = new Address($id_address); if (Validate::isLoadedObject($address)) { $address->deleted = true; $address->save(); } return parent::postProcess(); } } else { return parent::postProcess(); } } /** * @see AdminController::afterAdd() * * @param Supplier $object * * @return bool */ protected function afterAdd($object) { $id_address = (int) $_POST['id_address']; $address = new Address($id_address); if (Validate::isLoadedObject($address)) { $address->id_supplier = $object->id; $address->save(); } return true; } /** * @see AdminController::afterUpdate() * * @param Supplier $object * * @return bool */ protected function afterUpdate($object) { $id_address = (int) $_POST['id_address']; $address = new Address($id_address); if (Validate::isLoadedObject($address)) { if ($address->id_supplier != $object->id) { $address->id_supplier = $object->id; $address->save(); } } return true; } }