Skip to content
MiniShop3
MiniShop3
Современный компонент интернет-магазина для MODX 3
  1. Компоненты
  2. MiniShop3
  3. Разработка
  4. Backend API
  5. API опций

API опций

Программный интерфейс для работы с системой опций MiniShop3 из PHP-кода.

Опции в MiniShop3 реализованы через EAV-паттерн (Entity-Attribute-Value) и состоят из трёх моделей:

  • msOption — определение опции (ключ, название, тип)
  • msCategoryOption — привязка опции к категории (активность, обязательность, позиция)
  • msProductOption — значение опции для конкретного товара
msOption (color, "Цвет", combo-multiple)
    ├── msCategoryOption (option → категория "Одежда", active=true)
    │       ├── msProductOption (product=10, key=color, value="Red")
    │       ├── msProductOption (product=10, key=color, value="Blue")
    │       └── msProductOption (product=11, key=color, value="Green")
    └── msCategoryOption (option → категория "Обувь", active=true)
            └── msProductOption (product=20, key=color, value="Black")

OptionService (фасад)

Основной сервис для работы с опциями. Объединяет три специализированных суб-сервиса.

php
$optionService = $modx->services->get('ms3_option_service');

Чтение опций товара

php
// Все значения опций товара
$values = $optionService->getProductOptionValues($productId);
// ['color' => ['Red', 'Blue'], 'size' => ['L', 'XL']]

// Определённые опции
$values = $optionService->getProductOptionValues($productId, ['color']);
// ['color' => ['Red', 'Blue']]

Загрузка для шаблонов

Метод loadOptionsForProduct возвращает данные с метаинформацией, подготовленные для Fenom-шаблонов:

php
$options = $optionService->loadOptionsForProduct($productId);
// [
//     'color' => ['Red', 'Blue'],
//     'color.caption' => 'Цвет',
//     'color.type' => 'combo-multiple',
//     'color.description' => 'Выберите цвет',
//     'color.category_name' => 'Свойства товара',
//     'size' => ['L', 'XL'],
//     'size.caption' => 'Размер',
//     ...
// ]

// Без метаданных (только ключи и значения)
$options = $optionService->loadOptionsForProduct($productId, false);

Пакетная загрузка

Для каталога используйте пакетную загрузку — она предотвращает N+1 запросов:

php
$allOptions = $optionService->loadOptionsForProducts([1, 2, 3, 4, 5]);
// [
//     1 => ['color' => ['Red'], 'size' => ['L']],
//     2 => ['color' => ['Blue', 'Green']],
//     ...
// ]

Сохранение опций

php
// Сохранить опции (по умолчанию removeOther=true — удалит не указанные)
$optionService->saveProductOptions($productId, [
    'color' => ['Red', 'Blue'],
    'size' => ['L', 'XL'],
    'material' => ['Cotton'],
]);

// Добавить опции без удаления существующих
$optionService->saveProductOptions($productId, [
    'brand' => ['Nike'],
], false);  // removeOther = false

При сохранении значения автоматически очищаются: убираются пробелы, дубликаты и пустые строки.

Доступные ключи опций

php
// Какие опции доступны для товара (на основе его категорий)
$keys = $optionService->getAvailableOptionKeys($productId);
// ['color', 'size', 'material']

Назначение опций категориям

php
// Назначить опцию нескольким категориям
$assigned = $optionService->assignOptionToCategories($optionId, [5, 12, 18]);
// [5, 12, 18] — массив ID категорий, куда опция была назначена

// Назначить с параметрами
$optionService->addOptionToCategory(
    $optionId,
    $categoryId,
    '',       // значение по умолчанию
    true,     // active
    0         // position
);

// Удалить опцию из категории
$optionService->removeOptionFromCategory($optionId, $categoryId);

// Принудительное удаление (удалит значения даже если опция активна в других категориях)
$optionService->removeOptionFromCategory($optionId, $categoryId, true);

Автоназначение товарам

При назначении опции категории она автоматически добавляется ко всем товарам этой категории (включая товары в дополнительных категориях). При удалении — значения сохраняются, если опция активна в другой категории товара.

Доступ к суб-сервисам

Для специализированных задач доступны суб-сервисы напрямую:

php
$loader = $optionService->getLoader();     // OptionLoaderService — чтение
$sync = $optionService->getSync();         // OptionSyncService — запись
$category = $optionService->getCategory(); // OptionCategoryService — категории

Модель msOption

Определение опции хранится в таблице ms3_options.

Поля msOption

