* @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/AFL-3.0 Academic Free License 3.0 (AFL-3.0) */ if (!defined('_PS_VERSION_')) { exit; } use PrestaShop\PrestaShop\Core\Module\WidgetInterface; class Ps_CategoryTree extends Module implements WidgetInterface { /** * @var int A way to display the category tree: Home category */ const CATEGORY_ROOT_HOME = 0; /** * @var int A way to display the category tree: Current category */ const CATEGORY_ROOT_CURRENT = 1; /** * @var int A way to display the category tree: Parent category */ const CATEGORY_ROOT_PARENT = 2; /** * @var int A way to display the category tree: Current category and its parent (if exists) */ const CATEGORY_ROOT_CURRENT_PARENT = 3; public function __construct() { $this->name = 'ps_categorytree'; $this->tab = 'front_office_features'; $this->version = '3.0.1'; $this->author = 'PrestaShop'; $this->bootstrap = true; parent::__construct(); $this->displayName = $this->trans('Category tree links', [], 'Modules.Categorytree.Admin'); $this->description = $this->trans('Help navigation on your store, show your visitors current category and subcategories.', [], 'Modules.Categorytree.Admin'); $this->ps_versions_compliancy = ['min' => '1.7.7.0', 'max' => _PS_VERSION_]; } public function install() { return parent::install() && $this->registerHook('displayLeftColumn'); } public function uninstall() { if (!parent::uninstall() || !Configuration::deleteByName('BLOCK_CATEG_MAX_DEPTH') || !Configuration::deleteByName('BLOCK_CATEG_ROOT_CATEGORY')) { return false; } return true; } public function getContent() { $output = ''; if (Tools::isSubmit('submitBlockCategories')) { $maxDepth = (int) (Tools::getValue('BLOCK_CATEG_MAX_DEPTH')); if ($maxDepth < 0) { $output .= $this->displayError($this->trans('Maximum depth: Invalid number.', [], 'Admin.Notifications.Error')); } else { Configuration::updateValue('BLOCK_CATEG_MAX_DEPTH', (int) $maxDepth); Configuration::updateValue('BLOCK_CATEG_SORT_WAY', Tools::getValue('BLOCK_CATEG_SORT_WAY')); Configuration::updateValue('BLOCK_CATEG_SORT', Tools::getValue('BLOCK_CATEG_SORT')); Configuration::updateValue('BLOCK_CATEG_ROOT_CATEGORY', Tools::getValue('BLOCK_CATEG_ROOT_CATEGORY')); Tools::redirectAdmin(AdminController::$currentIndex . '&configure=' . $this->name . '&token=' . Tools::getAdminTokenLite('AdminModules') . '&conf=6'); } } return $output . $this->renderForm(); } /** * Format category into an array compatible with existing templates. */ private function formatCategory($rawCategory, $idsOfCategoriesInPath): array { $children = []; if (!empty($rawCategory['children'])) { foreach ($rawCategory['children'] as $k => $v) { $children[$k] = $this->formatCategory($v, $idsOfCategoriesInPath); } } return [ 'id' => $rawCategory['id_category'], 'link' => $this->context->link->getCategoryLink($rawCategory['id_category'], $rawCategory['link_rewrite']), 'name' => $rawCategory['name'], 'desc' => $rawCategory['description'], 'children' => $children, 'in_path' => in_array($rawCategory['id_category'], $idsOfCategoriesInPath), ]; } private function getCategories($category): array { // Determine max depth to get categories $maxdepth = (int) Configuration::get('BLOCK_CATEG_MAX_DEPTH'); if ($maxdepth > 0) { $maxdepth += $category->level_depth; } // Define filters to get categories $groups = Customer::getGroupsStatic((int) $this->context->customer->id); $sqlFilter = $maxdepth ? 'AND c.`level_depth` <= ' . (int) $maxdepth : ''; $orderBy = ' ORDER BY c.`level_depth` ASC, ' . (Configuration::get('BLOCK_CATEG_SORT') ? 'cl.`name`' : 'category_shop.`position`') . ' ' . (Configuration::get('BLOCK_CATEG_SORT_WAY') ? 'DESC' : 'ASC'); // Retrieve them using the built in method $categories = Category::getNestedCategories($category->id, $this->context->language->id, true, $groups, true, $sqlFilter, $orderBy); if (empty($categories)) { return []; } // Get path to current category so we can use it for marking $idsOfCategoriesInPath = $this->getIdsOfCategoriesInPathToCurrentCategory(); // And do our formatting foreach ($categories as $k => $v) { $categories[$k] = $this->formatCategory($v, $idsOfCategoriesInPath); } return array_shift($categories); } public function renderForm() { $fields_form = [ 'form' => [ 'legend' => [ 'title' => $this->trans('Settings', [], 'Admin.Global'), 'icon' => 'icon-cogs', ], 'input' => [ [ 'type' => 'radio', 'label' => $this->trans('Category root', [], 'Modules.Categorytree.Admin'), 'name' => 'BLOCK_CATEG_ROOT_CATEGORY', 'hint' => $this->trans('Select which category is displayed in the block. The current category is the one the visitor is currently browsing.', [], 'Modules.Categorytree.Admin'), 'values' => [ [ 'id' => 'home', 'value' => static::CATEGORY_ROOT_HOME, 'label' => $this->trans('Home category', [], 'Modules.Categorytree.Admin'), ], [ 'id' => 'current', 'value' => static::CATEGORY_ROOT_CURRENT, 'label' => $this->trans('Current category', [], 'Modules.Categorytree.Admin'), ], [ 'id' => 'parent', 'value' => static::CATEGORY_ROOT_PARENT, 'label' => $this->trans('Parent category', [], 'Modules.Categorytree.Admin'), ], [ 'id' => 'current_parent', 'value' => static::CATEGORY_ROOT_CURRENT_PARENT, 'label' => $this->trans('Current category, unless it has no subcategories, in which case the parent category of the current category is used', [], 'Modules.Categorytree.Admin'), ], ], ], [ 'type' => 'text', 'label' => $this->trans('Maximum depth', [], 'Modules.Categorytree.Admin'), 'name' => 'BLOCK_CATEG_MAX_DEPTH', 'desc' => $this->trans('Set the maximum depth of category sublevels displayed in this block (0 = infinite).', [], 'Modules.Categorytree.Admin'), ], [ 'type' => 'radio', 'label' => $this->trans('Sort', [], 'Admin.Actions'), 'name' => 'BLOCK_CATEG_SORT', 'values' => [ [ 'id' => 'name', 'value' => 1, 'label' => $this->trans('By name', [], 'Admin.Global'), ], [ 'id' => 'position', 'value' => 0, 'label' => $this->trans('By position', [], 'Admin.Global'), ], ], ], [ 'type' => 'radio', 'label' => $this->trans('Sort order', [], 'Admin.Actions'), 'name' => 'BLOCK_CATEG_SORT_WAY', 'values' => [ [ 'id' => 'name', 'value' => 1, 'label' => $this->trans('Descending', [], 'Admin.Global'), ], [ 'id' => 'position', 'value' => 0, 'label' => $this->trans('Ascending', [], 'Admin.Global'), ], ], ], ], 'submit' => [ 'title' => $this->trans('Save', [], 'Admin.Actions'), ], ], ]; $helper = new HelperForm(); $helper->show_toolbar = false; $helper->table = $this->table; $helper->submit_action = 'submitBlockCategories'; $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 = [ 'fields_value' => $this->getConfigFieldsValues(), ]; return $helper->generateForm([$fields_form]); } public function getConfigFieldsValues() { return [ 'BLOCK_CATEG_MAX_DEPTH' => Tools::getValue('BLOCK_CATEG_MAX_DEPTH', Configuration::get('BLOCK_CATEG_MAX_DEPTH')), 'BLOCK_CATEG_SORT_WAY' => Tools::getValue('BLOCK_CATEG_SORT_WAY', Configuration::get('BLOCK_CATEG_SORT_WAY')), 'BLOCK_CATEG_SORT' => Tools::getValue('BLOCK_CATEG_SORT', Configuration::get('BLOCK_CATEG_SORT')), 'BLOCK_CATEG_ROOT_CATEGORY' => Tools::getValue('BLOCK_CATEG_ROOT_CATEGORY', Configuration::get('BLOCK_CATEG_ROOT_CATEGORY')), ]; } public function renderWidget($hookName = null, array $configuration = []) { $this->smarty->assign($this->getWidgetVariables($hookName, $configuration)); return $this->fetch('module:ps_categorytree/views/templates/hook/ps_categorytree.tpl'); } public function getWidgetVariables($hookName = null, array $configuration = []) { switch (Configuration::get('BLOCK_CATEG_ROOT_CATEGORY')) { // Always the home category case static::CATEGORY_ROOT_HOME: $rootCategory = $this->getHomeCategory(); break; // Always the current category case static::CATEGORY_ROOT_CURRENT: $rootCategory = $this->getCurrentCategory(); break; // Always the parent category case static::CATEGORY_ROOT_PARENT: $rootCategory = $this->tryToGetParentCategoryIfAvailable($this->getCurrentCategory()); break; // Current category, unless it has no subcategories, in which case the parent category of the current category is used case static::CATEGORY_ROOT_CURRENT_PARENT: $rootCategory = $this->getCurrentCategory(); if (!$rootCategory->getSubCategories($rootCategory->id, true)) { $rootCategory = $this->tryToGetParentCategoryIfAvailable($rootCategory); } break; default: $rootCategory = $this->getHomeCategory(); } return [ 'categories' => $this->getCategories($rootCategory), 'currentCategory' => $rootCategory->id, ]; } /* * Tries to retrieve current category from the context. In case of category controller, it's the category. * In case of product controller, it's either the default category of the product, or the category the customer * came from. This is resolved by the ProductController. */ private function getCurrentCategory(): Category { /* * We check several things: * If the controller has the method * If we are on the correct controller * If we have some sensible data and the category is properly loaded */ if ( !method_exists($this->context->controller, 'getCategory') || (!$this->context->controller instanceof CategoryController && !$this->context->controller instanceof ProductController) || empty($this->context->controller->getCategory()) || !Validate::isLoadedObject($this->context->controller->getCategory()) ) { return $this->getHomeCategory(); } return $this->context->controller->getCategory(); } /* * Tries to get a parent of the current category. * If we are already on the top of the tree, it will return the input. * * Three cases can happen: * - This category has a normal active parent * - This category has a disabled parent * - This category is already the home category */ private function tryToGetParentCategoryIfAvailable($category): Category { // If we are already on the top of the tree, nothing to do here if ($category->is_root_category || !$category->id_parent || $category->id == Configuration::get('PS_HOME_CATEGORY')) { return $category; } // We try to load the parent $parentCategory = new Category($category->id_parent, $this->context->language->id); // If the parent is malfunctioned somehow, we can't do anything and we return the home category if (!Validate::isLoadedObject($parentCategory)) { return $this->getHomeCategory(); } // Now we have a valid parent category, let's check it. It must be active, accessible and belong to the shop. // Same conditions as in CategoryController. If it fails, we select the next parent. if (!$parentCategory->active || !$category->checkAccess((int) $this->context->customer->id) || !$category->existsInShop($this->context->shop->id)) { return $this->tryToGetParentCategoryIfAvailable($parentCategory); } return $parentCategory; } private function getIdsOfCategoriesInPathToCurrentCategory(): array { // Call built in method to retrieve all parents, including the current category $categories = $this->getCurrentCategory()->getParentsCategories(); return array_column($categories, 'id_category'); } private function getHomeCategory(): Category { return new Category((int) Configuration::get('PS_HOME_CATEGORY'), $this->context->language->id); } }