* @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0) */ namespace PrestaShop\PrestaShop\Core\Grid\Definition\Factory; use PrestaShop\PrestaShop\Core\CommandBus\CommandBusInterface; use PrestaShop\PrestaShop\Core\Domain\CmsPageCategory\Query\GetCmsPageCategoryNameForListing; use PrestaShop\PrestaShop\Core\Grid\Action\Bulk\BulkActionCollection; use PrestaShop\PrestaShop\Core\Grid\Action\Bulk\Type\SubmitBulkAction; use PrestaShop\PrestaShop\Core\Grid\Action\GridActionCollection; use PrestaShop\PrestaShop\Core\Grid\Action\Row\RowActionCollection; use PrestaShop\PrestaShop\Core\Grid\Action\Row\Type\LinkRowAction; use PrestaShop\PrestaShop\Core\Grid\Action\Type\SimpleGridAction; use PrestaShop\PrestaShop\Core\Grid\Column\ColumnCollection; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\ActionColumn; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\BulkActionColumn; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\DataColumn; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\PositionColumn; use PrestaShop\PrestaShop\Core\Grid\Column\Type\Common\ToggleColumn; use PrestaShop\PrestaShop\Core\Grid\Filter\Filter; use PrestaShop\PrestaShop\Core\Grid\Filter\FilterCollection; use PrestaShop\PrestaShop\Core\Hook\HookDispatcherInterface; use PrestaShop\PrestaShop\Core\Multistore\MultistoreContextCheckerInterface; use PrestaShopBundle\Form\Admin\Type\ReorderPositionsButtonType; use PrestaShopBundle\Form\Admin\Type\SearchAndResetType; use PrestaShopBundle\Form\Admin\Type\YesAndNoChoiceType; use Symfony\Component\Form\Extension\Core\Type\TextType; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\RequestStack; /** * Class responsible for providing columns, filters, actions for cms page list. */ class CmsPageDefinitionFactory extends AbstractGridDefinitionFactory { use BulkDeleteActionTrait; use DeleteActionTrait; public const GRID_ID = 'cms_page'; /** * @var int */ private $cmsCategoryParentId; /** * @var CommandBusInterface */ private $queryBus; /** * @var MultistoreContextCheckerInterface */ private $multistoreContextChecker; /** * @var bool */ private $isMultiStoreFeatureUsed; public function __construct( HookDispatcherInterface $hookDispatcher, CommandBusInterface $queryBus, RequestStack $requestStack, MultistoreContextCheckerInterface $multistoreContextChecker, $isMultiStoreFeatureUsed ) { parent::__construct($hookDispatcher); $this->queryBus = $queryBus; $this->setCmsPageCategoryParentId($requestStack); $this->multistoreContextChecker = $multistoreContextChecker; $this->isMultiStoreFeatureUsed = $isMultiStoreFeatureUsed; } /** * {@inheritdoc} */ protected function getId() { return self::GRID_ID; } /** * {@inheritdoc} */ protected function getName() { $cmsCategoryName = $this->queryBus->handle(new GetCmsPageCategoryNameForListing()); return $this->trans( 'Pages in category "%name%"', ['%name%' => $cmsCategoryName], 'Admin.Design.Feature' ); } /** * {@inheritdoc} */ protected function getColumns() { $columnCollection = (new ColumnCollection()) ->add((new BulkActionColumn('bulk')) ->setOptions([ 'bulk_field' => 'id_cms', ]) ) ->add((new DataColumn('id_cms')) ->setName($this->trans('ID', [], 'Admin.Global')) ->setOptions([ 'field' => 'id_cms', ]) ) ->add((new DataColumn('link_rewrite')) ->setName($this->trans('URL', [], 'Admin.Global')) ->setOptions([ 'field' => 'link_rewrite', ]) ) ->add((new DataColumn('meta_title')) ->setName($this->trans('Title', [], 'Admin.Global')) ->setOptions([ 'field' => 'meta_title', ]) ) ->add((new DataColumn('head_seo_title')) ->setName($this->trans('Meta title', [], 'Admin.Global')) ->setOptions([ 'field' => 'head_seo_title', ]) ) ->add((new ToggleColumn('active')) ->setName($this->trans('Displayed', [], 'Admin.Global')) ->setOptions([ 'field' => 'active', 'route' => 'admin_cms_pages_toggle', 'primary_field' => 'id_cms', 'route_param_name' => 'cmsId', ]) ) ->add((new ActionColumn('actions')) ->setName($this->trans('Actions', [], 'Admin.Global')) ->setOptions([ 'actions' => (new RowActionCollection()) ->add((new LinkRowAction('edit')) ->setName($this->trans('Edit', [], 'Admin.Actions')) ->setIcon('edit') ->setOptions([ 'route' => 'admin_cms_pages_edit', 'route_param_name' => 'cmsPageId', 'route_param_field' => 'id_cms', 'clickable_row' => true, ]) ) ->add( $this->buildDeleteAction( 'admin_cms_pages_delete', 'cmsId', 'id_cms', Request::METHOD_DELETE ) ), ]) ) ; if ($this->isAllShopContextOrShopFeatureIsNotUsed()) { $columnCollection ->addAfter( 'head_seo_title', (new PositionColumn('position')) ->setName($this->trans('Position', [], 'Admin.Global')) ->setOptions([ 'id_field' => 'id_cms', 'position_field' => 'position', 'update_method' => 'POST', 'update_route' => 'admin_cms_pages_update_position', 'record_route_params' => [ 'id_cms_category' => 'id_cms_category', ], ]) ); } return $columnCollection; } /** * {@inheritdoc} */ protected function getFilters() { $actionsTypeOptions = [ 'reset_route' => 'admin_common_reset_search_by_filter_id', 'reset_route_params' => [ 'filterId' => self::GRID_ID, ], 'redirect_route' => 'admin_cms_pages_index', ]; if ($this->cmsCategoryParentId) { $actionsTypeOptions['redirect_route_params'] = [ 'id_cms_category' => $this->cmsCategoryParentId, ]; $actionsTypeOptions['reset_route_params']['id_cms_category'] = $this->cmsCategoryParentId; } $filterCollection = (new FilterCollection()) ->add((new Filter('id_cms', TextType::class)) ->setTypeOptions([ 'required' => false, 'attr' => [ 'placeholder' => $this->trans('ID', [], 'Admin.Global'), ], ]) ->setAssociatedColumn('id_cms') ) ->add((new Filter('link_rewrite', TextType::class)) ->setTypeOptions([ 'required' => false, 'attr' => [ 'placeholder' => $this->trans('URL', [], 'Admin.Global'), ], ]) ->setAssociatedColumn('link_rewrite') ) ->add((new Filter('meta_title', TextType::class)) ->setTypeOptions([ 'required' => false, 'attr' => [ 'placeholder' => $this->trans('Title', [], 'Admin.Global'), ], ]) ->setAssociatedColumn('meta_title') ) ->add((new Filter('head_seo_title', TextType::class)) ->setTypeOptions([ 'required' => false, 'attr' => [ 'placeholder' => $this->trans('Meta title', [], 'Admin.Global'), ], ]) ->setAssociatedColumn('head_seo_title') ) ->add((new Filter('active', YesAndNoChoiceType::class)) ->setAssociatedColumn('active') ) ->add((new Filter('actions', SearchAndResetType::class)) ->setTypeOptions($actionsTypeOptions) ->setAssociatedColumn('actions') ) ; if ($this->isAllShopContextOrShopFeatureIsNotUsed()) { $filterCollection ->add((new Filter('position', ReorderPositionsButtonType::class)) ->setAssociatedColumn('position') ) ; } return $filterCollection; } /** * {@inheritdoc} */ protected function getGridActions() { return (new GridActionCollection()) ->add((new SimpleGridAction('common_refresh_list')) ->setName($this->trans('Refresh list', [], 'Admin.Advparameters.Feature')) ->setIcon('refresh') ) ->add((new SimpleGridAction('common_show_query')) ->setName($this->trans('Show SQL query', [], 'Admin.Actions')) ->setIcon('code') ) ->add((new SimpleGridAction('common_export_sql_manager')) ->setName($this->trans('Export to SQL Manager', [], 'Admin.Actions')) ->setIcon('storage') ) ; } /** * {@inheritdoc} */ protected function getBulkActions() { return (new BulkActionCollection()) ->add((new SubmitBulkAction('enable_selection')) ->setName($this->trans('Enable selection', [], 'Admin.Actions')) ->setOptions([ 'submit_route' => 'admin_cms_pages_bulk_enable_status', ]) ) ->add((new SubmitBulkAction('disable_selection')) ->setName($this->trans('Disable selection', [], 'Admin.Actions')) ->setOptions([ 'submit_route' => 'admin_cms_pages_bulk_disable_status', ]) ) ->add( $this->buildBulkDeleteAction('admin_cms_pages_bulk_delete') ) ; } /** * Sets cms page category parent id directly from request attribute. On not found case, it assigns the default one. * * @param RequestStack $requestStack */ private function setCmsPageCategoryParentId(RequestStack $requestStack) { $request = $requestStack->getCurrentRequest(); if (null !== $request && $request->query->getInt('id_cms_category')) { $this->cmsCategoryParentId = $request->query->getInt('id_cms_category'); } } /** * This function is required due to in cms_category contains position column - on ideal case cms_category_shop * should have this column configured instead. * In such case the condition would be $this->multistoreContextChecker->isSingleShopContext() * * @return bool */ private function isAllShopContextOrShopFeatureIsNotUsed() { return $this->multistoreContextChecker->isAllShopContext() || !$this->isMultiStoreFeatureUsed; } }