Files
taome/modules/ps_crossselling/ps_crossselling.php
2020-10-07 10:37:15 +02:00

348 lines
14 KiB
PHP

<?php
/*
* 2007-2016 PrestaShop
*
* NOTICE OF LICENSE
*
* This source file is subject to the Academic Free License (AFL 3.0)
* that is bundled with this package in the file LICENSE.txt.
* It is also available through the world-wide-web at this URL:
* http://opensource.org/licenses/afl-3.0.php
* If you did not receive a copy of the license and are unable to
* obtain it through the world-wide-web, please send an email
* to license@prestashop.com so we can send you a copy immediately.
*
* DISCLAIMER
*
* Do not edit or add to this file if you wish to upgrade PrestaShop to newer
* versions in the future. If you wish to customize PrestaShop for your
* needs please refer to http://www.prestashop.com for more information.
*
* @author PrestaShop SA <contact@prestashop.com>
* @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;
}
}