ПолеТипПо умолчаниюОписание
keyvarchar(191)''Уникальный ключ (латиница, цифры, дефис, подчёркивание)
captionvarchar(191)''Отображаемое название
descriptiontextnullОписание
measure_unittinytextnullЕдиница измерения
modcategory_idinteger0ID категории MODX (для группировки в админке)
typevarchar(191)''Тип опции (combo-multiple, textfield, numberfield и др.)
propertiesjsonnullДополнительные свойства

Валидация ключа

Ключ опции должен:

  • Содержать только латинские буквы, цифры, дефис, подчёркивание
  • Не начинаться с цифры или пробела
  • Не совпадать с зарезервированными именами полей (id, type, price, weight, image, published и другие поля modResource)

Программная работа

php
use MiniShop3\Model\msOption;

// Создать опцию
$option = $modx->newObject(msOption::class);
$option->set('key', 'material');
$option->set('caption', 'Материал');
$option->set('type', 'combo-multiple');
$option->save();

// Получить опцию
$option = $modx->getObject(msOption::class, ['key' => 'color']);

// Назначить категориям
$assigned = $option->setCategories([5, 12, 18]);

// Получить все опции
$options = $modx->getIterator(msOption::class);

Модель msCategoryOption

Связь опции с категорией. Таблица ms3_category_options.

Поля msCategoryOption

ПолеТипПо умолчаниюОписание
option_idinteger0ID опции (часть PK)
category_idinteger0ID категории (часть PK)
positioninteger0Порядок сортировки
activebooleanfalseАктивна ли опция в категории
requiredbooleanfalseОбязательна ли опция
valuetextnullЗначение по умолчанию

Составной первичный ключ

msCategoryOption использует составной PK из option_id + category_id.

Каскадное поведение

При сохранении msCategoryOption опция автоматически добавляется ко всем товарам категории, у которых её ещё нет.

При удалении msCategoryOption значения msProductOption удаляются только у тех товаров, где опция не активна ни в одной другой категории. Это предотвращает потерю данных, когда товар принадлежит нескольким категориям с одной и той же опцией.

Программная работа

php
use MiniShop3\Model\msCategoryOption;

// Назначить опцию категории
$link = $modx->newObject(msCategoryOption::class);
$link->set('option_id', $optionId);
$link->set('category_id', $categoryId);
$link->set('active', true);
$link->set('position', 0);
$link->save();
// При save() опция автоматически добавится ко всем товарам категории

// Получить опции категории
$categoryOptions = $modx->getIterator(msCategoryOption::class, [
    'category_id' => $categoryId,
    'active' => true,
]);

// Деактивировать опцию в категории
$link = $modx->getObject(msCategoryOption::class, [
    'option_id' => $optionId,
    'category_id' => $categoryId,
]);
$link->set('active', false);
$link->save();

// Удалить опцию из категории (каскадно удалит значения у товаров)
$link->remove();

Модель msProductOption

Значение опции для конкретного товара. Таблица ms3_product_options. Одна запись — одно значение. Для множественных опций (combo-multiple) создаётся несколько записей с одним key.

Поля msProductOption

ПолеТипПо умолчаниюОписание
product_idintegernullID товара
keyvarchar(191)nullКлюч опции
valuetext''Значение

Используйте OptionService

Для работы с опциями товара всегда используйте OptionService. Прямое создание msProductOption обходит логику синхронизации с JSON-полями msProductData (tags, color, size) и не учитывает каскадную логику категорий.

CategoryOptionService

Отдельный сервис для работы с опциями на стороне категории. Используется в админке для построения списка опций категории.

php
use MiniShop3\Model\msCategory;

$categoryOptionService = $modx->services->get('ms3_category_option_service');

// Получить ключи опций категории
$category = $modx->getObject(msCategory::class, $categoryId);
$keys = $categoryOptionService->getOptionKeys($category);
// ['color', 'size', 'material']

// Получить полные конфигурации полей
$fields = $categoryOptionService->getOptionFields($category);

// Очистить кеш опций
$categoryOptionService->clearCache($categoryId);
// Или весь кеш
$categoryOptionService->clearCache();

Суб-сервисы OptionService

OptionLoaderService

Отвечает за все операции чтения опций.

php
$loader = $optionService->getLoader();

// Загрузить опции для одного товара (с метаданными)
$data = $loader->loadForProduct($productId, true);

// Загрузить опции для нескольких товаров (без метаданных)
$data = $loader->loadForProducts([1, 2, 3], false);

// Получить конфигурации полей для админки
$fields = $loader->getFieldsForProduct($productId, $parentId);

