Настройка отображения наличия товара по точкам самовывоза 1С-Битрикс

Наша компания занимается разработкой, поддержкой и обслуживанием решений на Битрикс и Битрикс24 любой сложности. От простых одностраничных сайтов до сложных интернет магазинов, CRM систем с интеграцией 1С и телефонии. Опыт разработчиков подтвержден сертификатами от вендора.
Предлагаемые услуги
Показано 1 из 1 услугВсе 1626 услуг
Настройка отображения наличия товара по точкам самовывоза 1С-Битрикс
Простая
~1 рабочий день
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1177
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    811
  • image_bitrix-bitrix-24-1c_development_of_an_online_appointment_booking_widget_for_a_medical_center_594_0.webp
    Разработка на базе Битрикс, Битрикс24, 1С для компании Development of an Online Appointment Booking Widget for a Medical Center
    564
  • image_bitrix-bitrix-24-1c_mirsanbel_458_0.webp
    Разработка на базе 1С Предприятие для компании МИРСАНБЕЛ
    747
  • image_crm_dolbimby_434_0.webp
    Разработка сайта на CRM Битрикс24 для компании DOLBIMBY
    655
  • image_crm_technotorgcomplex_453_0.webp
    Разработка на базе Битрикс24 для компании ТЕХНОТОРГКОМПЛЕКС
    976

Настройка отображения наличия товара по точкам самовывоза 1С-Битрикс

На карточке товара нужно показать: «В наличии в 3 магазинах» или конкретно «ТЦ Центральный — 5 шт., ул. Ленина 15 — 2 шт., ТЦ Западный — нет». Стандартный компонент bitrix:catalog.element показывает только общий остаток по всем складам. Разбивка по точкам самовывоза требует отдельного запроса к b_catalog_store_product и доработки шаблона.

Структура данных складских остатков

-- Остатки товара по активным складам
SELECT
    s.ID,
    s.TITLE,
    s.ADDRESS,
    COALESCE(sp.AMOUNT, 0) as AMOUNT,
    COALESCE(sp.QUANTITY_RESERVED, 0) as RESERVED,
    COALESCE(sp.AMOUNT, 0) - COALESCE(sp.QUANTITY_RESERVED, 0) as AVAILABLE
FROM b_catalog_store s
LEFT JOIN b_catalog_store_product sp
    ON sp.STORE_ID = s.ID AND sp.PRODUCT_ID = ?
WHERE s.ACTIVE = 'Y' AND s.IS_SITE = 'Y'  -- только точки самовывоза
ORDER BY s.SORT ASC;

IS_SITE = 'Y' — флаг, что склад является точкой самовывоза для сайта. Устанавливается в Каталог → Склады → [Редактировать].

PHP-код для получения остатков по точкам

function getStoreAvailability(int $productId): array
{
    $result = [];

    $storesQuery = \Bitrix\Catalog\StoreTable::getList([
        'filter' => ['ACTIVE' => 'Y', 'IS_SITE' => 'Y'],
        'select' => ['ID', 'TITLE', 'ADDRESS', 'GPS_N', 'GPS_S', 'SORT'],
        'order'  => ['SORT' => 'ASC'],
    ]);

    $stores = [];
    while ($store = $storesQuery->fetch()) {
        $stores[$store['ID']] = $store;
    }

    if (empty($stores)) {
        return [];
    }

    // Остатки одним запросом для всех складов
    $stockQuery = \Bitrix\Catalog\StoreProductTable::getList([
        'filter' => [
            'PRODUCT_ID' => $productId,
            'STORE_ID'   => array_keys($stores),
        ],
        'select' => ['STORE_ID', 'AMOUNT', 'QUANTITY_RESERVED'],
    ]);

    $stocks = [];
    while ($stock = $stockQuery->fetch()) {
        $stocks[$stock['STORE_ID']] = $stock;
    }

    foreach ($stores as $storeId => $store) {
        $amount   = (float)($stocks[$storeId]['AMOUNT'] ?? 0);
        $reserved = (float)($stocks[$storeId]['QUANTITY_RESERVED'] ?? 0);
        $available = max(0, $amount - $reserved);

        $result[] = [
            'ID'        => $storeId,
            'TITLE'     => $store['TITLE'],
            'ADDRESS'   => $store['ADDRESS'],
            'GPS_N'     => $store['GPS_N'],
            'GPS_S'     => $store['GPS_S'],
            'AMOUNT'    => $amount,
            'AVAILABLE' => $available,
            'IN_STOCK'  => $available > 0,
        ];
    }

    return $result;
}

Интеграция в шаблон компонента карточки товара

В template.php компонента bitrix:catalog.element:

\Bitrix\Main\Loader::includeModule('catalog');

$storeAvailability = getStoreAvailability($arResult['ID']);
$inStockCount = count(array_filter($storeAvailability, fn($s) => $s['IN_STOCK']));
?>

<div class="store-availability">
    <?php if ($inStockCount > 0): ?>
        <div class="in-stock-summary">
            В наличии в <?= $inStockCount ?> магазин<?= \Local\Helpers\Declension::get($inStockCount, ['е', 'е', 'ах']) ?>
        </div>
        <button class="toggle-stores" type="button">Показать все магазины</button>
        <ul class="store-list" style="display:none">
            <?php foreach ($storeAvailability as $store): ?>
            <li class="store-item <?= $store['IN_STOCK'] ? 'in-stock' : 'out-of-stock' ?>">
                <span class="store-name"><?= htmlspecialchars($store['TITLE']) ?></span>
                <span class="store-address"><?= htmlspecialchars($store['ADDRESS']) ?></span>
                <span class="store-qty">
                    <?= $store['IN_STOCK']
                        ? $store['AVAILABLE'] . ' шт.'
                        : 'Нет в наличии' ?>
                </span>
            </li>
            <?php endforeach; ?>
        </ul>
    <?php else: ?>
        <div class="out-of-stock">Нет в наличии в магазинах</div>
    <?php endif; ?>
</div>

AJAX-обновление при выборе торгового предложения

Для товаров с торговыми предложениями (размеры, цвета) остатки нужно обновлять при смене предложения:

document.querySelectorAll('.offer-option').forEach(function(el) {
    el.addEventListener('change', function() {
        var offerId = this.value;
        fetch('/ajax/store-availability/?product_id=' + offerId)
            .then(r => r.json())
            .then(data => updateStoreList(data.stores));
    });
});

AJAX-эндпоинт /ajax/store-availability/ — компонент или отдельный PHP-файл, возвращающий JSON с результатом getStoreAvailability($offerId).

Сроки настройки

Функция получения остатков, доработка шаблона карточки товара с отображением списка складов, AJAX-обновление при смене торгового предложения — 4–8 часов.