* @copyright 2007-2011 PrestaShop SA * @version Release: $Revision: 8696 $ * @license http://opensource.org/licenses/afl-3.0.php Academic Free License (AFL 3.0) * International Registered Trademark & Property of PrestaShop SA */ if (!defined('_PS_VERSION_')) exit; class ProductComments extends Module { const INSTALL_SQL_FILE = 'install.sql'; private $_html = ''; private $_postErrors = array(); private $_productCommentsCriterionTypes = array(); private $_baseUrl; public function __construct() { $this->name = 'productcomments'; $this->tab = 'front_office_features'; $this->version = '2.1'; $this->author = 'PrestaShop'; $this->need_instance = 0; parent::__construct(); $this->displayName = $this->l('Product Comments'); $this->description = $this->l('Allow users to post comment about a product.'); } public function install() { if (!file_exists(dirname(__FILE__).'/'.self::INSTALL_SQL_FILE)) return false; elseif (!$sql = file_get_contents(dirname(__FILE__).'/'.self::INSTALL_SQL_FILE)) return false; $sql = str_replace(array('PREFIX_', 'ENGINE_TYPE'), array(_DB_PREFIX_, _MYSQL_ENGINE_), $sql); $sql = preg_split("/;\s*[\r\n]+/", trim($sql)); foreach ($sql AS $query) if (!Db::getInstance()->Execute(trim($query))) return false; if (parent::install() == false OR $this->registerHook('productTab') == false OR $this->registerHook('extraProductComparison') == false OR $this->registerHook('productTabContent') == false OR $this->registerHook('header') == false OR !Configuration::updateValue('PRODUCT_COMMENTS_MINIMAL_TIME', 30) OR !Configuration::updateValue('PRODUCT_COMMENTS_ALLOW_GUESTS', 0) OR !Configuration::updateValue('PRODUCT_COMMENTS_MODERATE', 1)) return false; return true; } function uninstall() { if (!parent::uninstall() OR !Configuration::deleteByName('PRODUCT_COMMENTS_MODERATE') OR !Configuration::deleteByName('PRODUCT_COMMENTS_ALLOW_GUESTS') OR !Configuration::deleteByName('PRODUCT_COMMENTS_MINIMAL_TIME')) return false; return true; } protected function _postProcess() { if (Tools::isSubmit('submitModerate')) { Configuration::updateValue('PRODUCT_COMMENTS_MODERATE', (int)Tools::getValue('moderate')); Configuration::updateValue('PRODUCT_COMMENTS_ALLOW_GUESTS', (int)Tools::getValue('allow_guest')); Configuration::updateValue('PRODUCT_COMMENTS_MINIMAL_TIME', (int)Tools::getValue('product_comments_minimal_time')); $this->_html .= '
'.$this->l('Confirmation').''.$this->l('Settings updated').'
'; } if ($id_criterion = (int)Tools::getValue('deleteCriterion')) { $productCommentCriterion = new ProductCommentCriterion((int)$id_criterion); if ($productCommentCriterion->id) if ($productCommentCriterion->delete()) $this->_html .= '
'.$this->l('Confirmation').''.$this->l('Criterion deleted').'
'; } } public function getContent() { include_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $this->_setBaseUrl(); $this->_productCommentsCriterionTypes = ProductCommentCriterion::getTypes(); $this->_html = '

'.$this->displayName.'