// Получить только ключи
$keys = $loader->getOptionKeys($productId, $parentId);

OptionSyncService

Отвечает за запись и синхронизацию значений опций.

php
$sync = $optionService->getSync();

// Сохранить опции
$sync->saveProductOptions($productId, [
    'color' => ['Red', 'Blue'],
], true);

// Прочитать текущие значения
$values = $sync->getForProduct($productId);

// Переименовать ключ опции во всех товарах
$sync->updateOptionKey('old_key', 'new_key');

Переименование ключа

updateOptionKey() выполняет массовый UPDATE всех записей ms3_product_options. Используйте только при переименовании опции в настройках.

OptionCategoryService

Управляет связями опций и категорий.

php
$categoryService = $optionService->getCategory();

// Назначить опцию категориям
$assigned = $categoryService->assignToCategories($optionId, [5, 12]);

// Добавить с параметрами
$categoryService->addToCategory($optionId, $categoryId, '', true, 0);

// Удалить из категории
$categoryService->removeFromCategory($optionId, $categoryId);

// Получить товары в категории
$productIds = $categoryService->getProductsInCategory($categoryId);

// Найти товары, у которых уже есть опция
$withOption = $categoryService->getProductsWithOption($productIds, 'color');

// Массовая вставка опции для товаров
$categoryService->batchInsertOptions($productIds, 'color', 'Red');

// Массовое удаление из категорий
$categoryService->removeFromCategories($optionId, [5, 12, 18]);

REST API

Начиная с v1.10.0-beta1

Legacy-процессоры Processors/Settings/Option/* и Processors/Category/Option/* полностью удалены вместе с ExtJS UI (22 файла, ~2600 строк). Все операции идут через два REST-контроллера: OptionsController и CategoryOptionsController.

Управление опциями — /api/mgr/options/*

Контроллер MiniShop3\Controllers\Api\Manager\OptionsController, permission mssetting_save.

МетодПутьОписание
GET/api/mgr/optionsСписок опций с фильтрами: query, modcategory_id, category_id, categories[]
GET/api/mgr/options/{id}Деталь опции + карта привязанных категорий
POST/api/mgr/optionsСоздать: нормализует key, проверяет уникальность, привязывает к категориям
PUT/api/mgr/options/{id}Partial update; при смене key пересинхронизирует msProductOption через OptionSyncService::updateOptionKey
DELETE/api/mgr/options/{id}Удалить (cascade через lifecycle msOption::removemsCategoryOptionmsProductOption)
DELETE/api/mgr/options/bulkМассовое удаление: ids[]
POST/api/mgr/options/bulk/assignНазначить options[] к categories[]
GET/api/mgr/options/typesСписок типов с локализованными названиями
GET/api/mgr/options/treeДерево категорий MODX (только class_key = msCategory), lazy по parent. Флаг checked для категорий, где опция уже назначена (если передан option_id)
GET/api/mgr/options/modcategoriesПлоский список modCategory для фильтра «Группа»
GET/api/mgr/options/suggestionsУникальные значения msProductOption.value по key для автодополнения comboOptions на карточке товара

Опции категории — /api/mgr/categories/{category_id}/options/*

Контроллер MiniShop3\Controllers\Api\Manager\CategoryOptionsController, permission mscategory_save.

МетодПутьОписание
GET``Опции, привязанные к категории. Каждая строка отдаёт caption (effective), global_caption/global_description + category_caption/category_description (override)
POST``Привязать опцию к категории: option_id, value, active, required, caption, description
PUT/{option_id}Partial update связки: value, active, required, position, caption, description
DELETE/{option_id}Удалить связку. Значения у товаров удаляются только если опция не активна ни в одной другой категории товара
POST/sortСохранить новый порядок (option_ids[])
POST/bulkМассовые действия: activate / deactivate / require / unrequire / remove для option_ids[]
POST/duplicateСкопировать связки из category_from; существующие в текущей категории пропускаются

Schema API — msOptionType::getSchema()

Начиная с v1.10.0-beta1

Каждый класс типа в Controllers/Options/Types/ теперь умеет отдавать декларативное описание своего поля вместо ExtJS JS-строки:

php
abstract class msOptionType
{
    abstract public function getField($field);        // legacy — возвращает JS-строку для ExtJS
    public function getSchema(array $field): array;   // новое — массив для Vue renderer
}

OptionLoaderService::getFieldsForProduct() прикрепляет schema к каждому option_fields ряду рядом с legacy ext_field. Карточка товара на Vue читает schema, legacy ext_field остаётся для обратной совместимости (но в ядре больше нигде не используется).