Розробка мегаменю із зображеннями товарів 1С-Бітрікс
Текстовий список підкатегорій працює для утилітарного магазину. Для fashion, електроніки, меблів — категорію продає зображення: кавомашина в розділі «Кавомашини», куртка в розділі «Верхній одяг». Мегаменю з превью товарів або зображеннями розділів підвищує клікабельність і знижує навігаційні помилки.
Зображення розділів інфоблока
Розділи інфоблока b_iblock_section мають поля PICTURE і DETAIL_PICTURE. Додатково — користувацьке поле UF_MENU_IMAGE для зображення, спеціально адаптованого під мегаменю (інший кроп, інші розміри).
// При побудові дерева розділів додаємо поля зображень
$res = \CIBlockSection::GetList(
['LEFT_MARGIN' => 'ASC'],
['IBLOCK_ID' => $iblockId, 'ACTIVE' => 'Y', 'DEPTH_LEVEL' => [1, 2]],
false,
['ID', 'NAME', 'CODE', 'SECTION_PAGE_URL', 'DEPTH_LEVEL',
'IBLOCK_SECTION_ID', 'PICTURE', 'UF_MENU_IMAGE']
);
while ($s = $res->GetNext()) {
// Пріоритет: спеціальне зображення меню > стандартна картинка розділу
$imageId = $s['UF_MENU_IMAGE'] ?: $s['PICTURE'];
$s['MENU_IMAGE_SRC'] = $imageId ? \CFile::ResizeImageGet(
$imageId,
['width' => 240, 'height' => 160],
BX_RESIZE_IMAGE_EXACT
)['src'] : null;
$result[$s['ID']] = $s;
}
Зображення популярних товарів розділу
Альтернатива — показувати в мегаменю не зображення розділу, а превью 3–4 популярних товарів. Це живіше та інформативніше.
namespace Local\Menu;
class SectionProductPreviews
{
public static function getForSections(array $sectionIds, int $limit = 4): array
{
if (empty($sectionIds)) return [];
$cache = new \CPHPCache();
$cacheId = 'menu_previews_' . md5(implode(',', $sectionIds));
if ($cache->InitCache(3600, $cacheId, '/megamenu/previews/')) {
return $cache->GetVars()['previews'];
}
$result = [];
foreach ($sectionIds as $sectionId) {
$res = \CIBlockElement::GetList(
['RAND' => 'ASC'], // випадкові, не завжди однакові
[
'IBLOCK_ID' => CATALOG_IBLOCK_ID,
'SECTION_ID' => $sectionId,
'ACTIVE' => 'Y',
'PREVIEW_PICTURE' => ['>', 0], // тільки з картинкою
],
false,
['nPageSize' => $limit, 'iNumPage' => 1],
['ID', 'NAME', 'PREVIEW_PICTURE', 'DETAIL_PAGE_URL']
);
$items = [];
while ($el = $res->GetNext()) {
$items[] = [
'name' => $el['NAME'],
'url' => $el['DETAIL_PAGE_URL'],
'image' => \CFile::ResizeImageGet(
$el['PREVIEW_PICTURE'],
['width' => 120, 'height' => 120],
BX_RESIZE_IMAGE_PROPORTIONAL
)['src'],
];
}
$result[$sectionId] = $items;
}
$cache->StartDataCache();
$cache->EndDataCache(['previews' => $result]);
return $result;
}
}
Шаблон мегаменю із зображеннями
// У шаблоні для кожної категорії першого рівня
foreach ($arResult['MENU'] as $category):
$previews = $productPreviews[$category['ID']] ?? [];
?>
<div class="megamenu__dropdown megamenu__dropdown--with-images">
<div class="megamenu__inner">
<!-- Колонки підкатегорій зліва -->
<div class="megamenu__nav">
<?php foreach ($category['children'] as $sub): ?>
<a href="<?= htmlspecialchars($sub['SECTION_PAGE_URL']) ?>"
class="megamenu__subcat"
data-section="<?= $sub['ID'] ?>">
<?php if ($sub['MENU_IMAGE_SRC']): ?>
<img src="<?= $sub['MENU_IMAGE_SRC'] ?>"
alt="<?= htmlspecialchars($sub['NAME']) ?>"
width="60" height="40" loading="lazy">
<?php endif ?>
<span><?= htmlspecialchars($sub['NAME']) ?></span>
</a>
<?php endforeach ?>
</div>
<!-- Превью товарів справа -->
<?php if (!empty($previews)): ?>
<div class="megamenu__products">
<div class="megamenu__products-label">Популярні товари</div>
<div class="megamenu__products-grid">
<?php foreach ($previews as $product): ?>
<a href="<?= htmlspecialchars($product['url']) ?>"
class="megamenu__product-card">
<img src="<?= htmlspecialchars($product['image']) ?>"
alt="<?= htmlspecialchars($product['name']) ?>"
width="120" height="120" loading="lazy">
<span class="megamenu__product-name">
<?= htmlspecialchars($product['name']) ?>
</span>
</a>
<?php endforeach ?>
</div>
</div>
<?php endif ?>
</div>
</div>
<?php endforeach ?>
CSS для layout із зображеннями
.megamenu__dropdown--with-images .megamenu__inner {
display: grid;
grid-template-columns: 260px 1fr;
gap: 0;
min-height: 300px;
}
.megamenu__nav {
border-right: 1px solid #eee;
padding: 1rem;
display: flex;
flex-direction: column;
gap: 0.25rem;
}
.megamenu__subcat {
display: flex;
align-items: center;
gap: 0.75rem;
padding: 0.5rem 0.75rem;
border-radius: 4px;
transition: background 0.15s;
text-decoration: none;
color: inherit;
}
.megamenu__subcat:hover,
.megamenu__subcat.is-active {
background: #f5f7fa;
}
.megamenu__products-grid {
display: grid;
grid-template-columns: repeat(4, 1fr);
gap: 1rem;
padding: 1rem 1.5rem;
}
.megamenu__product-card {
display: flex;
flex-direction: column;
gap: 0.5rem;
text-decoration: none;
color: inherit;
font-size: 0.8125rem;
}
.megamenu__product-card img {
border-radius: 4px;
object-fit: contain;
background: #f8f8f8;
}
Динамічне підвантаження превью
При наведенні на підкатегорію — підвантажуємо товари цієї підкатегорії через AJAX, не завантажуємо все одразу:
document.querySelectorAll('.megamenu__subcat').forEach(link => {
link.addEventListener('mouseenter', async () => {
const sectionId = link.dataset.section;
if (link.dataset.loaded) return;
const data = await fetch(`/local/ajax/menu-products.php?section=${sectionId}`)
.then(r => r.json());
const grid = link.closest('.megamenu__dropdown').querySelector('.megamenu__products-grid');
grid.innerHTML = data.products.map(p =>
`<a href="${p.url}" class="megamenu__product-card">
<img src="${p.image}" alt="${p.name}" width="120" height="120" loading="lazy">
<span>${p.name}</span>
</a>`
).join('');
link.dataset.loaded = '1';
});
});
Продуктивність та кешування
Завантажувати зображення всіх товарів усіх розділів при першому відкритті сторінки — надлишково. Стратегія:
- Зображення розділів (
UF_MENU_IMAGE) — завантажуються разом із деревом, кешуються 1 годину - Превью товарів — ліниве завантаження по hover через AJAX, кеш 30 хвилин на розділ
- Зображення через
loading="lazy"— браузер не вантажить приховані панелі
Терміни реалізації
| Конфігурація | Термін |
|---|---|
| Мегаменю із зображеннями розділів | 4–5 днів |
| + превью популярних товарів, Ajax-підвантаження | +2–3 дні |
| + анімації, доступність, мобільна версія | +2–3 дні |