'; $this->_postProcess(); $this->_checkModerateComment(); $this->_checkCriterion(); $this->_updateApplicationCriterion(); return $this->_html.$this->_displayForm(); } private function _setBaseUrl() { $this->_baseUrl = 'index.php?'; foreach ($_GET AS $k => $value) if (!in_array($k, array('deleteCriterion', 'editCriterion'))) $this->_baseUrl .= $k.'='.$value.'&'; $this->_baseUrl = rtrim($this->_baseUrl, '&'); } private function _checkModerateComment() { $action = Tools::getValue('action'); if (empty($action) === false && (int)(Configuration::get('PRODUCT_COMMENTS_MODERATE'))) { $product_comments = Tools::getValue('id_product_comment'); if (sizeof($product_comments)) { require_once(dirname(__FILE__).'/ProductComment.php'); switch ($action) { case 'accept': foreach ($product_comments AS $id_product_comment) { if (!$id_product_comment) continue; $comment = new ProductComment((int)$id_product_comment); $comment->validate(); } break; case 'delete': foreach ($product_comments AS $id_product_comment) { if (!$id_product_comment) continue; $comment = new ProductComment((int)$id_product_comment); $comment->delete(); ProductComment::deleteGrades((int)$id_product_comment); } break; default: ; } } } } private function _checkCriterion() { $action_criterion = Tools::getValue('criterion_action'); $name = Tools::getValue('criterion'); if (Tools::isSubmit('submitAddCriterion')) { global $cookie; require_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $languages = Language::getLanguages(); $id_criterion = (int)Tools::getValue('id_product_comment_criterion'); $productCommentCriterion = new ProductCommentCriterion((int)$id_criterion); foreach ($languages AS $lang) $productCommentCriterion->name[(int)$lang['id_lang']] = Tools::getValue('criterion_'.(int)$lang['id_lang']); // Check default language criterion name $defaultLanguage = new Language((int)(Configuration::get('PS_LANG_DEFAULT'))); if (!Tools::getValue('criterion_'.$defaultLanguage->id)) { $this->_html .= '
'.$this->l('The field Name is required at least in').' '.$defaultLanguage->name.'
'; return; } $productCommentCriterion->id_product_comment_criterion_type = (int)Tools::getValue('criterion_type'); $productCommentCriterion->active = (int)Tools::getValue('criterion_active'); if ($productCommentCriterion->save()) $this->_html .= '
'.$this->l('Confirmation').''.(Tools::getValue('editCriterion') ? $this->l('Criterion updated') : $this->l('Criterion added')).'
'; } elseif (!empty($action_criterion) AND empty($name)) { $id_product_comment_criterion = Tools::getValue('id_product_comment_criterion'); require_once(dirname(__FILE__).'/ProductCommentCriterion.php'); switch ($action_criterion) { case 'edit': ProductCommentCriterion::update($id_product_comment_criterion, Tools::getValue('criterion_id_lang'), Tools::getValue('criterion_name')); break; case 'delete': ProductCommentCriterion::delete($id_product_comment_criterion); break; default: ; } } } private function _updateApplicationCriterion() { if (Tools::isSubmit('submitApplicationCriterion')) { include_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $id_criterion = (int)Tools::getValue('id_criterion'); $productCommentCriterion = new ProductCommentCriterion((int)$id_criterion); if ($productCommentCriterion->id) { if ($productCommentCriterion->id_product_comment_criterion_type == 2) { $productCommentCriterion->deleteCategories(); if ($categories = Tools::getValue('id_product')) if (sizeof($categories)) foreach ($categories AS $id_category) $productCommentCriterion->addCategory((int)$id_category); } elseif ($productCommentCriterion->id_product_comment_criterion_type == 3) { $productCommentCriterion->deleteProducts(); if ($products = Tools::getValue('id_product')) if (sizeof($products)) foreach ($products AS $product) $productCommentCriterion->addProduct((int)$product); } } $this->_html .= '
'.$this->l('Confirmation').''.$this->l('Settings updated').'
'; } } private function _displayForm() { $this->_displayFormModerate(); $this->_displayFormConfigurationCriterion(); $this->_displayFormApplicationCriterion(); return $this->_html; } private function _displayFormModerate() { $this->_html = '
'.$this->l('Configuration').'
'.$this->l('seconds').'

'.$this->l('Moderate Comments').''; if (Configuration::get('PRODUCT_COMMENTS_MODERATE')) { require_once(dirname(__FILE__).'/ProductComment.php'); $comments = ProductComment::getByValidate(); if (sizeof($comments)) { $this->_html .= '

'; foreach ($comments AS $comment) $this->_html .= ''; $this->_html .= '
'.$this->l('Author').' '.$this->l('Comment').' '.$this->l('Product name').' '.$this->l('Actions').'
'.htmlspecialchars($comment['customer_name'], ENT_COMPAT, 'UTF-8').'. '.htmlspecialchars($comment['content'], ENT_COMPAT, 'UTF-8').' '.$comment['id_product'].' - '.htmlspecialchars($comment['name'], ENT_COMPAT, 'UTF-8').' '.$this->l('Accept').' '.$this->l('Delete').'
'.$this->l('Selection:').' '.$this->l('Accept').' '.$this->l('Delete').'
'; } else $this->_html .= $this->l('No comments to validate at this time.'); } $this->_html .= '

'; } private function _displayFormConfigurationCriterion() { global $cookie; $langs = Language::getLanguages(false); $id_lang_default = (int)Configuration::get('PS_LANG_DEFAULT'); $id_criterion = (int)Tools::getValue('editCriterion'); $criterion = new ProductCommentCriterion((int)$id_criterion); $languageIds = 'criterion'; $this->_html .= '
'.$this->l('Add a new comment criterion').'

'.$this->l('You can define several criterions to help your customers during their review. For instance: efficiency, lightness, design.').'

