Your IP : 216.73.216.220


Current Path : /home/rtorresani/www/vendor/magento/module-page-builder/Model/Catalog/
Upload File :
Current File : //home/rtorresani/www/vendor/magento/module-page-builder/Model/Catalog/ProductTotals.php

<?php
/**
 * Copyright © Magento, Inc. All rights reserved.
 * See COPYING.txt for license details.
 */

declare(strict_types=1);

namespace Magento\PageBuilder\Model\Catalog;

use Magento\Catalog\Api\CategoryRepositoryInterface;
use Magento\Catalog\Model\Product\Attribute\Source\Status;
use Magento\Catalog\Model\Product\Visibility;
use Magento\Catalog\Model\ResourceModel\Product\Collection;
use Magento\Catalog\Model\ResourceModel\Product\CollectionFactory;
use Magento\CatalogWidget\Model\Rule;
use Magento\Framework\Exception\LocalizedException;
use Magento\Framework\Exception\NoSuchEntityException;
use Magento\Rule\Model\Condition\Combine;
use Magento\Rule\Model\Condition\Sql\Builder;
use Magento\Widget\Helper\Conditions;
use Zend_Db_Select_Exception;

/**
 * Product totals for Products content type
 *
 * @SuppressWarnings(PHPMD.CouplingBetweenObjects)
 */
class ProductTotals
{
    /**
     * @var CollectionFactory
     */
    private $productCollectionFactory;

    /**
     * @var Builder
     */
    private $sqlBuilder;

    /**
     * @var Rule
     */
    private $rule;

    /**
     * @var Conditions
     */
    private $conditionsHelper;

    /**
     * @var CategoryRepositoryInterface
     */
    private $categoryRepository;

    /**
     * @param CollectionFactory $productCollectionFactory
     * @param Builder $sqlBuilder
     * @param Rule $rule
     * @param Conditions $conditionsHelper
     * @param CategoryRepositoryInterface $categoryRepository
     */
    public function __construct(
        CollectionFactory $productCollectionFactory,
        Builder $sqlBuilder,
        Rule $rule,
        Conditions $conditionsHelper,
        CategoryRepositoryInterface $categoryRepository
    ) {
        $this->productCollectionFactory = $productCollectionFactory;
        $this->sqlBuilder = $sqlBuilder;
        $this->rule = $rule;
        $this->conditionsHelper = $conditionsHelper;
        $this->categoryRepository = $categoryRepository;
    }

    /**
     * Decode the provided conditions
     *
     * @param string $conditions
     * @return Combine
     */
    private function decodeConditions(string $conditions): Combine
    {
        if ($conditions) {
            $conditions = $this->conditionsHelper->decode($conditions);
        }

        foreach ($conditions as $key => $condition) {
            if (!empty($condition['attribute'])) {
                if (in_array($condition['attribute'], ['special_from_date', 'special_to_date'])) {
                    $conditions[$key]['value'] = date('Y-m-d H:i:s', strtotime($condition['value']));
                }

                if ($condition['attribute'] == 'category_ids') {
                    $conditions[$key] = $this->updateAnchorCategoryConditions($condition);
                }
            }
        }

        $this->rule->loadPost(['conditions' => $conditions]);
        return $this->rule->getConditions();
    }

    /**
     * Update conditions if the category is an anchor category
     *
     * @param array $condition
     * @return array
     */
    private function updateAnchorCategoryConditions(array $condition): array
    {
        if (array_key_exists('value', $condition)) {
            $categoryId = $condition['value'];

            try {
                $category = $this->categoryRepository->get($categoryId);
            } catch (NoSuchEntityException $e) {
                return $condition;
            }

            if ($category->getIsAnchor() && $category->getChildren(true)) {
                $children = explode(',', $category->getChildren(true));

                $condition['operator'] = "()";
                $condition['value'] = array_merge([$categoryId], $children);
            }
        }

        return $condition;
    }

