* @copyright 2007-2016 PrestaShop SA * @license http://opensource.org/licenses/afl-3.0.php Academic Free License (AFL 3.0) * International Registered Trademark & Property of PrestaShop SA */ use PrestaShop\PrestaShop\Core\Module\WidgetInterface; use PrestaShop\PrestaShop\Adapter\Image\ImageRetriever; use PrestaShop\PrestaShop\Adapter\Product\PriceFormatter; use PrestaShop\PrestaShop\Core\Product\ProductListingPresenter; use PrestaShop\PrestaShop\Adapter\Product\ProductColorsRetriever; if (!defined('_PS_VERSION_')) { exit; } class Ps_Crossselling extends Module implements WidgetInterface { private $templateFile; public function __construct() { $this->name = 'ps_crossselling'; $this->author = 'PrestaShop'; $this->version = '2.0.0'; $this->need_instance = 0; $this->ps_versions_compliancy = array( 'min' => '1.7.2.0', 'max' => _PS_VERSION_, ); $this->bootstrap = true; parent::__construct(); $this->displayName = $this->trans('Cross-selling', array(), 'Modules.Crossselling.Admin'); $this->description = $this->trans('Adds a "Customers who bought this product also bought..." section to every product page.', array(), 'Modules.Crossselling.Admin'); $this->templateFile = 'module:ps_crossselling/views/templates/hook/ps_crossselling.tpl'; } public function install() { $this->_clearCache('*'); return parent::install() && Configuration::updateValue('CROSSSELLING_DISPLAY_PRICE', 1) && Configuration::updateValue('CROSSSELLING_NBR', 8) && $this->registerHook('displayFooterProduct') && $this->registerHook('actionOrderStatusPostUpdate'); } public function uninstall() { $this->_clearCache('*'); return parent::uninstall() && Configuration::deleteByName('CROSSSELLING_DISPLAY_PRICE') && Configuration::deleteByName('CROSSSELLING_NBR'); } public function getContent() { $html = ''; if (Tools::isSubmit('submitCross')) { if (0 != Tools::getValue('displayPrice') && 1 != Tools::getValue('CROSSSELLING_DISPLAY_PRICE')) { $html .= $this->displayError('Invalid displayPrice'); } elseif (!($product_nbr = Tools::getValue('CROSSSELLING_NBR')) || empty($product_nbr)) { $html .= $this->displayError($this->trans('You must fill in the "Number of displayed products" field.', array(), 'Modules.Crossselling.Admin')); } elseif (0 === (int)$product_nbr) { $html .= $this->displayError($this->trans('Invalid number.', array(), 'Modules.Crossselling.Admin')); } else { Configuration::updateValue('CROSSSELLING_DISPLAY_PRICE', (int)Tools::getValue('CROSSSELLING_DISPLAY_PRICE')); Configuration::updateValue('CROSSSELLING_NBR', (int)Tools::getValue('CROSSSELLING_NBR')); $this->_clearCache('*'); $html .= $this->displayConfirmation($this->trans('The settings have been updated.', array(), 'Admin.Notifications.Success')); } } return $html.$this->renderForm(); } public function hookActionOrderStatusPostUpdate($params) { $this->_clearCache('*'); } protected function _clearCache($template, $cacheId = null, $compileId = null) { parent::_clearCache($this->templateFile); } public function renderForm() { $fields_form = array( 'form' => array( 'legend' => array( 'title' => $this->trans('Settings', array(), 'Admin.Global'), 'icon' => 'icon-cogs', ), 'input' => array( array( 'type' => 'switch', 'label' => $this->trans('Display price on products', array(), 'Modules.Crossselling.Admin'), 'name' => 'CROSSSELLING_DISPLAY_PRICE', 'desc' => $this->trans('Show the price on the products in the block.', array(), 'Modules.Crossselling.Admin'), 'values' => array( array( 'id' => 'active_on', 'value' => 1, 'label' => $this->trans('Enabled', array(), 'Admin.Global'), ), array( 'id' => 'active_off', 'value' => 0, 'label' => $this->trans('Disabled', array(), 'Admin.Global'), ), ), ), array( 'type' => 'text', 'label' => $this->trans('Number of displayed products', array(), 'Modules.Crossselling.Admin'), 'name' => 'CROSSSELLING_NBR', 'class' => 'fixed-width-xs', 'desc' => $this->trans('Set the number of products displayed in this block.', array(), 'Modules.Crossselling.Admin'), ), ), 'submit' => array( 'title' => $this->trans('Save', array(), 'Admin.Actions'), ), ), ); $lang = new Language((int)Configuration::get('PS_LANG_DEFAULT')); $helper = new HelperForm(); $helper->show_toolbar = false; $helper->table = $this->table; $helper->default_form_language = $lang->id; $helper->allow_employee_form_lang = Configuration::get('PS_BO_ALLOW_EMPLOYEE_FORM_LANG') ? Configuration::get('PS_BO_ALLOW_EMPLOYEE_FORM_LANG') : 0; $helper->identifier = $this->identifier; $helper->submit_action = 'submitCross'; $helper->currentIndex = $this->context->link->getAdminLink('AdminModules', false) . '&configure=' . $this->name . '&tab_module=' . $this->tab . '&module_name=' . $this->name; $helper->token = Tools::getAdminTokenLite('AdminModules'); $helper->tpl_vars = array( 'fields_value' => $this->getConfigFieldsValues(), 'languages' => $this->context->controller->getLanguages(), 'id_language' => $this->context->language->id ); return $helper->generateForm(array($fields_form)); } public function getConfigFieldsValues() { return array( 'CROSSSELLING_NBR' => Tools::getValue('CROSSSELLING_NBR', Configuration::get('CROSSSELLING_NBR')), 'CROSSSELLING_DISPLAY_PRICE' => Tools::getValue('CROSSSELLING_DISPLAY_PRICE', Configuration::get('CROSSSELLING_DISPLAY_PRICE')), ); } public function getCacheIdKey($productIds) { return parent::getCacheId('ps_crossselling|' . implode('|', $productIds)); } private function getProductIds($hookName, array $configuration) { if ('displayShoppingCart' === $hookName) { $productIds = array_map(function ($elem) { return $elem['id_product']; }, $configuration['cart']->getProducts()); } else { $productIds = array($configuration['product']['id_product']); } return array_unique($productIds); } public function getWidgetVariables($hookName, array $configuration) { $productIds = $this->getProductIds($hookName, $configuration); if (!empty($productIds)) { $products = $this->getOrderProducts($productIds); if (!empty($products)) { return array( 'products' => $products, ); } } return false; } public function renderWidget($hookName, array $configuration) { $productIds = $this->getProductIds($hookName, $configuration); if (empty($productIds)) { return; } if (!$this->isCached($this->templateFile, $this->getCacheIdKey($productIds))) { $variables = $this->getWidgetVariables($hookName, $configuration); if (empty($variables)) { return false; } $this->smarty->assign($variables); } return $this->fetch($this->templateFile, $this->getCacheIdKey($productIds)); } protected function getOrderProducts(array $productIds = array()) { $q_orders = 'SELECT o.id_order FROM '._DB_PREFIX_.'orders o LEFT JOIN '._DB_PREFIX_.'order_detail od ON (od.id_order = o.id_order) WHERE o.valid = 1 AND od.product_id IN ('.implode(',', $productIds).')'; $orders = Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS($q_orders); if (0 < count($orders)) { $list = ''; foreach ($orders as $order) { $list .= (int)$order['id_order'].','; } $list = rtrim($list, ','); $list_product_ids = join(',', $productIds); if (Group::isFeatureActive()) { $sql_groups_join = ' LEFT JOIN `'._DB_PREFIX_.'category_product` cp ON (cp.`id_category` = product_shop.id_category_default AND cp.id_product = product_shop.id_product) LEFT JOIN `'._DB_PREFIX_.'category_group` cg ON (cp.`id_category` = cg.`id_category`)'; $groups = FrontController::getCurrentCustomerGroups(); $sql_groups_where = 'AND cg.`id_group` '. (count($groups) ? 'IN ('.implode(',', $groups) . ')' : '=' . (int)Group::getCurrent()->id); } $order_products = Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS(' SELECT DISTINCT od.product_id FROM '._DB_PREFIX_.'order_detail od LEFT JOIN '._DB_PREFIX_.'product p ON (p.id_product = od.product_id) '.Shop::addSqlAssociation('product', 'p'). (Combination::isFeatureActive() ? 'LEFT JOIN `' . _DB_PREFIX_.'product_attribute` pa ON (p.`id_product` = pa.`id_product`) ' . Shop::addSqlAssociation( 'product_attribute', 'pa', false, 'product_attribute_shop.`default_on` = 1' ).' ' . Product::sqlStock( 'p', 'product_attribute_shop', false, $this->context->shop ) : Product::sqlStock( 'p', 'product', false, $this->context->shop )).' LEFT JOIN '._DB_PREFIX_.'product_lang pl ON (pl.id_product = od.product_id' . Shop::addSqlRestrictionOnLang('pl').') LEFT JOIN '._DB_PREFIX_.'category_lang cl ON (cl.id_category = product_shop.id_category_default' .Shop::addSqlRestrictionOnLang('cl').') LEFT JOIN '._DB_PREFIX_.'image i ON (i.id_product = od.product_id) '.(Group::isFeatureActive() ? $sql_groups_join : '').' WHERE od.id_order IN ('.$list.') AND pl.id_lang = '.(int)$this->context->language->id.' AND cl.id_lang = '.(int)$this->context->language->id.' AND od.product_id NOT IN ('.$list_product_ids.') AND i.cover = 1 AND product_shop.active = 1 '.(Group::isFeatureActive() ? $sql_groups_where : '').' ORDER BY RAND() LIMIT '.(int)Configuration::get('CROSSSELLING_NBR') ); } if (!empty($order_products)) { $showPrice = (bool) Configuration::get('CROSSSELLING_DISPLAY_PRICE'); $assembler = new ProductAssembler($this->context); $presenterFactory = new ProductPresenterFactory($this->context); $presentationSettings = $presenterFactory->getPresentationSettings(); $presenter = new ProductListingPresenter( new ImageRetriever( $this->context->link ), $this->context->link, new PriceFormatter(), new ProductColorsRetriever(), $this->context->getTranslator() ); $productsForTemplate = array(); $presentationSettings->showPrices = $showPrice; if (is_array($order_products)) { foreach ($order_products as $productId) { $productsForTemplate[] = $presenter->present( $presentationSettings, $assembler->assembleProduct(array('id_product' => $productId['product_id'])), $this->context->language ); } } return $productsForTemplate; } return false; } }