'.$this->l('You can add a new criterion below:').'

'; foreach ($langs AS $lang) $this->_html .= '
'; $this->_html .= $this->displayFlags($langs, (int)$id_lang_default, $languageIds, 'criterion', true); $this->_html .= '
 
active ? 'checked="checked" ' : '').'/> active ? 'checked="checked" ' : '').'/>
'; require_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $criterions = ProductCommentCriterion::getCriterions((int)$cookie->id_lang); if (sizeof($criterions)) { $this->_html.= '
'; foreach ($criterions AS $criterion) { $this->_html .= ''; } $this->_html .= '
'.$this->l('Criterion').' '.$this->l('Type').' '.$this->l('Status').' '.$this->l('Actions').'
'.$criterion['name'].' '.$this->_productCommentsCriterionTypes[(int)$criterion['id_product_comment_criterion_type']].' '.$this->l('Edit').' '.$this->l('Delete').'
'; } $this->_html .= '

'; } private function _displayFormApplicationCriterion() { global $cookie; include_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $criterions = ProductCommentCriterion::getCriterions((int)$cookie->id_lang, false, true); $id_criterion = (int)Tools::getValue('updateCriterion'); if ($id_criterion) { $criterion = new ProductCommentCriterion((int)$id_criterion); if ($criterion->id_product_comment_criterion_type == 2) { $categories = Category::getSimpleCategories((int)$cookie->id_lang); $criterion_categories = $criterion->getCategories(); } elseif ($criterion->id_product_comment_criterion_type == 3) { $criterion_products = $criterion->getProducts(); $products = Product::getSimpleProducts((int)$cookie->id_lang); } } foreach ($criterions AS $key => $foo) if ($foo['id_product_comment_criterion_type'] == 1) unset($criterions[$key]); if (sizeof($criterions)) { $this->_html .= '
'.$this->l('Manage criterions scope').'

'.$this->l('Only criterions restricted to categories or products can be configured below:').'

'; if ($id_criterion AND $criterion->id_product_comment_criterion_type != 1) { $this->_html .='

