name = 'statsproduct'; $this->tab = 'Stats'; $this->version = 1.0; parent::__construct(); $this->displayName = $this->l('Product details'); $this->description = $this->l('Get detailed statistics for each product'); } public function install() { return (parent::install() AND $this->registerHook('AdminStatsModules')); } public function getTotalBought($id_product) { $dateBetween = ModuleGraph::getDateBetween(); $result = Db::getInstance()->getRow(' SELECT SUM(od.`product_quantity`) AS total FROM `'._DB_PREFIX_.'order_detail` od LEFT JOIN `'._DB_PREFIX_.'orders` o ON o.`id_order` = od.`id_order` WHERE od.`product_id` = '.intval($id_product).' AND o.valid = 1 AND o.`date_add` BETWEEN '.$dateBetween.''); return isset($result['total']) ? $result['total'] : 0; } public function getTotalViewed($id_product) { $dateBetween = ModuleGraph::getDateBetween(); $result = Db::getInstance()->getRow(' SELECT SUM(pv.`counter`) AS total FROM `'._DB_PREFIX_.'page_viewed` pv LEFT JOIN `'._DB_PREFIX_.'date_range` dr ON pv.`id_date_range` = dr.`id_date_range` LEFT JOIN `'._DB_PREFIX_.'page` p ON pv.`id_page` = p.`id_page` LEFT JOIN `'._DB_PREFIX_.'page_type` pt ON pt.`id_page_type` = p.`id_page_type` WHERE pt.`name` = \'product.php\' AND p.`id_object` = '.intval($id_product).' AND dr.`time_start` BETWEEN '.$dateBetween.' AND dr.`time_end` BETWEEN '.$dateBetween.''); return isset($result['total']) ? $result['total'] : 0; } private function getProducts($id_lang) { return Db::getInstance()->ExecuteS(' SELECT p.`id_product`, p.reference, pl.`name`, IFNULL((SELECT SUM(pa.quantity) FROM '._DB_PREFIX_.'product_attribute pa WHERE pa.id_product = p.id_product), p.quantity) as quantity FROM `'._DB_PREFIX_.'product` p LEFT JOIN `'._DB_PREFIX_.'product_lang` pl ON p.`id_product` = pl.`id_product` '.(Tools::getValue('id_category') ? 'LEFT JOIN `'._DB_PREFIX_.'category_product` cp ON p.`id_product` = cp.`id_product`' : '').' WHERE pl.`id_lang` = '.intval($id_lang).' '.(Tools::getValue('id_category') ? 'AND cp.id_category = '.intval(Tools::getValue('id_category')) : '').' ORDER BY pl.`name`'); } private function getSales($id_product, $id_lang) { return Db::getInstance()->ExecuteS(' SELECT o.date_add, o.id_order, o.id_customer, od.product_quantity, (od.product_price * od.product_quantity) as total, od.tax_name, od.product_name FROM `'._DB_PREFIX_.'orders` o LEFT JOIN `'._DB_PREFIX_.'order_detail` od ON o.id_order = od.id_order WHERE o.date_add BETWEEN '.$this->getDate().' AND o.valid = 1 AND od.product_id = '.intval($id_product)); } private function getCrossSales($id_product, $id_lang) { return Db::getInstance()->ExecuteS(' SELECT pl.name as pname, pl.id_product, SUM(od.product_quantity) as pqty, AVG(od.product_price) as pprice FROM `'._DB_PREFIX_.'orders` o LEFT JOIN `'._DB_PREFIX_.'order_detail` od ON o.id_order = od.id_order LEFT JOIN `'._DB_PREFIX_.'product_lang` pl ON (pl.id_product = od.product_id AND pl.id_lang = '.intval($id_lang).') WHERE o.id_customer IN ( SELECT o.id_customer FROM `'._DB_PREFIX_.'orders` o LEFT JOIN `'._DB_PREFIX_.'order_detail` od ON o.id_order = od.id_order WHERE o.date_add BETWEEN '.$this->getDate().' AND o.valid = 1 AND od.product_id = '.intval($id_product).' ) AND o.date_add BETWEEN '.$this->getDate().' AND o.valid = 1 AND od.product_id != '.intval($id_product).' GROUP BY od.product_id ORDER BY pqty DESC'); } public function hookAdminStatsModules($params) { global $cookie, $currentIndex; $id_category = intval(Tools::getValue('id_category')); $currency = new Currency(Configuration::get('PS_CURRENCY_DEFAULT')); $this->_html = '
'.$this->displayName.''; if ($id_product = intval(Tools::getValue('id_product'))) { $product = new Product($id_product, false, intval($cookie->id_lang)); $totalBought = $this->getTotalBought($product->id); $totalViewed = $this->getTotalViewed($product->id); $this->_html .= '

'.$product->name.' - '.$this->l('Details').'

'.$this->l('Total bought:').' '.$totalBought.'

'.$this->l('Total viewed:').' '.$totalViewed.'

'.$this->l('Conversion rate:').' '.number_format($totalViewed ? $totalBought / $totalViewed : 0, 2).'

'.ModuleGraph::engine(array('layers' => 2, 'type' => 'line', 'option' => '1-'.$id_product)).'
'; if ($hasAttribute = $product->hasAttributes() AND $totalBought) $this->_html .= '

'.$this->l('Attribute sales distribution').'

'.ModuleGraph::engine(array('type' => 'pie', 'option' => '3-'.$id_product)).'
'; if ($totalBought) { $sales = $this->getSales($id_product, $cookie->id_lang); $this->_html .= '

'.$this->l('Sales').'

'.($hasAttribute ? '' : '').' '; $tokenOrder = Tools::getAdminToken('AdminOrders'.intval(Tab::getIdFromClassName('AdminOrders')).intval($cookie->id_employee)); $tokenCustomer = Tools::getAdminToken('AdminCustomers'.intval(Tab::getIdFromClassName('AdminCustomers')).intval($cookie->id_employee)); foreach ($sales as $sale) $this->_html .= ' '.($hasAttribute ? '' : '').' '; $this->_html .= '
'.$this->l('Date').' '.$this->l('Order').' '.$this->l('Customer').''.$this->l('Attribute').''.$this->l('Qty').' '.$this->l('Price').'
'.Tools::displayDate($sale['date_add'], intval($cookie->id_lang), false).' '.intval($sale['id_order']).' '.intval($sale['id_customer']).''.$sale['product_name'].''.intval($sale['product_quantity']).' '.Tools::displayprice($sale['total'], $currency).'
'; $crossSelling = $this->getCrossSales($id_product, $cookie->id_lang); if (count($crossSelling)) { $this->_html .= '

'.$this->l('Cross Selling').'

'; $tokenProducts = Tools::getAdminToken('AdminCatalog'.intval(Tab::getIdFromClassName('AdminCatalog')).intval($cookie->id_employee)); foreach ($crossSelling as $selling) $this->_html .= ' '; $this->_html .= '
'.$this->l('Product name').' '.$this->l('Quantity sold').' '.$this->l('Average price').'
'.$selling['pname'].' '.intval($selling['pqty']).' '.Tools::displayprice($selling['pprice'], $currency).'
'; } } } else { $categories = Category::getCategories(intval($cookie->id_lang), true, false); $this->_html .= '
'.$this->l('Click on a product to access its statistics.').'

'.$this->l('Products available').'

'; foreach ($this->getProducts($cookie->id_lang) AS $product) $this->_html .= ''; $this->_html .= '
'.$this->l('Ref.').' '.$this->l('Name').' '.$this->l('Stock').'
'.$product['reference'].''.$product['name'].''.$product['quantity'].'
'; } $this->_html .= '

'.$this->l('Guide').'

'.$this->l('Number of purchases compared to number of viewings').'

'.$this->l('After choosing a category and selecting a product available in this category, some graphs appear. Then, you can analyze them.').'

'; return $this->_html; } public function setOption($option, $layers = 1) { list($this->_option, $this->_id_product) = explode('-', $option); $dateBetween = $this->getDate(); switch ($this->_option) { case 1: $this->_titles['main'][0] = $this->l('Popularity'); $this->_titles['main'][1] = $this->l('Sales'); $this->_titles['main'][2] = $this->l('Visits (x100)'); $this->_query[0] = ' SELECT o.`date_add`, SUM(od.`product_quantity`) AS total FROM `'._DB_PREFIX_.'order_detail` od LEFT JOIN `'._DB_PREFIX_.'orders` o ON o.`id_order` = od.`id_order` WHERE od.`product_id` = '.intval($this->_id_product).' AND o.valid = 1 AND o.`date_add` BETWEEN '.$dateBetween.' GROUP BY o.`date_add`'; $this->_query[1] = ' SELECT dr.`time_start` AS date_add, (SUM(pv.`counter`) / 100) AS total FROM `'._DB_PREFIX_.'page_viewed` pv LEFT JOIN `'._DB_PREFIX_.'date_range` dr ON pv.`id_date_range` = dr.`id_date_range` LEFT JOIN `'._DB_PREFIX_.'page` p ON pv.`id_page` = p.`id_page` LEFT JOIN `'._DB_PREFIX_.'page_type` pt ON pt.`id_page_type` = p.`id_page_type` WHERE pt.`name` = \'product.php\' AND p.`id_object` = '.intval($this->_id_product).' AND dr.`time_start` BETWEEN '.$dateBetween.' AND dr.`time_end` BETWEEN '.$dateBetween.' GROUP BY dr.`time_start`'; break; case 3: $this->_query = ' SELECT product_attribute_id, SUM(od.`product_quantity`) AS total FROM `'._DB_PREFIX_.'orders` o LEFT JOIN `'._DB_PREFIX_.'order_detail` od ON o.`id_order` = od.`id_order` WHERE od.`product_id` = '.intval($this->_id_product).' AND o.valid = 1 AND o.`date_add` BETWEEN '.$dateBetween.' GROUP BY od.`product_attribute_id`'; $this->_titles['main'] = $this->l('Attributes'); break; } } protected function getData($layers) { if ($this->_option != 3) $this->setDateGraph($layers, true); else { $product = new Product($this->_id_product, false, intval($this->getLang())); $combArray = array(); $assocNames = array(); $combinaisons = $product->getAttributeCombinaisons(intval($this->getLang())); foreach ($combinaisons AS $k => $combinaison) $combArray[$combinaison['id_product_attribute']][] = array('group' => $combinaison['group_name'], 'attr' => $combinaison['attribute_name']); foreach ($combArray AS $id_product_attribute => $product_attribute) { $list = ''; foreach ($product_attribute AS $attribute) $list .= trim($attribute['group']).' - '.trim($attribute['attr']).', '; $list = rtrim($list, ', '); $assocNames[$id_product_attribute] = $list; } $result = Db::getInstance()->ExecuteS($this->_query); foreach ($result as $row) { $this->_values[] = $row['total']; $this->_legend[] = @$assocNames[$row['product_attribute_id']]; } } } protected function setYearValues($layers) { for ($i = 0; $i < $layers; $i++) { $result = Db::getInstance()->ExecuteS($this->_query[$i]); foreach ($result AS $row) $this->_values[$i][intval(substr($row['date_add'], 5, 2))] += $row['total']; } } protected function setMonthValues($layers) { for ($i = 0; $i < $layers; $i++) { $result = Db::getInstance()->ExecuteS($this->_query[$i]); foreach ($result AS $row) $this->_values[$i][intval(substr($row['date_add'], 8, 2))] += $row['total']; } } protected function setDayValues($layers) { for ($i = 0; $i < $layers; $i++) { $result = Db::getInstance()->ExecuteS($this->_query[$i]); foreach ($result AS $row) $this->_values[$i][intval(substr($row['date_add'], 11, 2))] += $row['total']; } } } ?>