Собственные условия для правил работы с корзиной

Собственные условия для правил работы с корзиной

Отдел маркетинга может придумывать разнообразные скидки, но не все они реализуемы штатными условиями Битрикса. Тогда на помощь приходит событие OnCondSaleActionsControlBuildList.

Описание задачи

Правила работы с корзиной для интернет-магазина должны применяться только к товарам, если их цена не какого-то определенного типа (Типа цен).

Регистрация обработчика для события формирования правил работы с корзиной

// файл local/php_interface/init.php

// добавляем кастомное правило
$eventManager = \Bitrix\Main\EventManager::getInstance();
$eventManager->addEventHandlerCompatible(
    'sale',
    'OnCondSaleActionsControlBuildList',
    ['ProjectName\EventHelpers\SaleActionCustomPrice', 'GetControlDescr']
);

Класс с описанием логики работы

<?php
// файл local/php_interface/lib/eventhelpers/SaleActionCustomPrice.php

namespace ProjectName\EventHelpers;

use \Bitrix\Main\Loader;

class SaleActionCustomPrice  extends \CSaleActionCtrlAction
{
    /**
     * Получение имени класса
     * @return string
     */
    public static function GetClassName()
    {
        return __CLASS__;
    }

    /**
     * Получение ID условия
     * @return array|string
     */
    public static function GetControlID()
    {
        return "DiscountPriceType";
    }

    /**
     * Добавление пункта в список условий с указанием отдельной группы
     * @param $arParams
     * @return array
     * @throws \Bitrix\Main\ArgumentException
     * @throws \Bitrix\Main\LoaderException
     * @throws \Bitrix\Main\ObjectPropertyException
     * @throws \Bitrix\Main\SystemException
     */
    public static function GetControlShow($arParams)
    {
        $arControls = static::GetAtomsEx();
        $arResult = array(
            'controlgroup' => true,
            'group' =>  false,
            'label' => 'Кастомные правила',
            'showIn' => static::GetShowIn($arParams['SHOW_IN_GROUPS']),
            'children' =>  [array(
                'controlId' => static::GetControlID(),
                'group' => false,
                'label' => "Если тип цены не",
                'showIn' => static::GetShowIn($arParams['SHOW_IN_GROUPS']),
                'control' => array(
                    "Если тип цены не",
                    $arControls["PT"]
                )
            )]
        );

        return $arResult;
    }

    /**
     * Формирование данных для визуального представления условия
     * @param bool $strControlID
     * @param bool $boolEx
     * @return array
     * @throws \Bitrix\Main\ArgumentException
     * @throws \Bitrix\Main\LoaderException
     * @throws \Bitrix\Main\ObjectPropertyException
     * @throws \Bitrix\Main\SystemException
     */
    public static function GetAtomsEx($strControlID = false, $boolEx = false)
    {
        $boolEx = (true === $boolEx ? true : false);
        $priceList = [];
        if (Loader::includeModule('main')) {
            // Получение типов цен
            $arGroupPrice = \Bitrix\Catalog\GroupTable::getList([
                'select' => ['ID', 'NAME'],
                'cache' => [
                    'ttl' => 60,
                    'cache_joins' => true,
                ]
            ]);

            while ($el = $arGroupPrice->fetch()) {
                $priceList[$el['ID']] = $el['NAME'] . " [" . $el['ID'] . "]";
            }
        }

        $arAtomList = [
            "PT" => [
                "JS" => [
                    "id" => "PT",
                    "name" => "extra",
                    "type" => "select",
                    "values" => $priceList,
                    "defaultText" => "...",
                    "defaultValue" => "",
                    "first_option" => "..."
                ],
                "ATOM" => [
                    "ID" => "PT",
                    "FIELD_TYPE" => "string",
                    "FIELD_LENGTH" => 255,
                    "MULTIPLE" => "N",
                    "VALIDATE" => "list"
                ]
            ],
        ];
        if (!$boolEx) {
            foreach ($arAtomList as &$arOneAtom)
            {
                $arOneAtom = $arOneAtom["JS"];
            }
            if (isset($arOneAtom))
            {
                unset($arOneAtom);
            }
        }

        return $arAtomList;
    }

    /**
     * Функция должна вернуть колбэк того, что должно быть выполнено при наступлении условий
     * @param $arOneCondition
     * @param $arParams
     * @param $arControl
     * @param bool $arSubs
     * @return string
     */
    public static function Generate($arOneCondition, $arParams, $arControl, $arSubs = false)
    {
        return __CLASS__ . '::applyProductDiscount($row,' . $arOneCondition["PT"] . ')';
    }

    /**
     * Логика кастомного условия
     * @param $row
     * @param $priceType
     * @return bool
     */
    public static function applyProductDiscount($row, $priceType)
    {
        return $priceType != $row['PRICE_TYPE_ID'];
    }
}

Пример оформленного правила с использованием нового условия

Результат

Скидки в корзине перестали применяться к определенным типам цен. Данное условие можно расширять и модифицировать как угодно, добавляя более сложные условия применения. Модифицируя приведенный мной пример, можно реализовать разнообразные маркетинговые акции на сайте.