'; if ($criterion->id_product_comment_criterion_type == 3) foreach ($products AS $product) $this->_html .=''; elseif ($criterion->id_product_comment_criterion_type == 2) foreach ($categories AS $category) $this->_html .=''; $this->_html .='
'.$this->l('ID').' '.($criterion->id_product_comment_criterion_type == 3 ? $this->l('Product Name') : $this->l('Category Name')).'
'.(int)$product['id_product'].''.$product['name'].'
'.(int)$category['id_category'].''.$category['name'].'
'; } $this->_html .= '
'; } } public function hookProductTab($params) { global $smarty, $cookie; require_once(dirname(__FILE__).'/ProductComment.php'); require_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $smarty->assign(array( 'allow_guests' => (int)Configuration::get('PRODUCT_COMMENTS_ALLOW_GUESTS'), 'comments' => ProductComment::getByProduct((int)($_GET['id_product'])), 'criterions' => ProductCommentCriterion::getByProduct((int)($_GET['id_product']), (int)($cookie->id_lang)), 'nbComments' => (int)(ProductComment::getCommentNumber((int)($_GET['id_product']))))); return ($this->display(__FILE__, '/tab.tpl')); } private function _frontOfficePostProcess() { global $smarty, $cookie, $errors; require_once(dirname(__FILE__).'/ProductComment.php'); require_once(dirname(__FILE__).'/ProductCommentCriterion.php'); $allow_guests = (int)Configuration::get('PRODUCT_COMMENTS_ALLOW_GUESTS'); if (Tools::isSubmit('submitMessage') AND (empty($cookie->id_customer) === false OR ($cookie->id_guest AND $allow_guests))) { $id_guest = (!$id_customer = (int)$cookie->id_customer) ? (int)$cookie->id_guest : false; $customerComment = ProductComment::getByCustomer((int)(Tools::getValue('id_product')), (int)$cookie->id_customer, true, (int)$id_guest); if (!$customerComment OR ($customerComment AND (strtotime($customerComment['date_add']) + Configuration::get('PRODUCT_COMMENTS_MINIMAL_TIME')) < time())) { $customer_name = false; if ($id_guest AND (!$customer_name = Tools::getValue('customer_name'))) $errors[] = $this->l('Please fill your name'); if (!sizeof($errors) AND Tools::getValue('content')) { $comment = new ProductComment(); $comment->content = strip_tags(Tools::getValue('content')); $comment->id_product = (int)$_GET['id_product']; $comment->id_customer = (int)$cookie->id_customer; $comment->id_guest = (int)$id_guest; $comment->customer_name = pSQL($customer_name); $comment->title = pSQL(Tools::getValue('title')); $comment->grade = 0; $comment->validate = 0; if (!$comment->content) $errors[] = $this->l('Invalid comment text posted.'); else { $comment->save(); for ($i = 1, $grade = 0; isset($_POST[$i.'_grade']) === true; ++$i) { $cgrade = (int)Tools::getValue($i.'_grade'); $grade += $cgrade; $productCommentCriterion = new ProductCommentCriterion((int)Tools::getValue('id_product_comment_criterion_'.$i)); if ($productCommentCriterion->id) $productCommentCriterion->addGrade($comment->id, $cgrade); } if (($i - 1) > 0) $comment->grade = ($grade / ($i - 1)); if (!$comment->save()) $errors[] = $this->l('An error occurred while saving your comment.'); else $smarty->assign('confirmation', $this->l('Comment posted.').((int)(Configuration::get('PRODUCT_COMMENTS_MODERATE')) ? ' '.$this->l('Awaiting moderator validation.') : '')); } } else $errors[] = $this->l('Comment text is required.'); } else $errors[] = $this->l('You should wait').' '.Configuration::get('PRODUCT_COMMENTS_MINIMAL_TIME').' '.$this->l('seconds before posting a new comment'); } } public function hookProductTabContent($params) { global $smarty, $cookie; $id_guest = (!$id_customer = (int)$cookie->id_customer) ? (int)$cookie->id_guest : false; $customerComment = ProductComment::getByCustomer((int)(Tools::getValue('id_product')), (int)$cookie->id_customer, true, (int)$id_guest); $averages = ProductComment::getAveragesByProduct((int)Tools::getValue('id_product'), (int)$cookie->id_lang); $averageTotal = 0; foreach ($averages AS $average) $averageTotal += (float)($average); $averageTotal = count($averages) ? ($averageTotal / count($averages)) : 0; $smarty->assign(array( 'logged' => (int)$cookie->id_customer, 'action_url' => '', 'comments' => ProductComment::getByProduct((int)Tools::getValue('id_product')), 'criterions' => ProductCommentCriterion::getByProduct((int)Tools::getValue('id_product'), (int)$cookie->id_lang), 'averages' => $averages, 'product_comment_path' => $this->_path, 'averageTotal' => $averageTotal, 'allow_guests' => (int)Configuration::get('PRODUCT_COMMENTS_ALLOW_GUESTS'), 'too_early' => ($customerComment AND (strtotime($customerComment['date_add']) + Configuration::get('PRODUCT_COMMENTS_MINIMAL_TIME')) > time()), 'delay' => Configuration::get('PRODUCT_COMMENTS_MINIMAL_TIME'))); $controller = new FrontController(); $controller->pagination((int)ProductComment::getCommentNumber((int)Tools::getValue('id_product'))); return ($this->display(__FILE__, '/productcomments.tpl')); } public function hookHeader() { $this->_frontOfficePostProcess(); } public function hookExtraProductComparison($params) { global $smarty, $cookie; $list_grades = array(); $list_product_grades = array(); $list_product_average = array(); $list_product_comment = array(); foreach ($params['list_ids_product'] AS $id_product) { $grades = ProductComment::getAveragesByProduct((int)$id_product, (int)$cookie->id_lang); $criterions = ProductCommentCriterion::getByProduct((int)$id_product, (int)$cookie->id_lang); $grade_total = 0; if (sizeof($grades) > 0) { foreach ($criterions AS $criterion) { if(isset($grades[$criterion['id_product_comment_criterion']])) { $list_product_grades[$criterion['id_product_comment_criterion']][$id_product] = $grades[$criterion['id_product_comment_criterion']]; $grade_total += (float)($grades[$criterion['id_product_comment_criterion']]); } else $list_product_grades[$criterion['id_product_comment_criterion']][$id_product] = 0; if (!array_key_exists($criterion['id_product_comment_criterion'], $list_grades)) $list_grades[$criterion['id_product_comment_criterion']] = $criterion['name']; } $list_product_average[$id_product] = $grade_total / sizeof($criterion); $list_product_comment[$id_product] = ProductComment::getByProduct($id_product, 0, 3); } } if (sizeof($list_grades) < 1) return false; $smarty->assign(array('grades' => $list_grades, 'product_grades' => $list_product_grades, 'list_ids_product' => $params['list_ids_product'], 'list_product_average' => $list_product_average, 'product_comments' => $list_product_comment)); return $this->display(__FILE__,'/products-comparison.tpl'); } }