    /**
     * Retrieve product collection based on provided conditions
     *
     * @param string $conditions
     * @param bool $usePriceIndex use minimal price from price index to filter by price.
     * If TRUE only enabled products will be returned, as price indexer does not include disabled products
     * @return Collection
     * @throws LocalizedException
     */
    private function getProductCollection(string $conditions, bool $usePriceIndex = true): Collection
    {
        /** @var $collection Collection */
        $collection = $this->productCollectionFactory->create();
        if ($usePriceIndex && strpos($conditions, '"attribute":"price"') !== false) {
            $collection->addMinimalPrice();
        }

        $collection = $this->applyConditionsToCollection($conditions, $collection);
        $collection = $this->excludeLinkedProducts($collection);

        /**
         * Prevent retrieval of duplicate records. This may occur when multiselect product attribute matches
         * several allowed values from condition simultaneously
         */
        $collection->distinct(true);

        return $collection;
    }

    /**
     * Retrieve count of all enabled products
     *
     * @param string $conditions
     * @return int number of enabled products
     */
    private function getEnabledCount(string $conditions): int
    {
        $collection = $this->getProductCollection($conditions, true);
        $collection->addAttributeToFilter('status', Status::STATUS_ENABLED);
        return $collection->getSize();
    }

    /**
     * Retrieve count of all disabled products
     *
     * @param string $conditions
     * @return int number of disabled products
     */
    private function getDisabledCount(string $conditions): int
    {
        $collection = $this->getProductCollection($conditions, false);
        $collection->addAttributeToFilter('status', Status::STATUS_DISABLED);
        return $collection->getSize();
    }

    /**
     * Retrieve count of all not visible individually products
     *
     * @param string $conditions
     * @return int number of products not visible individually
     */
    private function getNotVisibleCount(string $conditions): int
    {
        $collection = $this->getProductCollection($conditions, true);
        $collection->addAttributeToFilter('status', Status::STATUS_ENABLED);
        $collection->addAttributeToFilter(
            'visibility',
            [
                Visibility::VISIBILITY_NOT_VISIBLE,
                Visibility::VISIBILITY_IN_SEARCH
            ]
        );
        return $collection->getSize();
    }

    /**
     * Exclude any linked products, e.g. simple products assigned to a configurable, bundle or group
     *
     * @param Collection $collection
     * @return Collection
     */
    private function excludeLinkedProducts(Collection $collection): Collection
    {
        $collection->getSelect()
            ->joinLeft(
                ['super_link_table' => $collection->getTable('catalog_product_super_link')],
                'super_link_table.product_id = e.entity_id',
                ['product_id']
            )
            ->joinLeft(
                ['link_table' => $collection->getTable('catalog_product_link')],
                'link_table.product_id = e.entity_id',
                ['product_id']
            )
            ->where('link_table.product_id IS NULL OR super_link_table.product_id IS NULL');
        return $collection;
    }

    /**
     * Apply conditions to collection
     *
     * @param string $conditions
     * @param Collection $collection
     * @return Collection
     * @throws LocalizedException
     */
    private function applyConditionsToCollection(string $conditions, Collection $collection): Collection
    {
        /** @var Combine $collectionConditions */
        $collectionConditions = $this->decodeConditions($conditions);
        $collectionConditions->collectValidatedAttributes($collection);
        $this->sqlBuilder->attachConditionToCollection($collection, $collectionConditions);
        return $collection;
    }

    /**
     * Retrieve product totals for collection
     *
     * @param string $conditions
     * @return array
     * @throws LocalizedException
     * @throws Zend_Db_Select_Exception
     */
    public function getProductTotals(string $conditions): array
    {
        $enabledCount = $this->getEnabledCount($conditions);
        $disabledCount = $this->getDisabledCount($conditions);
        $notVisibleCount = $this->getNotVisibleCount($conditions);

        return [
            'total' => $enabledCount + $disabledCount,
            'disabled' => $disabledCount,
            'notVisible' => $notVisibleCount,
        ];
    }
}