|
Поговорим о...
|
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> * { margin: 0; padding: 0; box-sizing: border-box; }
body { font-family: Arial, sans-serif; background-color: #fff; padding: 10px; font-size: 11px; }
.container { max-width: 1200px; margin: 0 auto; }
h1 { text-align: center; color: #333; font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.search-container { margin: 15px 0; text-align: center; }
.icon-search-input { width: 300px; padding: 8px 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 14px; }
.icon-search-input:focus { outline: none; border-color: #007bff; box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.controls-row { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: #f0f0f0; border-radius: 4px; border: 1px solid #ddd; }
.stats { font-size: 12px; }
.group-select { padding: 6px 10px; border: 1px solid #ddd; border-radius: 4px; font-size: 12px; background: white; }
.show-groups-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; }
.toggle-checkbox { width: 16px; height: 16px; }
.comparison-container { display: flex; flex-wrap: wrap; gap: 15px; align-items: flex-start; justify-content: center; }
.group-section { break-inside: avoid; }
.group-header { background: #f8f9fa; padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: #495057; font-size: 13px; }
.icon-column { border: 1px solid #e0e0e0; border-radius: 5px; padding: 8px; background: #fafafa; width: 420px; /* Увеличили ширину с 380px до 420px */ flex-shrink: 0; }
.column-header { display: grid; grid-template-columns: 32px 20px 32px 1fr; gap: 8px; align-items: center; padding: 5px 3px; margin-bottom: 5px; border-bottom: 1px solid #ddd; font-weight: bold; text-align: center; }
.header-label { font-size: 11px; color: #333; white-space: nowrap; }
.icon-row { display: grid; grid-template-columns: 32px 20px 32px 1fr; gap: 8px; align-items: center; padding: 6px 3px; border-bottom: 1px solid #f0f0f0; min-height: 40px; }
.icon-row:last-child { border-bottom: none; }
.icon-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; }
.bmp-processed { filter: none; }
.arrow { text-align: center; color: #999; font-size: 14px; font-weight: bold; white-space: nowrap; }
.icon-info { padding-left: 8px; padding-right: 5px; min-width: 0; width: 100%; }
.icon-name { color: #333; line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; /* Увеличили шрифт с 11px до 12px */ max-width: 240px; /* Увеличили максимальную ширину */ }
.loading { text-align: center; padding: 20px; color: #666; width: 100%; }
.no-results { text-align: center; padding: 40px; color: #666; width: 100%; font-size: 14px; }
.columns-container { display: flex; flex-wrap: wrap; gap: 15px; width: 100%; justify-content: center; }
.masonry-column { display: flex; flex-direction: column; gap: 15px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .container { max-width: 100%; padding: 0 10px; } .icon-column { width: 100%; max-width: 420px; /* Обновили максимальную ширину */ } .icon-search-input { width: 100%; max-width: 300px; } .controls-row { flex-wrap: wrap; gap: 10px; } .column-header { gap: 4px; padding: 5px 2px; } .header-label { font-size: 10px; } .icon-name { font-size: 11px; /* Чуть меньше на мобильных, но больше чем было */ } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-column { width: 100%; max-width: 100%; } .column-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .header-label { font-size: 9px; } .arrow { font-size: 12px; } .icon-name { font-size: 11px; max-width: 200px; } } </style> </head> <body> <div class="container"> <h1>Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="search-container"> <input type="text" id="iconSearchInput" class="icon-search-input" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="controls-row"> <div class="stats"> Всего иконок для сравнения: <span id="totalCount">0</span> | Отображено: <span id="shownCount">0</span> </div> <select id="groupSelect" class="group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="show-groups-toggle"> <input type="checkbox" id="showGroupsToggle" class="toggle-checkbox" checked> <label for="showGroupsToggle">Показывать группы</label> </div> </div>
<div class="comparison-container" id="iconsContainer"> <div class="loading">Загрузка иконок...</div> </div> </div>
<script> let allIcons = []; let filteredIcons = []; let availableGroups = []; let showGroups = true;
// Функция для разбивки больших групп на части function splitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function calculateGroupHeight(iconsCount) { const headerHeight = 40; const rowHeight = 32; const padding = 16; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function createAdaptiveMasonryLayout(groupsChunks) { const container = document.getElementById('iconsContainer'); const containerWidth = container.clientWidth; const columnWidth = 420 + 15; // Обновили ширину колонки с учетом gap const maxColumns = Math.floor(1200 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = createGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function renderIconsWithGroups() { const container = document.getElementById('iconsContainer'); container.innerHTML = ''; if (filteredIcons.length === 0) { container.innerHTML = '<div class="no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; updateStats(); return; } // Группируем иконки const { groups, noGroup } = groupIcons(filteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = splitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: calculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = splitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: calculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = createAdaptiveMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'columns-container'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); updateStats(); }
// Функция для создания секции группы function createGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'group-section'; const header = document.createElement('div'); header.className = 'group-header'; header.textContent = groupName; section.appendChild(header); const column = createIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function createIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-column'; const header = document.createElement('div'); header.className = 'column-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(createIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function setupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (showGroups && filteredIcons.length > 0) { renderIconsWithGroups(); } }, 250); }); }
// Остальные функции без изменений function processBMPWithPinkBackground(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function loadImageWithBMPProcessing(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await processBMPWithPinkBackground(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function groupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function createIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ loadImageWithBMPProcessing(icon.before, `${icon.name} (до)`), loadImageWithBMPProcessing(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-image' + (beforeResult.processed ? ' bmp-processed' : ''); afterImg.className = 'icon-image' + (afterResult.processed ? ' bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-info'; const name = document.createElement('div'); name.className = 'icon-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function renderIconsWithoutGroups() { const container = document.getElementById('iconsContainer'); container.innerHTML = ''; if (filteredIcons.length === 0) { container.innerHTML = '<div class="no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; updateStats(); return; } const sortedIcons = [...filteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = createIconColumn(columnIcons); container.appendChild(column); } updateStats(); }
function filterIcons(searchTerm, selectedGroup = '') { let filtered = [...allIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } filteredIcons = filtered; renderFilteredIcons(); }
function renderFilteredIcons() { if (showGroups) { renderIconsWithGroups(); } else { renderIconsWithoutGroups(); } }
function updateStats() { document.getElementById('totalCount').textContent = allIcons.length; document.getElementById('shownCount').textContent = filteredIcons.length; }
function initGroupSelect(groups) { const groupSelect = document.getElementById('groupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconSearchInput').value; filterIcons(searchTerm, this.value); }); }
function renderIcons(images, groups) { allIcons = images; filteredIcons = [...allIcons]; availableGroups = groups; initGroupSelect(groups); renderFilteredIcons(); }
async function loadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); renderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка загрузки иконок</div>'; } }
function setupSearch() { const searchInput = document.getElementById('iconSearchInput'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('groupSelect').value; filterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('groupSelect').value; filterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function setupGroupsToggle() { const toggle = document.getElementById('showGroupsToggle'); toggle.addEventListener('change', function() { showGroups = this.checked; renderFilteredIcons(); }); }
window.addEventListener('DOMContentLoaded', function() { loadIcons(); setupSearch(); setupGroupsToggle(); setupResizeHandler(); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> * { margin: 0; padding: 0; box-sizing: border-box; }
body { font-family: Arial, sans-serif; background-color: #fff; padding: 10px; font-size: 11px; }
.container { max-width: 1200px; margin: 0 auto; }
h1 { text-align: center; color: #333; font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.search-container { margin: 15px 0; text-align: center; }
.icon-search-input { width: 300px; padding: 8px 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 14px; }
.icon-search-input:focus { outline: none; border-color: #007bff; box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.controls-row { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: #f0f0f0; border-radius: 4px; border: 1px solid #ddd; }
.stats { font-size: 12px; }
.group-select { padding: 6px 10px; border: 1px solid #ddd; border-radius: 4px; font-size: 12px; background: white; }
.show-groups-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; }
.toggle-checkbox { width: 16px; height: 16px; }
.comparison-container { display: flex; flex-wrap: wrap; gap: 12px; /* Уменьшили gap с 15px до 12px */ align-items: flex-start; justify-content: center; }
.group-section { break-inside: avoid; }
.group-header { background: #f8f9fa; padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: #495057; font-size: 13px; }
.icon-column { border: 1px solid #e0e0e0; border-radius: 5px; padding: 6px; /* Уменьшили padding с 8px до 6px */ background: #fafafa; width: 400px; /* Уменьшили ширину с 420px до 400px */ flex-shrink: 0; }
.column-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; /* Уменьшили gap в заголовке */ gap: 6px; /* Уменьшили gap с 8px до 6px */ align-items: center; padding: 4px 2px; /* Уменьшили padding */ margin-bottom: 4px; /* Уменьшили отступ */ border-bottom: 1px solid #ddd; font-weight: bold; text-align: center; }
.header-label { font-size: 11px; color: #333; white-space: nowrap; }
.icon-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; /* Уменьшили gap */ gap: 6px; /* Уменьшили gap с 8px до 6px */ align-items: center; padding: 4px 2px; /* Уменьшили padding */ border-bottom: 1px solid #f0f0f0; min-height: 36px; /* Уменьшили минимальную высоту */ }
.icon-row:last-child { border-bottom: none; }
.icon-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; }
.bmp-processed { filter: none; }
.arrow { text-align: center; color: #999; font-size: 13px; /* Уменьшили шрифт стрелки */ font-weight: bold; white-space: nowrap; }
.icon-info { padding-left: 6px; /* Уменьшили padding */ padding-right: 3px; /* Уменьшили padding */ min-width: 0; width: 100%; }
.icon-name { color: #333; line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; /* Немного уменьшили максимальную ширину */ }
.loading { text-align: center; padding: 20px; color: #666; width: 100%; }
.no-results { text-align: center; padding: 40px; color: #666; width: 100%; font-size: 14px; }
.columns-container { display: flex; flex-wrap: wrap; gap: 12px; /* Уменьшили gap с 15px до 12px */ width: 100%; justify-content: center; }
.masonry-column { display: flex; flex-direction: column; gap: 12px; /* Уменьшили gap с 15px до 12px */ }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .container { max-width: 100%; padding: 0 10px; } .icon-column { width: 100%; max-width: 400px; } .icon-search-input { width: 100%; max-width: 300px; } .controls-row { flex-wrap: wrap; gap: 10px; } .column-header { gap: 4px; padding: 4px 2px; } .header-label { font-size: 10px; } .icon-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-column { width: 100%; max-width: 100%; } .column-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .header-label { font-size: 9px; } .arrow { font-size: 12px; } .icon-name { font-size: 11px; max-width: 200px; } } </style> </head> <body> <div class="container"> <h1>Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="search-container"> <input type="text" id="iconSearchInput" class="icon-search-input" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="controls-row"> <div class="stats"> Всего иконок для сравнения: <span id="totalCount">0</span> | Отображено: <span id="shownCount">0</span> </div> <select id="groupSelect" class="group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="show-groups-toggle"> <input type="checkbox" id="showGroupsToggle" class="toggle-checkbox" checked> <label for="showGroupsToggle">Показывать группы</label> </div> </div>
<div class="comparison-container" id="iconsContainer"> <div class="loading">Загрузка иконок...</div> </div> </div>
<script> let allIcons = []; let filteredIcons = []; let availableGroups = []; let showGroups = true;
// Функция для разбивки больших групп на части function splitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function calculateGroupHeight(iconsCount) { const headerHeight = 36; // Уменьшили высоту заголовка const rowHeight = 30; // Уменьшили высоту строки const padding = 12; // Уменьшили padding return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function createAdaptiveMasonryLayout(groupsChunks) { const container = document.getElementById('iconsContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; // Обновили ширину колонки с учетом уменьшенного gap const maxColumns = Math.floor(1200 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); console.log(`Container width: ${containerWidth}, Column width: ${columnWidth}, Columns: ${availableColumns}`); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = createGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function renderIconsWithGroups() { const container = document.getElementById('iconsContainer'); container.innerHTML = ''; if (filteredIcons.length === 0) { container.innerHTML = '<div class="no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; updateStats(); return; } // Группируем иконки const { groups, noGroup } = groupIcons(filteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = splitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: calculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = splitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: calculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = createAdaptiveMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'columns-container'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); updateStats(); }
// Функция для создания секции группы function createGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'group-section'; const header = document.createElement('div'); header.className = 'group-header'; header.textContent = groupName; section.appendChild(header); const column = createIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function createIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-column'; const header = document.createElement('div'); header.className = 'column-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(createIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function setupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (showGroups && filteredIcons.length > 0) { renderIconsWithGroups(); } }, 250); }); }
// Остальные функции без изменений function processBMPWithPinkBackground(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function loadImageWithBMPProcessing(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await processBMPWithPinkBackground(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function groupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function createIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ loadImageWithBMPProcessing(icon.before, `${icon.name} (до)`), loadImageWithBMPProcessing(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-image' + (beforeResult.processed ? ' bmp-processed' : ''); afterImg.className = 'icon-image' + (afterResult.processed ? ' bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-info'; const name = document.createElement('div'); name.className = 'icon-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function renderIconsWithoutGroups() { const container = document.getElementById('iconsContainer'); container.innerHTML = ''; if (filteredIcons.length === 0) { container.innerHTML = '<div class="no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; updateStats(); return; } const sortedIcons = [...filteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = createIconColumn(columnIcons); container.appendChild(column); } updateStats(); }
function filterIcons(searchTerm, selectedGroup = '') { let filtered = [...allIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } filteredIcons = filtered; renderFilteredIcons(); }
function renderFilteredIcons() { if (showGroups) { renderIconsWithGroups(); } else { renderIconsWithoutGroups(); } }
function updateStats() { document.getElementById('totalCount').textContent = allIcons.length; document.getElementById('shownCount').textContent = filteredIcons.length; }
function initGroupSelect(groups) { const groupSelect = document.getElementById('groupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconSearchInput').value; filterIcons(searchTerm, this.value); }); }
function renderIcons(images, groups) { allIcons = images; filteredIcons = [...allIcons]; availableGroups = groups; initGroupSelect(groups); renderFilteredIcons(); }
async function loadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); renderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconsContainer').innerHTML = '<div class="loading">Ошибка загрузки иконок</div>'; } }
function setupSearch() { const searchInput = document.getElementById('iconSearchInput'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('groupSelect').value; filterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('groupSelect').value; filterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function setupGroupsToggle() { const toggle = document.getElementById('showGroupsToggle'); toggle.addEventListener('change', function() { showGroups = this.checked; renderFilteredIcons(); }); }
window.addEventListener('DOMContentLoaded', function() { loadIcons(); setupSearch(); setupGroupsToggle(); setupResizeHandler(); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> .icon-comparison-container { margin: 0; padding: 0; box-sizing: border-box; }
.icon-comparison-body { font-family: Arial, sans-serif; background-color: #fff; padding: 10px; font-size: 11px; }
.icon-comparison-wrapper { max-width: 1200px; margin: 0 auto; }
.icon-comparison-title { text-align: center; color: #333; font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.icon-comparison-search-area { margin: 15px 0; text-align: center; }
.icon-comparison-search { width: 300px; padding: 8px 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 14px; }
.icon-comparison-search:focus { outline: none; border-color: #007bff; box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.icon-comparison-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: #f0f0f0; border-radius: 4px; border: 1px solid #ddd; }
.icon-comparison-stats { font-size: 12px; }
.icon-comparison-group-select { padding: 6px 10px; border: 1px solid #ddd; border-radius: 4px; font-size: 12px; background: white; }
.icon-comparison-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; }
.icon-comparison-checkbox { width: 16px; height: 16px; }
.icon-comparison-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.icon-comparison-group { break-inside: avoid; }
.icon-comparison-group-title { background: #f8f9fa; padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: #495057; font-size: 13px; }
.icon-comparison-panel { border: 1px solid #e0e0e0; border-radius: 5px; padding: 6px; background: #fafafa; width: 400px; flex-shrink: 0; }
.icon-comparison-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid #ddd; font-weight: bold; text-align: center; }
.icon-comparison-header-label { font-size: 11px; color: #333; white-space: nowrap; }
.icon-comparison-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid #f0f0f0; min-height: 36px; }
.icon-comparison-row:last-child { border-bottom: none; }
.icon-comparison-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; }
.icon-comparison-bmp-processed { filter: none; }
.icon-comparison-arrow { text-align: center; color: #999; font-size: 13px; font-weight: bold; white-space: nowrap; }
.icon-comparison-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.icon-comparison-name { color: #333; line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.icon-comparison-loading { text-align: center; padding: 20px; color: #666; width: 100%; }
.icon-comparison-no-results { text-align: center; padding: 40px; color: #666; width: 100%; font-size: 14px; }
.icon-comparison-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.icon-comparison-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .icon-comparison-wrapper { max-width: 100%; padding: 0 10px; } .icon-comparison-panel { width: 100%; max-width: 400px; } .icon-comparison-search { width: 100%; max-width: 300px; } .icon-comparison-controls { flex-wrap: wrap; gap: 10px; } .icon-comparison-panel-header { gap: 4px; padding: 4px 2px; } .icon-comparison-header-label { font-size: 10px; } .icon-comparison-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-comparison-panel { width: 100%; max-width: 100%; } .icon-comparison-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .icon-comparison-header-label { font-size: 9px; } .icon-comparison-arrow { font-size: 12px; } .icon-comparison-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1400px) { .icon-comparison-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .icon-comparison-wrapper { max-width: 1600px; } } </style> </head> <body class="icon-comparison-body"> <div class="icon-comparison-wrapper"> <h1 class="icon-comparison-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="icon-comparison-search-area"> <input type="text" id="iconComparisonSearch" class="icon-comparison-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="icon-comparison-controls"> <div class="icon-comparison-stats"> Всего иконок для сравнения: <span id="iconComparisonTotalCount">0</span> | Отображено: <span id="iconComparisonShownCount">0</span> </div> <select id="iconComparisonGroupSelect" class="icon-comparison-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="icon-comparison-toggle"> <input type="checkbox" id="iconComparisonShowGroups" class="icon-comparison-checkbox" checked> <label for="iconComparisonShowGroups">Показывать группы</label> </div> </div>
<div class="icon-comparison-grid" id="iconComparisonContainer"> <div class="icon-comparison-loading">Загрузка иконок...</div> </div> </div>
<script> let iconComparisonAllIcons = []; let iconComparisonFilteredIcons = []; let iconComparisonAvailableGroups = []; let iconComparisonShowGroups = true;
// Функция для разбивки больших групп на части function iconComparisonSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function iconComparisonCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function iconComparisonCreateMasonryLayout(groupsChunks) { const container = document.getElementById('iconComparisonContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); // Увеличили максимальное количество колонок const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); console.log(`Container width: ${containerWidth}, Column width: ${columnWidth}, Columns: ${availableColumns}`); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'icon-comparison-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = iconComparisonCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function iconComparisonRenderWithGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = iconComparisonGroupIcons(iconComparisonFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = iconComparisonSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: iconComparisonCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = iconComparisonSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: iconComparisonCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = iconComparisonCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'icon-comparison-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); iconComparisonUpdateStats(); }
// Функция для создания секции группы function iconComparisonCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'icon-comparison-group'; const header = document.createElement('div'); header.className = 'icon-comparison-group-title'; header.textContent = groupName; section.appendChild(header); const column = iconComparisonCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function iconComparisonCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-comparison-panel'; const header = document.createElement('div'); header.className = 'icon-comparison-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'icon-comparison-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'icon-comparison-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'icon-comparison-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'icon-comparison-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(iconComparisonCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function iconComparisonSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (iconComparisonShowGroups && iconComparisonFilteredIcons.length > 0) { iconComparisonRenderWithGroups(); } }, 250); }); }
// Остальные функции с переименованными названиями function iconComparisonProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function iconComparisonLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await iconComparisonProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function iconComparisonGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function iconComparisonCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-comparison-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ iconComparisonLoadImage(icon.before, `${icon.name} (до)`), iconComparisonLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-comparison-image' + (beforeResult.processed ? ' icon-comparison-bmp-processed' : ''); afterImg.className = 'icon-comparison-image' + (afterResult.processed ? ' icon-comparison-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'icon-comparison-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-comparison-info'; const name = document.createElement('div'); name.className = 'icon-comparison-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function iconComparisonRenderWithoutGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } const sortedIcons = [...iconComparisonFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = iconComparisonCreateIconColumn(columnIcons); container.appendChild(column); } iconComparisonUpdateStats(); }
function iconComparisonFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...iconComparisonAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } iconComparisonFilteredIcons = filtered; iconComparisonRenderFilteredIcons(); }
function iconComparisonRenderFilteredIcons() { if (iconComparisonShowGroups) { iconComparisonRenderWithGroups(); } else { iconComparisonRenderWithoutGroups(); } }
function iconComparisonUpdateStats() { document.getElementById('iconComparisonTotalCount').textContent = iconComparisonAllIcons.length; document.getElementById('iconComparisonShownCount').textContent = iconComparisonFilteredIcons.length; }
function iconComparisonInitGroupSelect(groups) { const groupSelect = document.getElementById('iconComparisonGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconComparisonSearch').value; iconComparisonFilterIcons(searchTerm, this.value); }); }
function iconComparisonRenderIcons(images, groups) { iconComparisonAllIcons = images; iconComparisonFilteredIcons = [...iconComparisonAllIcons]; iconComparisonAvailableGroups = groups; iconComparisonInitGroupSelect(groups); iconComparisonRenderFilteredIcons(); }
async function iconComparisonLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); iconComparisonRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки иконок</div>'; } }
function iconComparisonSetupSearch() { const searchInput = document.getElementById('iconComparisonSearch'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function iconComparisonSetupToggle() { const toggle = document.getElementById('iconComparisonShowGroups'); toggle.addEventListener('change', function() { iconComparisonShowGroups = this.checked; iconComparisonRenderFilteredIcons(); }); }
window.addEventListener('DOMContentLoaded', function() { iconComparisonLoadIcons(); iconComparisonSetupSearch(); iconComparisonSetupToggle(); iconComparisonSetupResizeHandler(); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> .icon-comparison-container { margin: 0; padding: 0; box-sizing: border-box; }
.icon-comparison-body { font-family: Arial, sans-serif; background-color: #fff; padding: 10px; font-size: 11px; transition: background-color 0.3s ease, color 0.3s ease; }
/* Темная тема */ .icon-comparison-body.dark-theme { background-color: #1a1a1a; color: #e0e0e0; }
.icon-comparison-wrapper { max-width: 1200px; margin: 0 auto; }
.icon-comparison-title { text-align: center; color: #333; font-size: 16px; margin-bottom: 15px; font-weight: bold; transition: color 0.3s ease; }
.dark-theme .icon-comparison-title { color: #e0e0e0; }
.icon-comparison-search-area { margin: 15px 0; text-align: center; }
.icon-comparison-search { width: 300px; padding: 8px 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 14px; background-color: white; color: #333; transition: all 0.3s ease; }
.dark-theme .icon-comparison-search { background-color: #2d2d2d; border-color: #555; color: #e0e0e0; }
.icon-comparison-search:focus { outline: none; border-color: #007bff; box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.dark-theme .icon-comparison-search:focus { border-color: #4dabf7; box-shadow: 0 0 0 2px rgba(77, 171, 247, 0.25); }
.icon-comparison-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: #f0f0f0; border-radius: 4px; border: 1px solid #ddd; transition: all 0.3s ease; }
.dark-theme .icon-comparison-controls { background: #2d2d2d; border-color: #444; }
.icon-comparison-stats { font-size: 12px; transition: color 0.3s ease; }
.dark-theme .icon-comparison-stats { color: #e0e0e0; }
.icon-comparison-group-select { padding: 6px 10px; border: 1px solid #ddd; border-radius: 4px; font-size: 12px; background: white; color: #333; transition: all 0.3s ease; }
.dark-theme .icon-comparison-group-select { background: #2d2d2d; border-color: #555; color: #e0e0e0; }
.icon-comparison-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; transition: color 0.3s ease; }
.dark-theme .icon-comparison-toggle { color: #e0e0e0; }
.icon-comparison-checkbox { width: 16px; height: 16px; }
.icon-comparison-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.icon-comparison-group { break-inside: avoid; }
.icon-comparison-group-title { background: #f8f9fa; padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: #495057; font-size: 13px; transition: all 0.3s ease; }
.dark-theme .icon-comparison-group-title { background: #2d2d2d; color: #e0e0e0; }
.icon-comparison-panel { border: 1px solid #e0e0e0; border-radius: 5px; padding: 6px; background: #fafafa; width: 400px; flex-shrink: 0; transition: all 0.3s ease; }
.dark-theme .icon-comparison-panel { background: #2d2d2d; border-color: #444; }
.icon-comparison-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid #ddd; font-weight: bold; text-align: center; transition: all 0.3s ease; }
.dark-theme .icon-comparison-panel-header { border-bottom-color: #555; }
.icon-comparison-header-label { font-size: 11px; color: #333; white-space: nowrap; transition: color 0.3s ease; }
.dark-theme .icon-comparison-header-label { color: #e0e0e0; }
.icon-comparison-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid #f0f0f0; min-height: 36px; transition: all 0.3s ease; }
.dark-theme .icon-comparison-row { border-bottom-color: #444; }
.icon-comparison-row:last-child { border-bottom: none; }
.icon-comparison-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: transparent; /* Для темных иконок на светлом фоне и наоборот */ mix-blend-mode: multiply; }
.dark-theme .icon-comparison-image { /* Инвертируем только темные иконки, светлые остаются как есть */ filter: brightness(1.2); mix-blend-mode: screen; }
/* Специальная обработка для обработанных BMP */ .icon-comparison-bmp-processed { filter: none; }
.dark-theme .icon-comparison-bmp-processed { /* Для обработанных BMP сохраняем оригинальный вид */ filter: none; mix-blend-mode: normal; }
.icon-comparison-arrow { text-align: center; color: #999; font-size: 13px; font-weight: bold; white-space: nowrap; transition: color 0.3s ease; }
.dark-theme .icon-comparison-arrow { color: #777; }
.icon-comparison-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.icon-comparison-name { color: #333; line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; transition: color 0.3s ease; }
.dark-theme .icon-comparison-name { color: #e0e0e0; }
.icon-comparison-loading { text-align: center; padding: 20px; color: #666; width: 100%; transition: color 0.3s ease; }
.dark-theme .icon-comparison-loading { color: #999; }
.icon-comparison-no-results { text-align: center; padding: 40px; color: #666; width: 100%; font-size: 14px; transition: color 0.3s ease; }
.dark-theme .icon-comparison-no-results { color: #999; }
.icon-comparison-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.icon-comparison-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Индикатор темы (для отладки) */ .icon-comparison-theme-indicator { position: fixed; bottom: 10px; right: 10px; padding: 5px 10px; background: #007bff; color: white; border-radius: 3px; font-size: 10px; z-index: 1000; opacity: 0.7; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .icon-comparison-wrapper { max-width: 100%; padding: 0 10px; } .icon-comparison-panel { width: 100%; max-width: 400px; } .icon-comparison-search { width: 100%; max-width: 300px; } .icon-comparison-controls { flex-wrap: wrap; gap: 10px; } .icon-comparison-panel-header { gap: 4px; padding: 4px 2px; } .icon-comparison-header-label { font-size: 10px; } .icon-comparison-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-comparison-panel { width: 100%; max-width: 100%; } .icon-comparison-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .icon-comparison-header-label { font-size: 9px; } .icon-comparison-arrow { font-size: 12px; } .icon-comparison-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1400px) { .icon-comparison-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .icon-comparison-wrapper { max-width: 1600px; } } </style> </head> <body class="icon-comparison-body"> <div class="icon-comparison-wrapper"> <h1 class="icon-comparison-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="icon-comparison-search-area"> <input type="text" id="iconComparisonSearch" class="icon-comparison-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="icon-comparison-controls"> <div class="icon-comparison-stats"> Всего иконок для сравнения: <span id="iconComparisonTotalCount">0</span> | Отображено: <span id="iconComparisonShownCount">0</span> </div> <select id="iconComparisonGroupSelect" class="icon-comparison-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="icon-comparison-toggle"> <input type="checkbox" id="iconComparisonShowGroups" class="icon-comparison-checkbox" checked> <label for="iconComparisonShowGroups">Показывать группы</label> </div> </div>
<div class="icon-comparison-grid" id="iconComparisonContainer"> <div class="icon-comparison-loading">Загрузка иконок...</div> </div> </div>
<!-- Индикатор темы (можно удалить в продакшене) --> <div id="iconComparisonThemeIndicator" class="icon-comparison-theme-indicator" style="display: none;"> Тема: <span id="iconComparisonThemeStatus">светлая</span> </div>
<script> let iconComparisonAllIcons = []; let iconComparisonFilteredIcons = []; let iconComparisonAvailableGroups = []; let iconComparisonShowGroups = true; let iconComparisonIsDarkTheme = false;
// Функция для проверки и применения темы Hugo function iconComparisonDetectAndApplyTheme() { const body = document.querySelector('.icon-comparison-body'); const indicator = document.getElementById('iconComparisonThemeIndicator'); // Проверяем наличие класса dark-theme на корневом элементе Hugo const htmlElement = document.documentElement; const hugoBody = document.body; // Различные способы определения темной темы в Hugo const isDarkByClass = htmlElement.classList.contains('dark') || hugoBody.classList.contains('dark') || htmlElement.classList.contains('dark-mode') || hugoBody.classList.contains('dark-mode') || htmlElement.classList.contains('theme-dark') || hugoBody.classList.contains('theme-dark'); // Проверяем атрибуты data-theme const isDarkByAttr = htmlElement.getAttribute('data-theme') === 'dark' || hugoBody.getAttribute('data-theme') === 'dark'; // Проверяем CSS переменные const computedStyle = window.getComputedStyle(htmlElement); const bgColor = computedStyle.backgroundColor; const rgb = bgColor.match(/\d+/g); if (rgb) { const brightness = (parseInt(rgb[0]) * 299 + parseInt(rgb[1]) * 587 + parseInt(rgb[2]) * 114) / 1000; const isDarkByColor = brightness < 128; if (isDarkByColor) { iconComparisonIsDarkTheme = true; } } // Применяем тему если нашли признаки темной темы if (isDarkByClass || isDarkByAttr || iconComparisonIsDarkTheme) { body.classList.add('dark-theme'); iconComparisonIsDarkTheme = true; // Обновляем индикатор if (indicator) { indicator.querySelector('#iconComparisonThemeStatus').textContent = 'тёмная'; indicator.style.display = 'block'; } } else { body.classList.remove('dark-theme'); iconComparisonIsDarkTheme = false; // Обновляем индикатор if (indicator) { indicator.querySelector('#iconComparisonThemeStatus').textContent = 'светлая'; indicator.style.display = 'block'; } } }
// Наблюдатель за изменениями темы function iconComparisonSetupThemeObserver() { // Создаем наблюдатель за изменениями атрибутов const observer = new MutationObserver((mutations) => { mutations.forEach((mutation) => { if (mutation.type === 'attributes' && (mutation.attributeName === 'class' || mutation.attributeName === 'data-theme')) { iconComparisonDetectAndApplyTheme(); } }); });
// Начинаем наблюдение за корневым элементом и body observer.observe(document.documentElement, { attributes: true, attributeFilter: ['class', 'data-theme'] }); observer.observe(document.body, { attributes: true, attributeFilter: ['class', 'data-theme'] });
// Также слушаем изменения CSS переменных window.addEventListener('storage', iconComparisonDetectAndApplyTheme); // Проверяем тему при загрузке setTimeout(iconComparisonDetectAndApplyTheme, 100); // И периодически проверяем (на случай динамического переключения) setInterval(iconComparisonDetectAndApplyTheme, 1000); }
// Функция для разбивки больших групп на части function iconComparisonSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function iconComparisonCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function iconComparisonCreateMasonryLayout(groupsChunks) { const container = document.getElementById('iconComparisonContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'icon-comparison-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = iconComparisonCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function iconComparisonRenderWithGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = iconComparisonGroupIcons(iconComparisonFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = iconComparisonSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: iconComparisonCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = iconComparisonSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: iconComparisonCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = iconComparisonCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'icon-comparison-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); iconComparisonUpdateStats(); }
// Функция для создания секции группы function iconComparisonCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'icon-comparison-group'; const header = document.createElement('div'); header.className = 'icon-comparison-group-title'; header.textContent = groupName; section.appendChild(header); const column = iconComparisonCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function iconComparisonCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-comparison-panel'; const header = document.createElement('div'); header.className = 'icon-comparison-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'icon-comparison-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'icon-comparison-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'icon-comparison-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'icon-comparison-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(iconComparisonCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function iconComparisonSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (iconComparisonShowGroups && iconComparisonFilteredIcons.length > 0) { iconComparisonRenderWithGroups(); } }, 250); }); }
// Остальные функции function iconComparisonProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function iconComparisonLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await iconComparisonProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function iconComparisonGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function iconComparisonCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-comparison-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ iconComparisonLoadImage(icon.before, `${icon.name} (до)`), iconComparisonLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-comparison-image' + (beforeResult.processed ? ' icon-comparison-bmp-processed' : ''); afterImg.className = 'icon-comparison-image' + (afterResult.processed ? ' icon-comparison-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'icon-comparison-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-comparison-info'; const name = document.createElement('div'); name.className = 'icon-comparison-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function iconComparisonRenderWithoutGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } const sortedIcons = [...iconComparisonFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = iconComparisonCreateIconColumn(columnIcons); container.appendChild(column); } iconComparisonUpdateStats(); }
function iconComparisonFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...iconComparisonAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } iconComparisonFilteredIcons = filtered; iconComparisonRenderFilteredIcons(); }
function iconComparisonRenderFilteredIcons() { if (iconComparisonShowGroups) { iconComparisonRenderWithGroups(); } else { iconComparisonRenderWithoutGroups(); } }
function iconComparisonUpdateStats() { document.getElementById('iconComparisonTotalCount').textContent = iconComparisonAllIcons.length; document.getElementById('iconComparisonShownCount').textContent = iconComparisonFilteredIcons.length; }
function iconComparisonInitGroupSelect(groups) { const groupSelect = document.getElementById('iconComparisonGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconComparisonSearch').value; iconComparisonFilterIcons(searchTerm, this.value); }); }
function iconComparisonRenderIcons(images, groups) { iconComparisonAllIcons = images; iconComparisonFilteredIcons = [...iconComparisonAllIcons]; iconComparisonAvailableGroups = groups; iconComparisonInitGroupSelect(groups); iconComparisonRenderFilteredIcons(); }
async function iconComparisonLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); iconComparisonRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки иконок</div>'; } }
function iconComparisonSetupSearch() { const searchInput = document.getElementById('iconComparisonSearch'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function iconComparisonSetupToggle() { const toggle = document.getElementById('iconComparisonShowGroups'); toggle.addEventListener('change', function() { iconComparisonShowGroups = this.checked; iconComparisonRenderFilteredIcons(); }); }
window.addEventListener('DOMContentLoaded', function() { // Сначала настраиваем определение темы iconComparisonSetupThemeObserver(); // Затем загружаем остальное iconComparisonLoadIcons(); iconComparisonSetupSearch(); iconComparisonSetupToggle(); iconComparisonSetupResizeHandler(); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> .icon-comparison-container { margin: 0; padding: 0; box-sizing: border-box; }
.icon-comparison-body { font-family: Arial, sans-serif; background-color: #fff; padding: 10px; font-size: 11px; }
/* Темная тема - упрощенная версия */ .icon-comparison-body.dark-mode { background-color: #1a1a1a; color: #e0e0e0; }
.dark-mode .icon-comparison-title { color: #e0e0e0; }
.dark-mode .icon-comparison-controls { background: #2d2d2d; border-color: #444; }
.dark-mode .icon-comparison-stats { color: #e0e0e0; }
.dark-mode .icon-comparison-search { background-color: #2d2d2d; border-color: #555; color: #e0e0e0; }
.dark-mode .icon-comparison-group-select { background: #2d2d2d; border-color: #555; color: #e0e0e0; }
.dark-mode .icon-comparison-toggle { color: #e0e0e0; }
.dark-mode .icon-comparison-group-title { background: #2d2d2d; color: #e0e0e0; }
.dark-mode .icon-comparison-panel { background: #2d2d2d; border-color: #444; }
.dark-mode .icon-comparison-panel-header { border-bottom-color: #555; }
.dark-mode .icon-comparison-header-label { color: #e0e0e0; }
.dark-mode .icon-comparison-row { border-bottom-color: #444; }
.dark-mode .icon-comparison-arrow { color: #777; }
.dark-mode .icon-comparison-name { color: #e0e0e0; }
.dark-mode .icon-comparison-loading { color: #999; }
.dark-mode .icon-comparison-no-results { color: #999; }
/* Основные стили */ .icon-comparison-wrapper { max-width: 1200px; margin: 0 auto; }
.icon-comparison-title { text-align: center; color: #333; font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.icon-comparison-search-area { margin: 15px 0; text-align: center; }
.icon-comparison-search { width: 300px; padding: 8px 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 14px; background-color: white; color: #333; }
.icon-comparison-search:focus { outline: none; border-color: #007bff; box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.icon-comparison-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: #f0f0f0; border-radius: 4px; border: 1px solid #ddd; }
.icon-comparison-stats { font-size: 12px; }
.icon-comparison-group-select { padding: 6px 10px; border: 1px solid #ddd; border-radius: 4px; font-size: 12px; background: white; color: #333; }
.icon-comparison-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; }
.icon-comparison-checkbox { width: 16px; height: 16px; }
.icon-comparison-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.icon-comparison-group { break-inside: avoid; }
.icon-comparison-group-title { background: #f8f9fa; padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: #495057; font-size: 13px; }
.icon-comparison-panel { border: 1px solid #e0e0e0; border-radius: 5px; padding: 6px; background: #fafafa; width: 400px; flex-shrink: 0; }
.icon-comparison-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid #ddd; font-weight: bold; text-align: center; }
.icon-comparison-header-label { font-size: 11px; color: #333; white-space: nowrap; }
.icon-comparison-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid #f0f0f0; min-height: 36px; }
.icon-comparison-row:last-child { border-bottom: none; }
.icon-comparison-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; }
.icon-comparison-bmp-processed { filter: none; }
.icon-comparison-arrow { text-align: center; color: #999; font-size: 13px; font-weight: bold; white-space: nowrap; }
.icon-comparison-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.icon-comparison-name { color: #333; line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.icon-comparison-loading { text-align: center; padding: 20px; color: #666; width: 100%; }
.icon-comparison-no-results { text-align: center; padding: 40px; color: #666; width: 100%; font-size: 14px; }
.icon-comparison-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.icon-comparison-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность */ @media (max-width: 768px) { .icon-comparison-wrapper { max-width: 100%; padding: 0 10px; } .icon-comparison-panel { width: 100%; max-width: 400px; } .icon-comparison-search { width: 100%; max-width: 300px; } .icon-comparison-controls { flex-wrap: wrap; gap: 10px; } .icon-comparison-panel-header { gap: 4px; padding: 4px 2px; } .icon-comparison-header-label { font-size: 10px; } .icon-comparison-name { font-size: 11px; } }
@media (max-width: 480px) { .icon-comparison-panel { width: 100%; max-width: 100%; } .icon-comparison-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .icon-comparison-header-label { font-size: 9px; } .icon-comparison-arrow { font-size: 12px; } .icon-comparison-name { font-size: 11px; max-width: 200px; } } </style> </head> <body class="icon-comparison-body"> <div class="icon-comparison-wrapper"> <h1 class="icon-comparison-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="icon-comparison-search-area"> <input type="text" id="iconComparisonSearch" class="icon-comparison-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="icon-comparison-controls"> <div class="icon-comparison-stats"> Всего иконок для сравнения: <span id="iconComparisonTotalCount">0</span> | Отображено: <span id="iconComparisonShownCount">0</span> </div> <select id="iconComparisonGroupSelect" class="icon-comparison-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="icon-comparison-toggle"> <input type="checkbox" id="iconComparisonShowGroups" class="icon-comparison-checkbox" checked> <label for="iconComparisonShowGroups">Показывать группы</label> </div> </div>
<div class="icon-comparison-grid" id="iconComparisonContainer"> <div class="icon-comparison-loading">Загрузка иконок...</div> </div> </div>
<script> let iconComparisonAllIcons = []; let iconComparisonFilteredIcons = []; let iconComparisonAvailableGroups = []; let iconComparisonShowGroups = true;
// Простая функция для проверки темной темы function iconComparisonCheckDarkTheme() { const body = document.querySelector('.icon-comparison-body'); // Проверяем несколько способов определения темной темы if (document.documentElement.classList.contains('dark') || document.body.classList.contains('dark') || document.documentElement.classList.contains('dark-mode') || document.body.classList.contains('dark-mode') || document.documentElement.getAttribute('data-theme') === 'dark' || document.body.getAttribute('data-theme') === 'dark') { body.classList.add('dark-mode'); return true; } // Проверяем по цвету фона const computedStyle = window.getComputedStyle(document.documentElement); const bgColor = computedStyle.backgroundColor; if (bgColor) { const rgb = bgColor.match(/\d+/g); if (rgb) { const brightness = (parseInt(rgb[0]) * 299 + parseInt(rgb[1]) * 587 + parseInt(rgb[2]) * 114) / 1000; if (brightness < 128) { body.classList.add('dark-mode'); return true; } } } // Если не темная тема - убираем класс body.classList.remove('dark-mode'); return false; }
// Функция для разбивки больших групп на части function iconComparisonSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function iconComparisonCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function iconComparisonCreateMasonryLayout(groupsChunks) { const container = document.getElementById('iconComparisonContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'icon-comparison-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = iconComparisonCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой function iconComparisonRenderWithGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = iconComparisonGroupIcons(iconComparisonFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = iconComparisonSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: iconComparisonCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = iconComparisonSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: iconComparisonCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте allGroupChunks.sort((a, b) => b.height - a.height); // Создаем masonry компоновку const columns = iconComparisonCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'icon-comparison-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); iconComparisonUpdateStats(); }
// Функция для создания секции группы function iconComparisonCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'icon-comparison-group'; const header = document.createElement('div'); header.className = 'icon-comparison-group-title'; header.textContent = groupName; section.appendChild(header); const column = iconComparisonCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function iconComparisonCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-comparison-panel'; const header = document.createElement('div'); header.className = 'icon-comparison-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'icon-comparison-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'icon-comparison-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'icon-comparison-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'icon-comparison-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(iconComparisonCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function iconComparisonSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (iconComparisonShowGroups && iconComparisonFilteredIcons.length > 0) { iconComparisonRenderWithGroups(); } }, 250); }); }
// Обработка BMP function iconComparisonProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function iconComparisonLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await iconComparisonProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function iconComparisonGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function iconComparisonCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-comparison-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ iconComparisonLoadImage(icon.before, `${icon.name} (до)`), iconComparisonLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-comparison-image' + (beforeResult.processed ? ' icon-comparison-bmp-processed' : ''); afterImg.className = 'icon-comparison-image' + (afterResult.processed ? ' icon-comparison-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'icon-comparison-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-comparison-info'; const name = document.createElement('div'); name.className = 'icon-comparison-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function iconComparisonRenderWithoutGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } const sortedIcons = [...iconComparisonFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = iconComparisonCreateIconColumn(columnIcons); container.appendChild(column); } iconComparisonUpdateStats(); }
function iconComparisonFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...iconComparisonAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } iconComparisonFilteredIcons = filtered; iconComparisonRenderFilteredIcons(); }
function iconComparisonRenderFilteredIcons() { if (iconComparisonShowGroups) { iconComparisonRenderWithGroups(); } else { iconComparisonRenderWithoutGroups(); } }
function iconComparisonUpdateStats() { document.getElementById('iconComparisonTotalCount').textContent = iconComparisonAllIcons.length; document.getElementById('iconComparisonShownCount').textContent = iconComparisonFilteredIcons.length; }
function iconComparisonInitGroupSelect(groups) { const groupSelect = document.getElementById('iconComparisonGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconComparisonSearch').value; iconComparisonFilterIcons(searchTerm, this.value); }); }
function iconComparisonRenderIcons(images, groups) { iconComparisonAllIcons = images; iconComparisonFilteredIcons = [...iconComparisonAllIcons]; iconComparisonAvailableGroups = groups; iconComparisonInitGroupSelect(groups); iconComparisonRenderFilteredIcons(); }
async function iconComparisonLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); iconComparisonRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки иконок</div>'; } }
function iconComparisonSetupSearch() { const searchInput = document.getElementById('iconComparisonSearch'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function iconComparisonSetupToggle() { const toggle = document.getElementById('iconComparisonShowGroups'); toggle.addEventListener('change', function() { iconComparisonShowGroups = this.checked; iconComparisonRenderFilteredIcons(); }); }
window.addEventListener('DOMContentLoaded', function() { // Простая проверка темы при загрузке setTimeout(iconComparisonCheckDarkTheme, 100); // Загружаем основной функционал iconComparisonLoadIcons(); iconComparisonSetupSearch(); iconComparisonSetupToggle(); iconComparisonSetupResizeHandler(); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Наши стили полностью изолированы */ .icon-comparison-page { margin: 0; padding: 0; box-sizing: border-box; font-family: Arial, sans-serif; background-color: #fff; padding: 10px; font-size: 11px; color: #333; /* Изолируем наши стили от Hugo */ all: initial; font-family: Arial, sans-serif !important; }
.icon-comparison-page * { box-sizing: border-box; }
/* Светлая тема по умолчанию */ .icon-comparison-page[data-theme="light"] { background-color: #fff; color: #333; }
/* Темная тема */ .icon-comparison-page[data-theme="dark"] { background-color: #1a1a1a; color: #e0e0e0; }
.icon-comparison-wrapper { max-width: 1200px; margin: 0 auto; }
.icon-comparison-title { text-align: center; color: #333; font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-title { color: #e0e0e0; }
.icon-comparison-search-area { margin: 15px 0; text-align: center; }
.icon-comparison-search { width: 300px; padding: 8px 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 14px; background-color: white; color: #333; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-search { background-color: #2d2d2d; border-color: #555; color: #e0e0e0; }
.icon-comparison-search:focus { outline: none; border-color: #007bff; box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.icon-comparison-page[data-theme="dark"] .icon-comparison-search:focus { border-color: #4dabf7; box-shadow: 0 0 0 2px rgba(77, 171, 247, 0.25); }
.icon-comparison-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: #f0f0f0; border-radius: 4px; border: 1px solid #ddd; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-controls { background: #2d2d2d; border-color: #444; }
.icon-comparison-stats { font-size: 12px; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-stats { color: #e0e0e0; }
.icon-comparison-group-select { padding: 6px 10px; border: 1px solid #ddd; border-radius: 4px; font-size: 12px; background: white; color: #333; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-group-select { background: #2d2d2d; border-color: #555; color: #e0e0e0; }
.icon-comparison-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-toggle { color: #e0e0e0; }
.icon-comparison-checkbox { width: 16px; height: 16px; }
.icon-comparison-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.icon-comparison-group { break-inside: avoid; }
.icon-comparison-group-title { background: #f8f9fa; padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: #495057; font-size: 13px; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-group-title { background: #2d2d2d; color: #e0e0e0; }
.icon-comparison-panel { border: 1px solid #e0e0e0; border-radius: 5px; padding: 6px; background: #fafafa; width: 400px; flex-shrink: 0; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-panel { background: #2d2d2d; border-color: #444; }
.icon-comparison-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid #ddd; font-weight: bold; text-align: center; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-panel-header { border-bottom-color: #555; }
.icon-comparison-header-label { font-size: 11px; color: #333; white-space: nowrap; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-header-label { color: #e0e0e0; }
.icon-comparison-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid #f0f0f0; min-height: 36px; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-row { border-bottom-color: #444; }
.icon-comparison-row:last-child { border-bottom: none; }
.icon-comparison-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; /* Для темных иконок на светлом фоне */ background-color: transparent; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-image { /* Небольшая коррекция для темной темы */ filter: brightness(1.1); }
.icon-comparison-bmp-processed { filter: none; }
.icon-comparison-arrow { text-align: center; color: #999; font-size: 13px; font-weight: bold; white-space: nowrap; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-arrow { color: #777; }
.icon-comparison-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.icon-comparison-name { color: #333; line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-name { color: #e0e0e0; }
.icon-comparison-loading { text-align: center; padding: 20px; color: #666; width: 100%; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-loading { color: #999; }
.icon-comparison-no-results { text-align: center; padding: 40px; color: #666; width: 100%; font-size: 14px; }
.icon-comparison-page[data-theme="dark"] .icon-comparison-no-results { color: #999; }
.icon-comparison-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.icon-comparison-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .icon-comparison-wrapper { max-width: 100%; padding: 0 10px; } .icon-comparison-panel { width: 100%; max-width: 400px; } .icon-comparison-search { width: 100%; max-width: 300px; } .icon-comparison-controls { flex-wrap: wrap; gap: 10px; } .icon-comparison-panel-header { gap: 4px; padding: 4px 2px; } .icon-comparison-header-label { font-size: 10px; } .icon-comparison-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-comparison-panel { width: 100%; max-width: 100%; } .icon-comparison-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .icon-comparison-header-label { font-size: 9px; } .icon-comparison-arrow { font-size: 12px; } .icon-comparison-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1400px) { .icon-comparison-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .icon-comparison-wrapper { max-width: 1600px; } } </style> </head> <body class="icon-comparison-page" data-theme="light"> <div class="icon-comparison-wrapper"> <h1 class="icon-comparison-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="icon-comparison-search-area"> <input type="text" id="iconComparisonSearch" class="icon-comparison-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="icon-comparison-controls"> <div class="icon-comparison-stats"> Всего иконок для сравнения: <span id="iconComparisonTotalCount">0</span> | Отображено: <span id="iconComparisonShownCount">0</span> </div> <select id="iconComparisonGroupSelect" class="icon-comparison-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="icon-comparison-toggle"> <input type="checkbox" id="iconComparisonShowGroups" class="icon-comparison-checkbox" checked> <label for="iconComparisonShowGroups">Показывать группы</label> </div> </div>
<div class="icon-comparison-grid" id="iconComparisonContainer"> <div class="icon-comparison-loading">Загрузка иконок...</div> </div> </div>
<script> // Глобальные переменные (изолированные) window.iconComparisonData = { allIcons: [], filteredIcons: [], availableGroups: [], showGroups: true, isDarkTheme: false };
// Простая функция определения темы function detectTheme() { const pageElement = document.querySelector('.icon-comparison-page'); // Если страница открыта отдельно - всегда светлая тема if (!window.parent || window.parent === window) { pageElement.setAttribute('data-theme', 'light'); return; } try { // Пробуем определить тему из родительского окна (Hugo) const parentHtml = window.parent.document.documentElement; const parentBody = window.parent.document.body; // Проверяем различные признаки темной темы const isDark = parentHtml.classList.contains('dark') || parentHtml.classList.contains('dark-mode') || parentHtml.classList.contains('theme-dark') || parentBody.classList.contains('dark') || parentBody.classList.contains('dark-mode') || parentBody.classList.contains('theme-dark') || parentHtml.getAttribute('data-theme') === 'dark' || parentBody.getAttribute('data-theme') === 'dark' || window.parent.localStorage.getItem('theme') === 'dark' || window.parent.localStorage.getItem('color-scheme') === 'dark'; if (isDark) { pageElement.setAttribute('data-theme', 'dark'); window.iconComparisonData.isDarkTheme = true; } else { pageElement.setAttribute('data-theme', 'light'); window.iconComparisonData.isDarkTheme = false; } } catch (error) { // Если нет доступа к родительскому окну (CORS) или ошибка pageElement.setAttribute('data-theme', 'light'); window.iconComparisonData.isDarkTheme = false; } }
// Функция для разбивки больших групп на части function splitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function calculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function createMasonryLayout(groupsChunks) { const container = document.getElementById('iconComparisonContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'icon-comparison-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = createGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function renderWithGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (window.iconComparisonData.filteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; updateStats(); return; } // Группируем иконки const { groups, noGroup } = groupIcons(window.iconComparisonData.filteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = splitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: calculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = splitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: calculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = createMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'icon-comparison-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); updateStats(); }
// Функция для создания секции группы function createGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'icon-comparison-group'; const header = document.createElement('div'); header.className = 'icon-comparison-group-title'; header.textContent = groupName; section.appendChild(header); const column = createIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function createIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-comparison-panel'; const header = document.createElement('div'); header.className = 'icon-comparison-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'icon-comparison-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'icon-comparison-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'icon-comparison-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'icon-comparison-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(createIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function setupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (window.iconComparisonData.showGroups && window.iconComparisonData.filteredIcons.length > 0) { renderWithGroups(); } }, 250); }); }
// Остальные функции function processBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function loadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await processBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function groupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function createIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-comparison-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ loadImage(icon.before, `${icon.name} (до)`), loadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-comparison-image' + (beforeResult.processed ? ' icon-comparison-bmp-processed' : ''); afterImg.className = 'icon-comparison-image' + (afterResult.processed ? ' icon-comparison-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'icon-comparison-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-comparison-info'; const name = document.createElement('div'); name.className = 'icon-comparison-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function renderWithoutGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (window.iconComparisonData.filteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; updateStats(); return; } const sortedIcons = [...window.iconComparisonData.filteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = createIconColumn(columnIcons); container.appendChild(column); } updateStats(); }
function filterIcons(searchTerm, selectedGroup = '') { let filtered = [...window.iconComparisonData.allIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } window.iconComparisonData.filteredIcons = filtered; renderFilteredIcons(); }
function renderFilteredIcons() { if (window.iconComparisonData.showGroups) { renderWithGroups(); } else { renderWithoutGroups(); } }
function updateStats() { document.getElementById('iconComparisonTotalCount').textContent = window.iconComparisonData.allIcons.length; document.getElementById('iconComparisonShownCount').textContent = window.iconComparisonData.filteredIcons.length; }
function initGroupSelect(groups) { const groupSelect = document.getElementById('iconComparisonGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconComparisonSearch').value; filterIcons(searchTerm, this.value); }); }
function renderIcons(images, groups) { window.iconComparisonData.allIcons = images; window.iconComparisonData.filteredIcons = [...window.iconComparisonData.allIcons]; window.iconComparisonData.availableGroups = groups; initGroupSelect(groups); renderFilteredIcons(); }
async function loadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); renderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки иконок</div>'; } }
function setupSearch() { const searchInput = document.getElementById('iconComparisonSearch'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; filterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; filterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function setupToggle() { const toggle = document.getElementById('iconComparisonShowGroups'); toggle.addEventListener('change', function() { window.iconComparisonData.showGroups = this.checked; renderFilteredIcons(); }); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { // 1. Определяем тему (простая проверка) detectTheme(); // 2. Загружаем иконки loadIcons(); // 3. Настраиваем остальное setupSearch(); setupToggle(); setupResizeHandler(); // 4. Простая проверка темы при загрузке (без сложных наблюдателей) setTimeout(detectTheme, 100); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> :root { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-icon-bg: #f5f5f5; } /* Тёмная тема - активируется, когда Hugo устанавливает темную тему */ :root[data-theme="dark"], html.dark-mode, body.dark, .dark-theme { /* Переопределяем переменные для темной темы */ --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-icon-bg: #2a2a2a; /* Светло-серый фон для иконок */ }
/* Дополнительные проверки для различных способов Hugo включения темной темы */ [data-mode="dark"], [data-theme="dark"], body[class*="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-icon-bg: #2a2a2a; }
.icon-comparison-container { margin: 0; padding: 0; box-sizing: border-box; }
.icon-comparison-body { font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; transition: background-color 0.3s ease, color 0.3s ease; }
.icon-comparison-wrapper { max-width: 1200px; margin: 0 auto; }
.icon-comparison-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.icon-comparison-search-area { margin: 15px 0; text-align: center; }
.icon-comparison-search { width: 300px; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.icon-comparison-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(77, 171, 247, 0.25); }
.icon-comparison-search::placeholder { color: var(--ic-arrow-color); }
.icon-comparison-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; }
.icon-comparison-stats { font-size: 12px; color: var(--ic-text-color); }
.icon-comparison-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.icon-comparison-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.icon-comparison-checkbox { width: 16px; height: 16px; accent-color: var(--ic-search-focus); }
.icon-comparison-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.icon-comparison-group { break-inside: avoid; }
.icon-comparison-group-title { background: var(--ic-header-bg); padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; }
.icon-comparison-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 6px; background: var(--ic-panel-bg); width: 400px; flex-shrink: 0; transition: all 0.3s ease; }
.icon-comparison-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); }
.icon-comparison-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.icon-comparison-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; }
.icon-comparison-row:last-child { border-bottom: none; }
.icon-comparison-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 2px; box-sizing: border-box; }
.icon-comparison-bmp-processed { filter: none; }
.icon-comparison-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.icon-comparison-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.icon-comparison-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.icon-comparison-loading { text-align: center; padding: 20px; color: var(--ic-loading-color); width: 100%; }
.icon-comparison-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.icon-comparison-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.icon-comparison-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .icon-comparison-wrapper { max-width: 100%; padding: 0 10px; } .icon-comparison-panel { width: 100%; max-width: 400px; } .icon-comparison-search { width: 100%; max-width: 300px; } .icon-comparison-controls { flex-wrap: wrap; gap: 10px; } .icon-comparison-panel-header { gap: 4px; padding: 4px 2px; } .icon-comparison-header-label { font-size: 10px; } .icon-comparison-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-comparison-panel { width: 100%; max-width: 100%; } .icon-comparison-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .icon-comparison-header-label { font-size: 9px; } .icon-comparison-arrow { font-size: 12px; } .icon-comparison-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1400px) { .icon-comparison-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .icon-comparison-wrapper { max-width: 1600px; } }
/* Стили для placeholder в темной теме */ :root[data-theme="dark"] .icon-comparison-image, html.dark-mode .icon-comparison-image, body.dark .icon-comparison-image { background-color: var(--ic-icon-bg); } </style> </head> <body class="icon-comparison-body"> <div class="icon-comparison-wrapper"> <h1 class="icon-comparison-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="icon-comparison-search-area"> <input type="text" id="iconComparisonSearch" class="icon-comparison-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="icon-comparison-controls"> <div class="icon-comparison-stats"> Всего иконок для сравнения: <span id="iconComparisonTotalCount">0</span> | Отображено: <span id="iconComparisonShownCount">0</span> </div> <select id="iconComparisonGroupSelect" class="icon-comparison-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="icon-comparison-toggle"> <input type="checkbox" id="iconComparisonShowGroups" class="icon-comparison-checkbox" checked> <label for="iconComparisonShowGroups">Показывать группы</label> </div> </div>
<div class="icon-comparison-grid" id="iconComparisonContainer"> <div class="icon-comparison-loading">Загрузка иконок...</div> </div> </div>
<script> let iconComparisonAllIcons = []; let iconComparisonFilteredIcons = []; let iconComparisonAvailableGroups = []; let iconComparisonShowGroups = true;
// Функция для определения текущей темы Hugo function iconComparisonDetectHugoTheme() { // Проверяем различные способы, которыми Hugo может указывать тему const htmlElement = document.documentElement; const bodyElement = document.body; // Проверяем data-атрибуты if (htmlElement.hasAttribute('data-theme') && htmlElement.getAttribute('data-theme') === 'dark') { return 'dark'; } if (htmlElement.hasAttribute('data-mode') && htmlElement.getAttribute('data-mode') === 'dark') { return 'dark'; } // Проверяем классы на html if (htmlElement.classList.contains('dark')) { return 'dark'; } if (htmlElement.classList.contains('dark-mode')) { return 'dark'; } // Проверяем классы на body if (bodyElement.classList.contains('dark')) { return 'dark'; } if (bodyElement.classList.contains('dark-theme')) { return 'dark'; } if (bodyElement.classList.contains('dark-mode')) { return 'dark'; } // Проверяем наличие класса dark в любом родительском элементе if (document.querySelector('html.dark, body.dark, .dark, [data-theme="dark"], [data-mode="dark"]')) { return 'dark'; } // По умолчанию светлая тема return 'light'; }
// Функция для применения темы к странице function iconComparisonApplyTheme() { const theme = iconComparisonDetectHugoTheme(); const body = document.querySelector('.icon-comparison-body'); const html = document.documentElement; if (theme === 'dark') { // Устанавливаем data-атрибут для CSS-переменных html.setAttribute('data-theme', 'dark'); // Также добавляем класс для дополнительной специфичности body.classList.add('dark-theme'); } else { html.removeAttribute('data-theme'); body.classList.remove('dark-theme'); } console.log('Тема Hugo определена как:', theme); }
// Наблюдатель за изменениями атрибутов (если Hugo динамически меняет тему) function iconComparisonSetupThemeObserver() { const observer = new MutationObserver(function(mutations) { mutations.forEach(function(mutation) { if (mutation.type === 'attributes' && (mutation.attributeName === 'class' || mutation.attributeName === 'data-theme' || mutation.attributeName === 'data-mode')) { iconComparisonApplyTheme(); } }); }); // Наблюдаем за изменениями в html и body observer.observe(document.documentElement, { attributes: true, attributeFilter: ['class', 'data-theme', 'data-mode'] }); observer.observe(document.body, { attributes: true, attributeFilter: ['class'] }); }
// Функция для разбивки больших групп на части function iconComparisonSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function iconComparisonCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function iconComparisonCreateMasonryLayout(groupsChunks) { const container = document.getElementById('iconComparisonContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); // Увеличили максимальное количество колонок const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); console.log(`Container width: ${containerWidth}, Column width: ${columnWidth}, Columns: ${availableColumns}`); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'icon-comparison-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = iconComparisonCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function iconComparisonRenderWithGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = iconComparisonGroupIcons(iconComparisonFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = iconComparisonSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: iconComparisonCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = iconComparisonSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: iconComparisonCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = iconComparisonCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'icon-comparison-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); iconComparisonUpdateStats(); }
// Функция для создания секции группы function iconComparisonCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'icon-comparison-group'; const header = document.createElement('div'); header.className = 'icon-comparison-group-title'; header.textContent = groupName; section.appendChild(header); const column = iconComparisonCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function iconComparisonCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-comparison-panel'; const header = document.createElement('div'); header.className = 'icon-comparison-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'icon-comparison-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'icon-comparison-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'icon-comparison-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'icon-comparison-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(iconComparisonCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function iconComparisonSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (iconComparisonShowGroups && iconComparisonFilteredIcons.length > 0) { iconComparisonRenderWithGroups(); } }, 250); }); }
// Остальные функции с переименованными названиями function iconComparisonProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function iconComparisonLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await iconComparisonProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function iconComparisonGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function iconComparisonCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-comparison-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ iconComparisonLoadImage(icon.before, `${icon.name} (до)`), iconComparisonLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-comparison-image' + (beforeResult.processed ? ' icon-comparison-bmp-processed' : ''); afterImg.className = 'icon-comparison-image' + (afterResult.processed ? ' icon-comparison-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'icon-comparison-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-comparison-info'; const name = document.createElement('div'); name.className = 'icon-comparison-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function iconComparisonRenderWithoutGroups() { const container = document.getElementById('iconComparisonContainer'); container.innerHTML = ''; if (iconComparisonFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-comparison-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconComparisonUpdateStats(); return; } const sortedIcons = [...iconComparisonFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = iconComparisonCreateIconColumn(columnIcons); container.appendChild(column); } iconComparisonUpdateStats(); }
function iconComparisonFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...iconComparisonAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } iconComparisonFilteredIcons = filtered; iconComparisonRenderFilteredIcons(); }
function iconComparisonRenderFilteredIcons() { if (iconComparisonShowGroups) { iconComparisonRenderWithGroups(); } else { iconComparisonRenderWithoutGroups(); } }
function iconComparisonUpdateStats() { document.getElementById('iconComparisonTotalCount').textContent = iconComparisonAllIcons.length; document.getElementById('iconComparisonShownCount').textContent = iconComparisonFilteredIcons.length; }
function iconComparisonInitGroupSelect(groups) { const groupSelect = document.getElementById('iconComparisonGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconComparisonSearch').value; iconComparisonFilterIcons(searchTerm, this.value); }); }
function iconComparisonRenderIcons(images, groups) { iconComparisonAllIcons = images; iconComparisonFilteredIcons = [...iconComparisonAllIcons]; iconComparisonAvailableGroups = groups; iconComparisonInitGroupSelect(groups); iconComparisonRenderFilteredIcons(); }
async function iconComparisonLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); iconComparisonRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('iconComparisonContainer').innerHTML = '<div class="icon-comparison-loading">Ошибка загрузки иконок</div>'; } }
function iconComparisonSetupSearch() { const searchInput = document.getElementById('iconComparisonSearch'); let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('iconComparisonGroupSelect').value; iconComparisonFilterIcons('', selectedGroup); } }); searchInput.addEventListener('keypress', function(e) { e.stopPropagation(); }); searchInput.addEventListener('keydown', function(e) { e.stopPropagation(); }); }
function iconComparisonSetupToggle() { const toggle = document.getElementById('iconComparisonShowGroups'); toggle.addEventListener('change', function() { iconComparisonShowGroups = this.checked; iconComparisonRenderFilteredIcons(); }); }
window.addEventListener('DOMContentLoaded', function() { // Применяем тему при загрузке iconComparisonApplyTheme(); // Настраиваем наблюдатель за изменениями темы iconComparisonSetupThemeObserver(); // Загружаем остальные компоненты iconComparisonLoadIcons(); iconComparisonSetupSearch(); iconComparisonSetupToggle(); iconComparisonSetupResizeHandler(); }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Изолированные CSS-переменные с уникальными префиксами */ .icon-compare-isolated-root { /* Светлая тема (по умолчанию) */ --icon-compare-bg-color: #ffffff; --icon-compare-text-color: #333333; --icon-compare-panel-bg: #fafafa; --icon-compare-border-color: #e0e0e0; --icon-compare-header-bg: #f8f9fa; --icon-compare-header-text: #495057; --icon-compare-controls-bg: #f0f0f0; --icon-compare-search-border: #ddd; --icon-compare-search-focus: #007bff; --icon-compare-arrow-color: #999; --icon-compare-loading-color: #666; --icon-compare-row-border: #f0f0f0; --icon-compare-icon-bg: #f5f5f5; --icon-compare-select-bg: #ffffff; --icon-compare-checkbox-color: #007bff; }
/* Темная тема */ .icon-compare-isolated-root[data-icon-compare-theme="dark"], .icon-compare-isolated-root.icon-compare-dark { --icon-compare-bg-color: #1a1a1a; --icon-compare-text-color: #e0e0e0; --icon-compare-panel-bg: #2d2d2d; --icon-compare-border-color: #404040; --icon-compare-header-bg: #3d3d3d; --icon-compare-header-text: #cccccc; --icon-compare-controls-bg: #333333; --icon-compare-search-border: #555; --icon-compare-search-focus: #4dabf7; --icon-compare-arrow-color: #888; --icon-compare-loading-color: #999; --icon-compare-row-border: #3a3a3a; --icon-compare-icon-bg: #2a2a2a; --icon-compare-select-bg: #2d2d2d; --icon-compare-checkbox-color: #4dabf7; }
/* Основной контейнер с уникальным классом */ .icon-compare-isolated-root { margin: 0; padding: 0; box-sizing: border-box; font-family: Arial, sans-serif; background-color: var(--icon-compare-bg-color); color: var(--icon-compare-text-color); padding: 10px; font-size: 11px; min-height: 100vh; transition: background-color 0.3s ease, color 0.3s ease; isolation: isolate; /* Изолируем от внешних стилей */ }
/* Все классы с уникальным префиксом */ .icon-compare-wrapper { max-width: 1200px; margin: 0 auto; }
.icon-compare-title { text-align: center; color: var(--icon-compare-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.icon-compare-search-area { margin: 15px 0; text-align: center; }
.icon-compare-search { width: 300px; padding: 8px 12px; border: 1px solid var(--icon-compare-search-border); border-radius: 4px; font-size: 14px; background-color: var(--icon-compare-bg-color); color: var(--icon-compare-text-color); transition: all 0.3s ease; }
.icon-compare-search:focus { outline: none; border-color: var(--icon-compare-search-focus); box-shadow: 0 0 0 2px rgba(77, 171, 247, 0.25); }
.icon-compare-search::placeholder { color: var(--icon-compare-arrow-color); }
.icon-compare-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--icon-compare-controls-bg); border-radius: 4px; border: 1px solid var(--icon-compare-border-color); transition: all 0.3s ease; }
.icon-compare-stats { font-size: 12px; color: var(--icon-compare-text-color); }
.icon-compare-group-select { padding: 6px 10px; border: 1px solid var(--icon-compare-search-border); border-radius: 4px; font-size: 12px; background: var(--icon-compare-select-bg); color: var(--icon-compare-text-color); transition: all 0.3s ease; cursor: pointer; }
.icon-compare-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--icon-compare-text-color); }
.icon-compare-checkbox { width: 16px; height: 16px; accent-color: var(--icon-compare-checkbox-color); cursor: pointer; }
.icon-compare-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.icon-compare-group { break-inside: avoid; }
.icon-compare-group-title { background: var(--icon-compare-header-bg); padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--icon-compare-header-text); font-size: 13px; transition: all 0.3s ease; }
.icon-compare-panel { border: 1px solid var(--icon-compare-border-color); border-radius: 5px; padding: 6px; background: var(--icon-compare-panel-bg); width: 400px; flex-shrink: 0; transition: all 0.3s ease; }
.icon-compare-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--icon-compare-border-color); font-weight: bold; text-align: center; color: var(--icon-compare-text-color); }
.icon-compare-header-label { font-size: 11px; color: var(--icon-compare-text-color); white-space: nowrap; }
.icon-compare-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--icon-compare-row-border); min-height: 36px; transition: border-color 0.3s ease; }
.icon-compare-row:last-child { border-bottom: none; }
.icon-compare-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: var(--icon-compare-icon-bg); border-radius: 3px; padding: 2px; box-sizing: border-box; }
.icon-compare-bmp-processed { filter: none; }
.icon-compare-arrow { text-align: center; color: var(--icon-compare-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.icon-compare-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.icon-compare-name { color: var(--icon-compare-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.icon-compare-loading { text-align: center; padding: 20px; color: var(--icon-compare-loading-color); width: 100%; }
.icon-compare-no-results { text-align: center; padding: 40px; color: var(--icon-compare-loading-color); width: 100%; font-size: 14px; }
.icon-compare-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.icon-compare-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Стили для оптимизации перерисовки */ .icon-compare-transition-none { transition: none !important; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .icon-compare-wrapper { max-width: 100%; padding: 0 10px; } .icon-compare-panel { width: 100%; max-width: 400px; } .icon-compare-search { width: 100%; max-width: 300px; } .icon-compare-controls { flex-wrap: wrap; gap: 10px; } .icon-compare-panel-header { gap: 4px; padding: 4px 2px; } .icon-compare-header-label { font-size: 10px; } .icon-compare-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .icon-compare-panel { width: 100%; max-width: 100%; } .icon-compare-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .icon-compare-header-label { font-size: 9px; } .icon-compare-arrow { font-size: 12px; } .icon-compare-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1400px) { .icon-compare-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .icon-compare-wrapper { max-width: 1600px; } }
/* Глобальный сброс только для нашего контейнера */ .icon-compare-isolated-root * { box-sizing: border-box; margin: 0; padding: 0; font-family: inherit; font-size: inherit; } </style> </head> <body class="icon-compare-isolated-root"> <div class="icon-compare-wrapper"> <h1 class="icon-compare-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="icon-compare-search-area"> <input type="text" id="iconCompareSearch" class="icon-compare-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="icon-compare-controls"> <div class="icon-compare-stats"> Всего иконок для сравнения: <span id="iconCompareTotalCount">0</span> | Отображено: <span id="iconCompareShownCount">0</span> </div> <select id="iconCompareGroupSelect" class="icon-compare-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="icon-compare-toggle"> <input type="checkbox" id="iconCompareShowGroups" class="icon-compare-checkbox" checked> <label for="iconCompareShowGroups">Показывать группы</label> </div> </div>
<div class="icon-compare-grid" id="iconCompareContainer"> <div class="icon-compare-loading">Загрузка иконок...</div> </div> </div>
<script> // Изолированные переменные с уникальными префиксами let iconCompareAllIcons = []; let iconCompareFilteredIcons = []; let iconCompareAvailableGroups = []; let iconCompareShowGroups = true; let iconCompareThemeObserver = null; let iconCompareThemeCheckInterval = null;
// Оптимизированная функция для определения темы Hugo function iconCompareDetectHugoTheme() { const htmlElement = document.documentElement; const bodyElement = document.body; // Быстрая проверка основных способов Hugo if (htmlElement.getAttribute('data-theme') === 'dark' || htmlElement.getAttribute('data-mode') === 'dark' || htmlElement.classList.contains('dark') || htmlElement.classList.contains('dark-mode') || bodyElement.classList.contains('dark') || bodyElement.classList.contains('dark-theme') || bodyElement.classList.contains('dark-mode')) { return 'dark'; } return 'light'; }
// Оптимизированная функция для применения темы function iconCompareApplyTheme() { try { const theme = iconCompareDetectHugoTheme(); const rootElement = document.querySelector('.icon-compare-isolated-root'); if (!rootElement) return; // Временно отключаем transitions для предотвращения зависания rootElement.classList.add('icon-compare-transition-none'); if (theme === 'dark') { rootElement.setAttribute('data-icon-compare-theme', 'dark'); rootElement.classList.add('icon-compare-dark'); } else { rootElement.removeAttribute('data-icon-compare-theme'); rootElement.classList.remove('icon-compare-dark'); } // Возвращаем transitions через небольшой таймаут setTimeout(() => { rootElement.classList.remove('icon-compare-transition-none'); }, 50); } catch (error) { console.warn('Ошибка применения темы:', error); } }
// Оптимизированный наблюдатель с ограничением частоты проверки function iconCompareSetupThemeObserver() { // Очищаем предыдущие наблюдатели if (iconCompareThemeObserver) { iconCompareThemeObserver.disconnect(); } if (iconCompareThemeCheckInterval) { clearInterval(iconCompareThemeCheckInterval); } let lastTheme = iconCompareDetectHugoTheme(); // Используем легковесный интервал вместо MutationObserver iconCompareThemeCheckInterval = setInterval(() => { const currentTheme = iconCompareDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; iconCompareApplyTheme(); } }, 500); // Проверка каждые 500ms вместо мгновенной реакции // Минимальный MutationObserver только для атрибутов iconCompareThemeObserver = new MutationObserver((mutations) => { let shouldUpdate = false; for (const mutation of mutations) { if (mutation.type === 'attributes' && (mutation.attributeName === 'class' || mutation.attributeName === 'data-theme' || mutation.attributeName === 'data-mode')) { shouldUpdate = true; break; } } if (shouldUpdate) { // Используем requestAnimationFrame для предотвращения зависания requestAnimationFrame(() => { iconCompareApplyTheme(); }); } }); // Наблюдаем только за основными элементами try { iconCompareThemeObserver.observe(document.documentElement, { attributes: true, attributeFilter: ['class', 'data-theme', 'data-mode'], attributeOldValue: false // Убираем лишнюю нагрузку }); iconCompareThemeObserver.observe(document.body, { attributes: true, attributeFilter: ['class'], attributeOldValue: false }); } catch (e) { console.warn('Не удалось настроить наблюдатель тем:', e); } }
// Функция для разбивки больших групп на части function iconCompareSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function iconCompareCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function iconCompareCreateMasonryLayout(groupsChunks) { const container = document.getElementById('iconCompareContainer'); if (!container) return []; const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'icon-compare-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = iconCompareCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function iconCompareRenderWithGroups() { const container = document.getElementById('iconCompareContainer'); if (!container) return; container.innerHTML = ''; if (iconCompareFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-compare-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconCompareUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = iconCompareGroupIcons(iconCompareFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = iconCompareSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: iconCompareCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = iconCompareSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: iconCompareCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = iconCompareCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'icon-compare-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); iconCompareUpdateStats(); }
// Функция для создания секции группы function iconCompareCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'icon-compare-group'; const header = document.createElement('div'); header.className = 'icon-compare-group-title'; header.textContent = groupName; section.appendChild(header); const column = iconCompareCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function iconCompareCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'icon-compare-panel'; const header = document.createElement('div'); header.className = 'icon-compare-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'icon-compare-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'icon-compare-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'icon-compare-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'icon-compare-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(iconCompareCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function iconCompareSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (iconCompareShowGroups && iconCompareFilteredIcons.length > 0) { iconCompareRenderWithGroups(); } }, 250); }); }
// Остальные функции с уникальными префиксами function iconCompareProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { console.warn('Ошибка обработки BMP:', error); resolve({ element: img, processed: false }); } }); }
function iconCompareLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await iconCompareProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { console.warn('Ошибка обработки изображения:', error, src); img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { console.warn('Ошибка загрузки изображения:', src); const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function iconCompareGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function iconCompareCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'icon-compare-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ iconCompareLoadImage(icon.before, `${icon.name} (до)`), iconCompareLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'icon-compare-image' + (beforeResult.processed ? ' icon-compare-bmp-processed' : ''); afterImg.className = 'icon-compare-image' + (afterResult.processed ? ' icon-compare-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'icon-compare-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'icon-compare-info'; const name = document.createElement('div'); name.className = 'icon-compare-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function iconCompareRenderWithoutGroups() { const container = document.getElementById('iconCompareContainer'); if (!container) return; container.innerHTML = ''; if (iconCompareFilteredIcons.length === 0) { container.innerHTML = '<div class="icon-compare-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; iconCompareUpdateStats(); return; } const sortedIcons = [...iconCompareFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = iconCompareCreateIconColumn(columnIcons); container.appendChild(column); } iconCompareUpdateStats(); }
function iconCompareFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...iconCompareAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } iconCompareFilteredIcons = filtered; iconCompareRenderFilteredIcons(); }
function iconCompareRenderFilteredIcons() { if (iconCompareShowGroups) { iconCompareRenderWithGroups(); } else { iconCompareRenderWithoutGroups(); } }
function iconCompareUpdateStats() { const totalCount = document.getElementById('iconCompareTotalCount'); const shownCount = document.getElementById('iconCompareShownCount'); if (totalCount) totalCount.textContent = iconCompareAllIcons.length; if (shownCount) shownCount.textContent = iconCompareFilteredIcons.length; }
function iconCompareInitGroupSelect(groups) { const groupSelect = document.getElementById('iconCompareGroupSelect'); if (!groupSelect) return; groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('iconCompareSearch').value; iconCompareFilterIcons(searchTerm, this.value); }); }
function iconCompareRenderIcons(images, groups) { iconCompareAllIcons = images; iconCompareFilteredIcons = [...iconCompareAllIcons]; iconCompareAvailableGroups = groups; iconCompareInitGroupSelect(groups); iconCompareRenderFilteredIcons(); }
async function iconCompareLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); iconCompareRenderIcons(processedImages, data.availableGroups || []); } else { const container = document.getElementById('iconCompareContainer'); if (container) container.innerHTML = '<div class="icon-compare-loading">Ошибка загрузки данных</div>'; } } catch (error) { const container = document.getElementById('iconCompareContainer'); if (container) container.innerHTML = '<div class="icon-compare-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { const container = document.getElementById('iconCompareContainer'); if (container) container.innerHTML = '<div class="icon-compare-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { const container = document.getElementById('iconCompareContainer'); if (container) container.innerHTML = '<div class="icon-compare-loading">Ошибка загрузки иконок</div>'; } }
function iconCompareSetupSearch() { const searchInput = document.getElementById('iconCompareSearch'); if (!searchInput) return; let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('iconCompareGroupSelect').value; iconCompareFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('iconCompareGroupSelect').value; iconCompareFilterIcons('', selectedGroup); } }); }
function iconCompareSetupToggle() { const toggle = document.getElementById('iconCompareShowGroups'); if (!toggle) return; toggle.addEventListener('change', function() { iconCompareShowGroups = this.checked; iconCompareRenderFilteredIcons(); }); }
// Очистка при разгрузке страницы function iconCompareCleanup() { if (iconCompareThemeObserver) { iconCompareThemeObserver.disconnect(); iconCompareThemeObserver = null; } if (iconCompareThemeCheckInterval) { clearInterval(iconCompareThemeCheckInterval); iconCompareThemeCheckInterval = null; } }
// Основная инициализация document.addEventListener('DOMContentLoaded', function() { try { // Применяем тему при загрузке iconCompareApplyTheme(); // Настраиваем наблюдатель за изменениями темы iconCompareSetupThemeObserver(); // Загружаем остальные компоненты iconCompareLoadIcons(); iconCompareSetupSearch(); iconCompareSetupToggle(); iconCompareSetupResizeHandler(); // Очистка при закрытии страницы window.addEventListener('beforeunload', iconCompareCleanup); window.addEventListener('pagehide', iconCompareCleanup); } catch (error) { console.error('Ошибка инициализации:', error); } }); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полностью изолированные стили только для этой страницы */ html, body { margin: 0; padding: 0; width: 100%; height: 100%; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-icon-bg: #f5f5f5; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; box-sizing: border-box; transition: background-color 0.3s ease, color 0.3s ease; /* Изоляция от внешних стилей */ all: initial; font-family: Arial, sans-serif !important; } /* Темная тема - будет активироваться когда Hugo в темном режиме */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-icon-bg: #2a2a2a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; }
/* Восстанавливаем оригинальные стили с уникальными префиксами */ .ic-isolated-page .ic-container { margin: 0; padding: 0; box-sizing: border-box; }
.ic-isolated-page .ic-body { font-family: Arial, sans-serif; background-color: var(--ic-bg-color); padding: 10px; font-size: 11px; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1200px; margin: 0 auto; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; }
.ic-isolated-page .ic-search { width: 300px; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.ic-isolated-page .ic-group { break-inside: avoid; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 6px; background: var(--ic-panel-bg); width: 400px; flex-shrink: 0; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 2px; box-sizing: border-box; }
.ic-isolated-page .ic-bmp-processed { filter: none; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.ic-isolated-page .ic-loading { text-align: center; padding: 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.ic-isolated-page .ic-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .ic-isolated-page .ic-wrapper { max-width: 100%; padding: 0 10px; } .ic-isolated-page .ic-panel { width: 100%; max-width: 400px; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { flex-wrap: wrap; gap: 10px; } .ic-isolated-page .ic-panel-header { gap: 4px; padding: 4px 2px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel { width: 100%; max-width: 100%; } .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .ic-isolated-page .ic-header-label { font-size: 9px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .ic-isolated-page .ic-wrapper { max-width: 1600px; } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> // Используем IIFE для полной изоляции (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo - легковесная function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; // Быстрая проверка основных способов Hugo if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы - без transitions чтобы не зависать function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; // Быстрое переключение без анимации if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { // Игнорируем ошибки, чтобы не ломать страницу } }
// Простой наблюдатель за темой function icSetupThemeObserver() { // Очищаем предыдущий интервал if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); // Проверяем тему раз в секунду - этого достаточно icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); // Применяем тему сразу icApplyTheme(); }
// Оригинальные функции с короткими именами function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
function icCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); const containerWidth = container.clientWidth; const columnWidth = 400 + 12; const maxColumns = Math.floor(1600 / columnWidth); const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'ic-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = icCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: icCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: icCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
function icCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'ic-group'; const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; section.appendChild(header); const column = icCreateIconColumn(icons); section.appendChild(column); return section; }
function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
function icSetupResizeHandler() { window.addEventListener('resize', function() { clearTimeout(icResizeTimeout); icResizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { resolve({ element: img, processed: false }); } }); }
function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'ic-image' + (beforeResult.processed ? ' ic-bmp-processed' : ''); afterImg.className = 'ic-image' + (afterResult.processed ? ' ic-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); container.appendChild(column); } icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { document.getElementById('icTotalCount').textContent = icAllIcons.length; document.getElementById('icShownCount').textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); searchInput.addEventListener('input', function(e) { clearTimeout(icSearchTimeout); icSearchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { // Настраиваем тему icSetupThemeObserver(); // Загружаем иконки icLoadIcons(); // Настраиваем остальное icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); // Очистка window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); }); })(); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полностью изолированные стили только для этой страницы */ html, body { margin: 0; padding: 0; width: 100%; height: 100%; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-icon-bg: #f5f5f5; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; box-sizing: border-box; transition: background-color 0.3s ease, color 0.3s ease; /* Изоляция от внешних стилей */ all: initial; font-family: Arial, sans-serif !important; } /* Темная тема - будет активироваться когда Hugo в темном режиме */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-icon-bg: #2a2a2a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; }
/* Восстанавливаем оригинальные стили с уникальными префиксами */ .ic-isolated-page .ic-container { margin: 0; padding: 0; box-sizing: border-box; }
.ic-isolated-page .ic-body { font-family: Arial, sans-serif; background-color: var(--ic-bg-color); padding: 10px; font-size: 11px; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1200px; margin: 0 auto; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; }
.ic-isolated-page .ic-search { width: 300px; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.ic-isolated-page .ic-group { break-inside: avoid; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 6px; background: var(--ic-panel-bg); width: 400px; flex-shrink: 0; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 2px; box-sizing: border-box; }
.ic-isolated-page .ic-bmp-processed { filter: none; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.ic-isolated-page .ic-loading { text-align: center; padding: 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.ic-isolated-page .ic-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Медиа-запросы для адаптивной ширины колонок */ @media (max-width: 831px) { .ic-isolated-page .ic-panel { width: 100%; max-width: 400px; } .ic-isolated-page .ic-wrapper { max-width: 100%; padding: 0 10px; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { flex-wrap: wrap; gap: 10px; } .ic-isolated-page .ic-panel-header { gap: 4px; padding: 4px 2px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-name { font-size: 11px; } }
/* При ширине 832px и выше - две колонки */ @media (min-width: 832px) and (max-width: 1199px) { .ic-isolated-page .ic-panel { width: calc(50% - 6px); max-width: none; min-width: 300px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel { width: 100%; max-width: 100%; } .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .ic-isolated-page .ic-header-label { font-size: 9px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов - больше колонок */ @media (min-width: 1200px) { .ic-isolated-page .ic-wrapper { max-width: 1200px; } }
@media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .ic-isolated-page .ic-wrapper { max-width: 1600px; } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo - легковесная function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; // Быстрая проверка основных способов Hugo if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы - без transitions чтобы не зависать function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; // Быстрое переключение без анимации if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { // Игнорируем ошибки, чтобы не ломать страницу } }
// Простой наблюдатель за темой function icSetupThemeObserver() { // Очищаем предыдущий интервал if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); // Проверяем тему раз в секунду - этого достаточно icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); // Применяем тему сразу icApplyTheme(); }
// Оригинальные функции с короткими именами function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
function icCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); if (!container) return []; const containerWidth = container.clientWidth; // Определяем количество колонок на основе ширины контейнера let columnCount; if (containerWidth >= 1200) { // Широкие экраны - 3 колонки columnCount = Math.min(3, groupsChunks.length); } else if (containerWidth >= 832) { // Средние экраны (832px и выше) - 2 колонки columnCount = Math.min(2, groupsChunks.length); } else { // Узкие экраны - 1 колонка columnCount = 1; } // Создаем колонки const columns = Array.from({ length: columnCount }, () => { const col = document.createElement('div'); col.className = 'ic-masonry-column'; col.style.flex = '1'; col.style.minWidth = '0'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(columnCount).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = icCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); // Учитываем высоту группы columnHeights[columnIndex] += group.height; }); return columns; }
function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: icCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: icCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columnsContainer.style.display = 'flex'; columnsContainer.style.flexWrap = 'wrap'; columnsContainer.style.gap = '12px'; columnsContainer.style.justifyContent = 'center'; columnsContainer.style.width = '100%'; columns.forEach(col => { if (col.children.length > 0) { // Рассчитываем ширину колонки в зависимости от их количества const containerWidth = container.clientWidth; let columnWidth; if (containerWidth >= 1200) { // 3 колонки columnWidth = `calc(33.333% - 8px)`; } else if (containerWidth >= 832) { // 2 колонки columnWidth = `calc(50% - 6px)`; } else { // 1 колонка columnWidth = '100%'; } col.style.width = columnWidth; col.style.maxWidth = '400px'; col.style.minWidth = '300px'; col.style.flexShrink = '0'; columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
function icCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'ic-group'; const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; section.appendChild(header); const column = icCreateIconColumn(icons); section.appendChild(column); return section; }
function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; column.style.width = '100%'; column.style.boxSizing = 'border-box'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
function icSetupResizeHandler() { window.addEventListener('resize', function() { clearTimeout(icResizeTimeout); icResizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { resolve({ element: img, processed: false }); } }); }
function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'ic-image' + (beforeResult.processed ? ' ic-bmp-processed' : ''); afterImg.className = 'ic-image' + (afterResult.processed ? ' ic-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); // Создаем контейнер для колонок const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columnsContainer.style.display = 'flex'; columnsContainer.style.flexWrap = 'wrap'; columnsContainer.style.gap = '12px'; columnsContainer.style.justifyContent = 'center'; columnsContainer.style.width = '100%'; for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); // Применяем ту же логику ширины что и в режиме с группами const containerWidth = container.clientWidth; let columnWidth; if (containerWidth >= 1200) { columnWidth = '400px'; } else if (containerWidth >= 832) { columnWidth = `calc(50% - 6px)`; } else { columnWidth = '100%'; } column.style.width = columnWidth; column.style.maxWidth = '400px'; column.style.minWidth = '300px'; column.style.flexShrink = '0'; columnsContainer.appendChild(column); } container.appendChild(columnsContainer); icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { filtered = filtered.filter(icon => icon.group === selectedGroup); } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { document.getElementById('icTotalCount').textContent = icAllIcons.length; document.getElementById('icShownCount').textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = groups.length > 0 ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); searchInput.addEventListener('input', function(e) { clearTimeout(icSearchTimeout); icSearchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { // Настраиваем тему icSetupThemeObserver(); // Загружаем иконки icLoadIcons(); // Настраиваем остальное icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); // Очистка window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); }); })(); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полностью изолированные стили только для этой страницы */ html, body { margin: 0; padding: 0; width: 100%; height: 100%; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-icon-bg: #f5f5f5; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; box-sizing: border-box; transition: background-color 0.3s ease, color 0.3s ease; /* Изоляция от внешних стилей */ all: initial; font-family: Arial, sans-serif !important; } /* Темная тема - будет активироваться когда Hugo в темном режиме */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-icon-bg: #2a2a2a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; }
/* Восстанавливаем оригинальные стили с уникальными префиксами */ .ic-isolated-page .ic-container { margin: 0; padding: 0; box-sizing: border-box; }
.ic-isolated-page .ic-body { font-family: Arial, sans-serif; background-color: var(--ic-bg-color); padding: 10px; font-size: 11px; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1400px; margin: 0 auto; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; }
.ic-isolated-page .ic-search { width: 300px; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.ic-isolated-page .ic-group { break-inside: avoid; width: 100%; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 6px; background: var(--ic-panel-bg); width: 400px; flex-shrink: 0; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 2px; box-sizing: border-box; }
.ic-isolated-page .ic-bmp-processed { filter: none; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.ic-isolated-page .ic-loading { text-align: center; padding: 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.ic-isolated-page .ic-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .ic-isolated-page .ic-wrapper { max-width: 100%; padding: 0 10px; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { flex-wrap: wrap; gap: 10px; } .ic-isolated-page .ic-panel-header { gap: 4px; padding: 4px 2px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-name { font-size: 11px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .ic-isolated-page .ic-header-label { font-size: 9px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 200px; } }
/* Для широких экранов */ @media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .ic-isolated-page .ic-wrapper { max-width: 1600px; } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> <option value="no_group">Прочие</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo - легковесная function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; // Быстрая проверка основных способов Hugo if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы - без transitions чтобы не зависать function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; // Быстрое переключение без анимации if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { // Игнорируем ошибки, чтобы не ломать страницу } }
// Простой наблюдатель за темой function icSetupThemeObserver() { // Очищаем предыдущий интервал if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); // Проверяем тему раз в секунду - этого достаточно icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); // Применяем тему сразу icApplyTheme(); }
// Функция для разбивки больших групп на части function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function icCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); if (!container) return []; const containerWidth = container.clientWidth; const columnWidth = 400 + 12; // Ширина панели + отступ // Динамическое количество колонок let maxColumns; if (containerWidth >= 1400) { maxColumns = 4; } else if (containerWidth >= 1200) { maxColumns = 3; } else if (containerWidth >= 832) { maxColumns = 2; } else { maxColumns = 1; } const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); console.log(`Container width: ${containerWidth}, Column width: ${columnWidth}, Columns: ${availableColumns}`); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'ic-masonry-column'; col.style.width = '100%'; col.style.maxWidth = '412px'; // 400px + 12px отступ return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = icCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: icCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы как "Прочие" if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: icCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
// Функция для создания секции группы function icCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'ic-group'; const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; section.appendChild(header); const column = icCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function icSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
// Остальные функции с уникальными префиксами function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { resolve({ element: img, processed: false }); } }); }
function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'ic-image' + (beforeResult.processed ? ' ic-bmp-processed' : ''); afterImg.className = 'ic-image' + (afterResult.processed ? ' ic-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); container.appendChild(column); } icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { if (selectedGroup === 'no_group') { // Фильтруем иконки без группы filtered = filtered.filter(icon => !icon.group || icon.group.trim() === ''); } else { filtered = filtered.filter(icon => icon.group === selectedGroup); } } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { document.getElementById('icTotalCount').textContent = icAllIcons.length; document.getElementById('icShownCount').textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; // Добавляем опцию "Прочие" groupSelect.innerHTML += '<option value="no_group">Прочие</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = (groups.length > 0) ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
async function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); searchInput.addEventListener('input', function(e) { clearTimeout(icSearchTimeout); icSearchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { // Настраиваем тему icSetupThemeObserver(); // Загружаем иконки icLoadIcons(); // Настраиваем остальное icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); // Очистка window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); }); })(); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полностью изолированные стили только для этой страницы */ html, body { margin: 0; padding: 0; width: 100%; height: 100%; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-icon-bg: #f5f5f5; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; box-sizing: border-box; transition: background-color 0.3s ease, color 0.3s ease; /* Изоляция от внешних стилей */ all: initial; font-family: Arial, sans-serif !important; } /* Темная тема - будет активироваться когда Hugo в темном режиме */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-icon-bg: #2a2a2a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; }
/* Восстанавливаем оригинальные стили с уникальными префиксами */ .ic-isolated-page .ic-container { margin: 0; padding: 0; box-sizing: border-box; }
.ic-isolated-page .ic-body { font-family: Arial, sans-serif; background-color: var(--ic-bg-color); padding: 10px; font-size: 11px; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1400px; margin: 0 auto; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; }
.ic-isolated-page .ic-search { width: 300px; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { display: flex; flex-wrap: wrap; gap: 12px; align-items: flex-start; justify-content: center; }
.ic-isolated-page .ic-group { break-inside: avoid; width: 100%; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 6px 10px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 6px; background: var(--ic-panel-bg); width: 380px; /* Уменьшено с 400px до 380px */ flex-shrink: 0; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; image-rendering: -webkit-optimize-contrast; image-rendering: crisp-edges; image-rendering: pixelated; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 2px; box-sizing: border-box; }
.ic-isolated-page .ic-bmp-processed { filter: none; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 200px; /* Уменьшено с 220px до 200px */ }
.ic-isolated-page .ic-loading { text-align: center; padding: 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 10px; /* Уменьшено с 12px до 10px */ width: 100%; justify-content: center; }
.ic-isolated-page .ic-masonry-column { display: flex; flex-direction: column; gap: 12px; }
/* Адаптивность для мобильных устройств */ @media (max-width: 768px) { .ic-isolated-page .ic-wrapper { max-width: 100%; padding: 0 10px; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { flex-wrap: wrap; gap: 10px; } .ic-isolated-page .ic-panel-header { gap: 4px; padding: 4px 2px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 180px; } .ic-isolated-page .ic-panel { width: 100%; max-width: 380px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 3px; } .ic-isolated-page .ic-header-label { font-size: 9px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 150px; } .ic-isolated-page .ic-group-title { font-size: 12px; padding: 4px 8px; } }
/* Для широких экранов */ @media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } }
@media (min-width: 1600px) { .ic-isolated-page .ic-wrapper { max-width: 1600px; } .ic-isolated-page .ic-panel { width: 380px; } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> <option value="no_group">Прочие</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo - легковесная function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; // Быстрая проверка основных способов Hugo if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы - без transitions чтобы не зависать function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; // Быстрое переключение без анимации if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { // Игнорируем ошибки, чтобы не ломать страницу } }
// Простой наблюдатель за темой function icSetupThemeObserver() { // Очищаем предыдущий интервал if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); // Проверяем тему раз в секунду - этого достаточно icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); // Применяем тему сразу icApplyTheme(); }
// Функция для разбивки больших групп на части function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function icCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); if (!container) return []; const containerWidth = container.clientWidth; const panelWidth = 380; // Ширина панели const gap = 10; // Отступ между колонками const totalPanelWidth = panelWidth + gap; // Динамическое количество колонок let maxColumns; if (containerWidth >= 1400) { maxColumns = 4; } else if (containerWidth >= 1200) { maxColumns = 3; } else if (containerWidth >= 800) { // Уменьшено с 832px до 800px maxColumns = 2; } else { maxColumns = 1; } const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / totalPanelWidth))); console.log(`Container width: ${containerWidth}, Panel width: ${panelWidth}, Columns: ${availableColumns}`); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'ic-masonry-column'; col.style.width = `${panelWidth}px`; col.style.flexShrink = '0'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = icCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой (masonry компоновка) function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: icCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы как "Прочие" if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: icCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columnsContainer.style.display = 'flex'; columnsContainer.style.flexWrap = 'wrap'; columnsContainer.style.gap = '10px'; columnsContainer.style.justifyContent = 'center'; columnsContainer.style.width = '100%'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
// Функция для создания секции группы function icCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'ic-group'; const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; header.title = groupName; section.appendChild(header); const column = icCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; beforeLabel.style.whiteSpace = 'nowrap'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; arrowSpace.style.whiteSpace = 'nowrap'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; afterLabel.style.whiteSpace = 'nowrap'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; nameLabel.style.whiteSpace = 'nowrap'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function icSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
// Остальные функции с уникальными префиксами function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth; canvas.height = img.naturalHeight; ctx.drawImage(img, 0, 0); const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { resolve({ element: img, processed: false }); } }); }
function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; resolve(result); } else { img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { const placeholder = new Image(); placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'ic-image' + (beforeResult.processed ? ' ic-bmp-processed' : ''); afterImg.className = 'ic-image' + (afterResult.processed ? ' ic-bmp-processed' : ''); beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); // Создаем контейнер для колонок const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columnsContainer.style.display = 'flex'; columnsContainer.style.flexWrap = 'wrap'; columnsContainer.style.gap = '10px'; columnsContainer.style.justifyContent = 'center'; columnsContainer.style.width = '100%'; for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); columnsContainer.appendChild(column); } container.appendChild(columnsContainer); icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { if (selectedGroup === 'no_group') { // Фильтруем иконки без группы filtered = filtered.filter(icon => !icon.group || icon.group.trim() === ''); } else { filtered = filtered.filter(icon => icon.group === selectedGroup); } } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { document.getElementById('icTotalCount').textContent = icAllIcons.length; document.getElementById('icShownCount').textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); groupSelect.innerHTML = '<option value="">Все группы</option>'; // Добавляем опцию "Прочие" groupSelect.innerHTML += '<option value="no_group">Прочие</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = (groups.length > 0) ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
async function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { document.getElementById('icContainer').innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); searchInput.addEventListener('input', function(e) { clearTimeout(icSearchTimeout); icSearchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { // Настраиваем тему icSetupThemeObserver(); // Загружаем иконки icLoadIcons(); // Настраиваем остальное icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); // Очистка window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); }); })(); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полный сброс стилей для полной изоляции */ * { margin: 0; padding: 0; box-sizing: border-box; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-icon-bg: #f5f5f5; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; /* Базовые стили */ font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; width: 100%; overflow-x: hidden; } /* Темная тема - ТОЛЬКО для фона и текста, не для иконок */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; /* Фон для иконок остается светлым для контраста */ --ic-icon-bg: #2a2a2a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; }
/* Основные стили с уникальными префиксами */ .ic-isolated-page .ic-container { width: 100%; margin: 0; padding: 0; }
.ic-isolated-page .ic-body { width: 100%; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1400px; margin: 0 auto; width: 100%; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; padding: 0 10px; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; padding: 0 10px; }
.ic-isolated-page .ic-search { width: 300px; max-width: 100%; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 10px; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; flex-wrap: wrap; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; min-width: 150px; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { width: 100%; padding: 0 10px; }
.ic-isolated-page .ic-group { margin-bottom: 12px; width: 100%; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 8px 12px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; width: 100%; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 8px; background: var(--ic-panel-bg); width: 100%; max-width: 380px; min-width: 300px; flex-shrink: 0; transition: all 0.3s ease; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); width: 100%; }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; width: 100%; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 0; display: block; /* Убираем все фильтры для сохранения цветов */ filter: none !important; -webkit-filter: none !important; image-rendering: auto; }
.ic-isolated-page .ic-bmp-processed { /* Убираем все фильтры */ filter: none !important; -webkit-filter: none !important; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 200px; }
.ic-isolated-page .ic-loading { text-align: center; padding: 40px 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.ic-isolated-page .ic-masonry-column { display: flex; flex-direction: column; gap: 12px; width: 100%; max-width: 380px; }
/* Адаптивность */ @media (max-width: 768px) { .ic-isolated-page .ic-wrapper { max-width: 100%; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { gap: 10px; margin: 12px 5px; } .ic-isolated-page .ic-group-select { min-width: 120px; } .ic-isolated-page .ic-name { max-width: 180px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 4px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 150px; } .ic-isolated-page .ic-group-title { font-size: 12px; padding: 6px 10px; } .ic-isolated-page .ic-masonry-column { max-width: 100%; } .ic-isolated-page .ic-panel { max-width: 100%; min-width: 280px; } }
/* Для ширины 800px и выше - 2 колонки */ @media (min-width: 800px) { .ic-isolated-page .ic-masonry-column { max-width: calc(50% - 6px); } }
/* Для ширины 1200px и выше - 3 колонки */ @media (min-width: 1200px) { .ic-isolated-page .ic-masonry-column { max-width: calc(33.333% - 8px); } }
/* Для ширины 1400px и выше - 4 колонки */ @media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } .ic-isolated-page .ic-masonry-column { max-width: calc(25% - 9px); } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> <option value="no_group">Прочие</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo - легковесная function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; // Быстрая проверка основных способов Hugo if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы - без transitions чтобы не зависать function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; // Быстрое переключение без анимации if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { // Игнорируем ошибки, чтобы не ломать страницу } }
// Простой наблюдатель за темой function icSetupThemeObserver() { // Очищаем предыдущий интервал if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); // Проверяем тему раз в секунду - этого достаточно icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); // Применяем тему сразу icApplyTheme(); }
// Функция для разбивки больших групп на части function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для создания адаптивной masonry компоновки function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); if (!container) return []; const containerWidth = container.clientWidth; // Определяем количество колонок на основе ширины let columnCount; if (containerWidth >= 1400) { columnCount = 4; } else if (containerWidth >= 1200) { columnCount = 3; } else if (containerWidth >= 800) { columnCount = 2; } else { columnCount = 1; } // Ограничиваем количеством групп columnCount = Math.min(columnCount, groupsChunks.length); // Создаем колонки const columns = Array.from({ length: columnCount }, () => { const col = document.createElement('div'); col.className = 'ic-masonry-column'; return col; }); // Распределяем группы по колонкам равномерно groupsChunks.forEach((group, index) => { const columnIndex = index % columnCount; const section = icCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); }); return columns; }
// Функция для отображения иконок с группировкой function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы const allGroupChunks = []; // Добавляем группы с иконками Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName }); }); }); // Добавляем иконки без группы как "Прочие" if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group' }); }); } // Создаем masonry раскладку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
// Функция для создания секции группы function icCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'ic-group'; const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; header.title = groupName; section.appendChild(header); const column = icCreateIconColumn(icons); section.appendChild(column); return section; }
// Функция для создания колонки с иконками function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function icSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
// Упрощенная функция для обработки BMP (только убираем magenta) function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = 32; canvas.height = 32; // Рисуем изображение с масштабированием до 32x32 ctx.drawImage(img, 0, 0, 32, 32); // Убираем только magenta (#FF00FF) const imageData = ctx.getImageData(0, 0, 32, 32); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { // Проверяем на чистый magenta if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; // Делаем полностью прозрачным } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { // В случае ошибки возвращаем оригинальное изображение const fallbackImg = new Image(); fallbackImg.onload = () => resolve({ element: fallbackImg, processed: false }); fallbackImg.src = img.src; } }); }
// Упрощенная функция загрузки изображений function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { // Только для BMP файлов делаем простую обработку if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; resolve(result); } else { // Для остальных форматов - оригинальное изображение img.alt = alt; resolve({ element: img, processed: false }); } } catch (error) { // В случае ошибки возвращаем placeholder img.alt = alt; resolve({ element: img, processed: false }); } }; img.onerror = function() { // Заглушка для ошибок загрузки const placeholder = new Image(); placeholder.width = 32; placeholder.height = 32; placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; img.width = 32; img.height = 32; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); // Загружаем изображения Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; // Применяем классы beforeImg.className = 'ic-image'; afterImg.className = 'ic-image'; // Добавляем класс для обработанных BMP if (beforeResult.processed) { beforeImg.classList.add('ic-bmp-processed'); } if (afterResult.processed) { afterImg.classList.add('ic-bmp-processed'); } // Гарантируем размер 32x32 beforeImg.width = 32; beforeImg.height = 32; afterImg.width = 32; afterImg.height = 32; // Добавляем в контейнеры beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); // Создаем контейнер для колонок const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); columnsContainer.appendChild(column); } container.appendChild(columnsContainer); icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { if (selectedGroup === 'no_group') { // Фильтруем иконки без группы filtered = filtered.filter(icon => !icon.group || icon.group.trim() === ''); } else { filtered = filtered.filter(icon => icon.group === selectedGroup); } } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { const totalCount = document.getElementById('icTotalCount'); const shownCount = document.getElementById('icShownCount'); if (totalCount) totalCount.textContent = icAllIcons.length; if (shownCount) shownCount.textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); if (!groupSelect) return; groupSelect.innerHTML = '<option value="">Все группы</option>'; // Добавляем опцию "Прочие" groupSelect.innerHTML += '<option value="no_group">Прочие</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = (groups.length > 0) ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
async function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); if (!searchInput) return; let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); if (!toggle) return; toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { try { // Настраиваем тему icSetupThemeObserver(); // Загружаем иконки icLoadIcons(); // Настраиваем остальное icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); // Очистка window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); } catch (error) { console.error('Ошибка инициализации:', error); } }); })(); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полный сброс стилей для полной изоляции */ * { margin: 0; padding: 0; box-sizing: border-box; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; /* Иконки НЕ меняются в темной теме */ --ic-icon-bg: #f5f5f5; /* Всегда светло-серый */ /* Базовые стили */ font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; width: 100%; overflow-x: hidden; } /* Темная тема - ТОЛЬКО для фона и текста, НЕ для иконок */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; /* Иконки остаются со светлым фоном для контраста */ --ic-icon-bg: #2a2a2a; }
/* Основные стили с уникальными префиксами */ .ic-isolated-page .ic-container { width: 100%; margin: 0; padding: 0; }
.ic-isolated-page .ic-body { width: 100%; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1400px; margin: 0 auto; width: 100%; padding: 0 10px; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; }
.ic-isolated-page .ic-search { width: 300px; max-width: 100%; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; flex-wrap: wrap; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; min-width: 150px; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { width: 100%; }
.ic-isolated-page .ic-group-container { width: 100%; margin-bottom: 20px; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 8px 12px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; transition: all 0.3s ease; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; width: 100%; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 8px; background: var(--ic-panel-bg); width: 400px; flex-shrink: 0; transition: all 0.3s ease; margin-bottom: 12px; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); width: 100%; }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; transition: border-color 0.3s ease; width: 100%; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 0; display: block; /* Иконки защищены от любых фильтров */ filter: none !important; -webkit-filter: none !important; image-rendering: auto; }
.ic-isolated-page .ic-bmp-processed { /* Только для BMP */ filter: none !important; -webkit-filter: none !important; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; width: 100%; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.ic-isolated-page .ic-loading { text-align: center; padding: 40px 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
.ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 12px; width: 100%; justify-content: center; }
.ic-isolated-page .ic-column-panels { display: flex; flex-direction: column; gap: 12px; width: 400px; flex-shrink: 0; }
/* Адаптивность */ @media (max-width: 768px) { .ic-isolated-page .ic-wrapper { max-width: 100%; padding: 0 10px; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { gap: 10px; } .ic-isolated-page .ic-group-select { min-width: 120px; } .ic-isolated-page .ic-name { max-width: 180px; } .ic-isolated-page .ic-panel { width: 100%; max-width: 400px; } .ic-isolated-page .ic-column-panels { width: 100%; max-width: 400px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 4px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 150px; } .ic-isolated-page .ic-group-title { font-size: 12px; padding: 6px 10px; } }
/* Для ширины 800px и выше - 2 колонки */ @media (min-width: 800px) { .ic-isolated-page .ic-column-panels { width: calc(50% - 6px); max-width: 400px; } }
/* Для ширины 1200px и выше - 3 колонки */ @media (min-width: 1200px) { .ic-isolated-page .ic-wrapper { max-width: 1200px; } .ic-isolated-page .ic-column-panels { width: calc(33.333% - 8px); max-width: 400px; } }
/* Для ширины 1400px и выше - 4 колонки */ @media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } .ic-isolated-page .ic-column-panels { width: calc(25% - 9px); max-width: 400px; } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> <option value="no_group">Прочие</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; // Быстрая проверка основных способов Hugo if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; // Быстрое переключение if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { console.warn('Ошибка применения темы:', error); } }
// Простой наблюдатель за темой function icSetupThemeObserver() { // Очищаем предыдущий интервал if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); // Проверяем тему раз в секунду icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); // Применяем тему сразу icApplyTheme(); }
// Функция для разбивки больших групп на части function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для создания адаптивной masonry компоновки - УПРОЩЕННАЯ function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); if (!container) return []; const containerWidth = container.clientWidth; // Определяем количество колонок на основе ширины let columnCount; if (containerWidth >= 1400) { columnCount = 4; } else if (containerWidth >= 1200) { columnCount = 3; } else if (containerWidth >= 800) { columnCount = 2; } else { columnCount = 1; } // Создаем колонки const columns = Array.from({ length: columnCount }, () => { const col = document.createElement('div'); col.className = 'ic-column-panels'; return col; }); // Распределяем группы по колонкам равномерно groupsChunks.forEach((group, index) => { const columnIndex = index % columnCount; const panel = icCreateGroupPanel(group.name, group.icons); columns[columnIndex].appendChild(panel); }); return columns; }
// Функция для отображения иконок с группировкой - НОВАЯ function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; // Добавляем группы с иконками Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName }); }); }); // Добавляем иконки без группы как "Прочие" if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group' }); }); } // Создаем masonry раскладку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
// Функция для создания панели с группой - НОВАЯ function icCreateGroupPanel(groupName, icons) { const container = document.createElement('div'); container.className = 'ic-group-container'; const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; header.title = groupName; container.appendChild(header); const panel = icCreateIconColumn(icons); container.appendChild(panel); return container; }
// Функция для создания колонки с иконками function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function icSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
// Упрощенная функция для обработки BMP (только убираем magenta) function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth || 32; canvas.height = img.naturalHeight || 32; // Рисуем оригинальное изображение ctx.drawImage(img, 0, 0); // Убираем только magenta (#FF00FF) const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { // Проверяем на чистый magenta if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; // Делаем полностью прозрачным } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { // В случае ошибки возвращаем оригинальное изображение resolve({ element: img, processed: false }); } }); }
// Функция загрузки изображений function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { // Только для BMP файлов делаем обработку magenta if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; result.element.width = 32; result.element.height = 32; resolve(result); } else { // Для остальных форматов - оригинальное изображение img.alt = alt; img.width = 32; img.height = 32; resolve({ element: img, processed: false }); } } catch (error) { // В случае ошибки возвращаем оригинальное изображение img.alt = alt; img.width = 32; img.height = 32; resolve({ element: img, processed: false }); } }; img.onerror = function() { // Заглушка для ошибок загрузки const placeholder = new Image(); placeholder.width = 32; placeholder.height = 32; placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); // Загружаем изображения Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; // Применяем классы beforeImg.className = 'ic-image'; afterImg.className = 'ic-image'; // Добавляем класс для обработанных BMP if (beforeResult.processed) { beforeImg.classList.add('ic-bmp-processed'); } if (afterResult.processed) { afterImg.classList.add('ic-bmp-processed'); } // Гарантируем размер 32x32 beforeImg.width = 32; beforeImg.height = 32; beforeImg.style.width = '32px'; beforeImg.style.height = '32px'; afterImg.width = 32; afterImg.height = 32; afterImg.style.width = '32px'; afterImg.style.height = '32px'; // Добавляем в контейнеры beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); // Создаем контейнер для колонок const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); columnsContainer.appendChild(column); } container.appendChild(columnsContainer); icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { if (selectedGroup === 'no_group') { // Фильтруем иконки без группы filtered = filtered.filter(icon => !icon.group || icon.group.trim() === ''); } else { filtered = filtered.filter(icon => icon.group === selectedGroup); } } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { const totalCount = document.getElementById('icTotalCount'); const shownCount = document.getElementById('icShownCount'); if (totalCount) totalCount.textContent = icAllIcons.length; if (shownCount) shownCount.textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); if (!groupSelect) return; groupSelect.innerHTML = '<option value="">Все группы</option>'; // Добавляем опцию "Прочие" groupSelect.innerHTML += '<option value="no_group">Прочие</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = (groups.length > 0) ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
async function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); if (!searchInput) return; let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); if (!toggle) return; toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { try { // Настраиваем тему icSetupThemeObserver(); // Загружаем иконки icLoadIcons(); // Настраиваем остальное icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); // Очистка window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); } catch (error) { console.error('Ошибка инициализации:', error); } }); })(); </script> </body> </html>
|
<!DOCTYPE html> <html lang="ru"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Сравнение иконок ИСИХОГИ</title> <style> /* Полный сброс стилей */ * { margin: 0; padding: 0; box-sizing: border-box; } /* Изолированный контейнер для всей страницы */ .ic-isolated-page { /* Светлая тема (по умолчанию) */ --ic-bg-color: #ffffff; --ic-text-color: #333333; --ic-panel-bg: #fafafa; --ic-border-color: #e0e0e0; --ic-header-bg: #f8f9fa; --ic-header-text: #495057; --ic-controls-bg: #f0f0f0; --ic-search-border: #ddd; --ic-search-focus: #007bff; --ic-arrow-color: #999; --ic-loading-color: #666; --ic-row-border: #f0f0f0; --ic-select-bg: #ffffff; --ic-checkbox-color: #007bff; /* Иконки НЕ меняются в темной теме */ --ic-icon-bg: #f5f5f5; font-family: Arial, sans-serif; background-color: var(--ic-bg-color); color: var(--ic-text-color); padding: 10px; font-size: 11px; min-height: 100vh; width: 100%; overflow-x: hidden; } /* Темная тема - ТОЛЬКО для фона и текста */ .ic-isolated-page[data-ic-theme="dark"] { --ic-bg-color: #1a1a1a; --ic-text-color: #e0e0e0; --ic-panel-bg: #2d2d2d; --ic-border-color: #404040; --ic-header-bg: #3d3d3d; --ic-header-text: #cccccc; --ic-controls-bg: #333333; --ic-search-border: #555; --ic-search-focus: #4dabf7; --ic-arrow-color: #888; --ic-loading-color: #999; --ic-row-border: #3a3a3a; --ic-select-bg: #2d2d2d; --ic-checkbox-color: #4dabf7; /* Иконки остаются со светлым фоном */ --ic-icon-bg: #2a2a2a; }
/* Основные стили */ .ic-isolated-page .ic-container { width: 100%; }
.ic-isolated-page .ic-body { width: 100%; min-height: 100vh; }
.ic-isolated-page .ic-wrapper { max-width: 1400px; margin: 0 auto; width: 100%; padding: 0 10px; }
.ic-isolated-page .ic-title { text-align: center; color: var(--ic-text-color); font-size: 16px; margin-bottom: 15px; font-weight: bold; }
.ic-isolated-page .ic-search-area { margin: 15px 0; text-align: center; }
.ic-isolated-page .ic-search { width: 300px; max-width: 100%; padding: 8px 12px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 14px; background-color: var(--ic-bg-color); color: var(--ic-text-color); transition: all 0.3s ease; }
.ic-isolated-page .ic-search:focus { outline: none; border-color: var(--ic-search-focus); box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25); }
.ic-isolated-page .ic-search::placeholder { color: var(--ic-arrow-color); }
.ic-isolated-page .ic-controls { display: flex; justify-content: center; align-items: center; gap: 20px; margin: 12px 0; padding: 8px; background: var(--ic-controls-bg); border-radius: 4px; border: 1px solid var(--ic-border-color); transition: all 0.3s ease; flex-wrap: wrap; }
.ic-isolated-page .ic-stats { font-size: 12px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-group-select { padding: 6px 10px; border: 1px solid var(--ic-search-border); border-radius: 4px; font-size: 12px; background: var(--ic-select-bg); color: var(--ic-text-color); transition: all 0.3s ease; min-width: 150px; }
.ic-isolated-page .ic-toggle { display: flex; align-items: center; gap: 8px; font-size: 12px; order: 2; color: var(--ic-text-color); }
.ic-isolated-page .ic-checkbox { width: 16px; height: 16px; accent-color: var(--ic-checkbox-color); }
.ic-isolated-page .ic-grid { width: 100%; }
/* Стили для групп с masonry раскладкой */ .ic-isolated-page .ic-group-section { width: 100%; margin-bottom: 15px; }
.ic-isolated-page .ic-group-title { background: var(--ic-header-bg); padding: 8px 12px; border-radius: 4px; margin-bottom: 8px; font-weight: bold; color: var(--ic-header-text); font-size: 13px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; width: 400px; /* Фиксированная ширина как у панелей */ max-width: 100%; flex-shrink: 0; }
.ic-isolated-page .ic-group-content { display: flex; flex-direction: column; width: 400px; /* Фиксированная ширина */ max-width: 100%; }
.ic-isolated-page .ic-panel { border: 1px solid var(--ic-border-color); border-radius: 5px; padding: 8px; background: var(--ic-panel-bg); width: 400px; max-width: 100%; flex-shrink: 0; }
.ic-isolated-page .ic-panel-header { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; margin-bottom: 4px; border-bottom: 1px solid var(--ic-border-color); font-weight: bold; text-align: center; color: var(--ic-text-color); }
.ic-isolated-page .ic-header-label { font-size: 11px; color: var(--ic-text-color); white-space: nowrap; }
.ic-isolated-page .ic-row { display: grid; grid-template-columns: 32px 18px 32px 1fr; gap: 6px; align-items: center; padding: 4px 2px; border-bottom: 1px solid var(--ic-row-border); min-height: 36px; }
.ic-isolated-page .ic-row:last-child { border-bottom: none; }
.ic-isolated-page .ic-image { width: 32px; height: 32px; object-fit: contain; background-color: var(--ic-icon-bg); border-radius: 3px; padding: 0; display: block; filter: none !important; -webkit-filter: none !important; }
.ic-isolated-page .ic-bmp-processed { filter: none !important; -webkit-filter: none !important; }
.ic-isolated-page .ic-arrow { text-align: center; color: var(--ic-arrow-color); font-size: 13px; font-weight: bold; white-space: nowrap; }
.ic-isolated-page .ic-info { padding-left: 6px; padding-right: 3px; min-width: 0; }
.ic-isolated-page .ic-name { color: var(--ic-text-color); line-height: 1.3; word-wrap: break-word; overflow-wrap: break-word; hyphens: auto; text-align: left; font-size: 12px; max-width: 220px; }
.ic-isolated-page .ic-loading { text-align: center; padding: 40px 20px; color: var(--ic-loading-color); width: 100%; }
.ic-isolated-page .ic-no-results { text-align: center; padding: 40px; color: var(--ic-loading-color); width: 100%; font-size: 14px; }
/* Masonry колонки */ .ic-isolated-page .ic-masonry-columns { display: flex; flex-wrap: wrap; gap: 15px; width: 100%; justify-content: center; }
.ic-isolated-page .ic-masonry-column { display: flex; flex-direction: column; gap: 15px; width: 400px; /* Фиксированная ширина колонки */ max-width: 100%; }
/* Обычные колонки (без групп) */ .ic-isolated-page .ic-columns { display: flex; flex-wrap: wrap; gap: 15px; width: 100%; justify-content: center; }
/* Адаптивность */ @media (max-width: 768px) { .ic-isolated-page .ic-wrapper { max-width: 100%; padding: 0 10px; } .ic-isolated-page .ic-search { width: 100%; max-width: 300px; } .ic-isolated-page .ic-controls { gap: 10px; } .ic-isolated-page .ic-group-select { min-width: 120px; } .ic-isolated-page .ic-name { max-width: 180px; } .ic-isolated-page .ic-group-title, .ic-isolated-page .ic-group-content, .ic-isolated-page .ic-panel, .ic-isolated-page .ic-masonry-column { width: 100%; max-width: 400px; } }
/* Для очень маленьких экранов */ @media (max-width: 480px) { .ic-isolated-page .ic-panel-header { grid-template-columns: 30px 16px 30px 1fr; gap: 4px; } .ic-isolated-page .ic-header-label { font-size: 10px; } .ic-isolated-page .ic-arrow { font-size: 12px; } .ic-isolated-page .ic-name { font-size: 11px; max-width: 150px; } .ic-isolated-page .ic-group-title { font-size: 12px; padding: 6px 10px; } }
/* Для ширины 800px и выше - 2 колонки */ @media (min-width: 800px) { .ic-isolated-page .ic-masonry-column { width: calc(50% - 8px); max-width: 400px; } }
/* Для ширины 1200px и выше - 3 колонки */ @media (min-width: 1200px) { .ic-isolated-page .ic-wrapper { max-width: 1200px; } .ic-isolated-page .ic-masonry-column { width: calc(33.333% - 10px); max-width: 400px; } }
/* Для ширины 1400px и выше - 4 колонки */ @media (min-width: 1400px) { .ic-isolated-page .ic-wrapper { max-width: 1400px; } .ic-isolated-page .ic-masonry-column { width: calc(25% - 12px); max-width: 400px; } } </style> </head> <body class="ic-isolated-page"> <div class="ic-container"> <div class="ic-body"> <div class="ic-wrapper"> <h1 class="ic-title">Сравнение иконок ИСИХОГИ старой и новой версий</h1> <div class="ic-search-area"> <input type="text" id="icSearch" class="ic-search" placeholder="Поиск по названию иконки..." autocomplete="off"> </div> <div class="ic-controls"> <div class="ic-stats"> Всего иконок для сравнения: <span id="icTotalCount">0</span> | Отображено: <span id="icShownCount">0</span> </div> <select id="icGroupSelect" class="ic-group-select" style="display: none;"> <option value="">Все группы</option> <option value="no_group">Прочие</option> </select> <div class="ic-toggle"> <input type="checkbox" id="icShowGroups" class="ic-checkbox" checked> <label for="icShowGroups">Показывать группы</label> </div> </div>
<div class="ic-grid" id="icContainer"> <div class="ic-loading">Загрузка иконок...</div> </div> </div> </div> </div>
<script> (function() { 'use strict'; let icAllIcons = []; let icFilteredIcons = []; let icAvailableGroups = []; let icShowGroups = true; let icThemeCheckInterval = null; let icResizeTimeout = null; let icSearchTimeout = null;
// Функция для определения темы Hugo function icDetectHugoTheme() { const html = document.documentElement; const body = document.body; if (html.getAttribute('data-theme') === 'dark' || html.getAttribute('data-mode') === 'dark' || html.classList.contains('dark') || html.classList.contains('dark-mode') || body.classList.contains('dark') || body.classList.contains('dark-theme') || body.classList.contains('dark-mode') || document.querySelector('.dark-mode, .dark-theme, [data-theme="dark"]')) { return 'dark'; } return 'light'; }
// Функция для применения темы function icApplyTheme() { try { const theme = icDetectHugoTheme(); const page = document.querySelector('.ic-isolated-page'); if (!page) return; if (theme === 'dark') { page.setAttribute('data-ic-theme', 'dark'); } else { page.removeAttribute('data-ic-theme'); } } catch (error) { console.warn('Ошибка применения темы:', error); } }
// Простой наблюдатель за темой function icSetupThemeObserver() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); } let lastTheme = icDetectHugoTheme(); icThemeCheckInterval = setInterval(() => { const currentTheme = icDetectHugoTheme(); if (currentTheme !== lastTheme) { lastTheme = currentTheme; icApplyTheme(); } }, 1000); icApplyTheme(); }
// Функция для разбивки больших групп на части function icSplitLargeGroup(icons, maxItems = 15) { if (icons.length <= maxItems) { return [icons]; } const chunks = []; for (let i = 0; i < icons.length; i += maxItems) { chunks.push(icons.slice(i, i + maxItems)); } return chunks; }
// Функция для расчета примерной высоты группы function icCalculateGroupHeight(iconsCount) { const headerHeight = 36; const rowHeight = 30; const padding = 12; return headerHeight + (iconsCount * rowHeight) + padding; }
// Функция для создания адаптивной masonry компоновки - ВОЗВРАЩАЕМ ОРИГИНАЛЬНУЮ ЛОГИКУ function icCreateMasonryLayout(groupsChunks) { const container = document.getElementById('icContainer'); if (!container) return []; const containerWidth = container.clientWidth; const columnWidth = 400 + 15; // Ширина панели + отступ // Динамическое количество колонок let maxColumns; if (containerWidth >= 1400) { maxColumns = 4; } else if (containerWidth >= 1200) { maxColumns = 3; } else if (containerWidth >= 800) { maxColumns = 2; } else { maxColumns = 1; } const availableColumns = Math.min(maxColumns, Math.max(1, Math.floor(containerWidth / columnWidth))); // Создаем колонки const columns = Array.from({ length: availableColumns }, () => { const col = document.createElement('div'); col.className = 'ic-masonry-column'; return col; }); // Массив для отслеживания высоты каждой колонки const columnHeights = Array(availableColumns).fill(0); // Распределяем группы по колонкам (алгоритм "наименьшая высота") groupsChunks.forEach(group => { let minHeight = Math.min(...columnHeights); let columnIndex = columnHeights.indexOf(minHeight); const section = icCreateGroupSection(group.name, group.icons); columns[columnIndex].appendChild(section); columnHeights[columnIndex] += group.height; }); return columns; }
// Функция для отображения иконок с группировкой function icRenderWithGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } // Группируем иконки const { groups, noGroup } = icGroupIcons(icFilteredIcons); // Собираем все группы и разбиваем большие const allGroupChunks = []; Object.entries(groups).forEach(([groupName, icons]) => { const chunks = icSplitLargeGroup(icons, 15); chunks.forEach((chunk, index) => { const chunkName = chunks.length > 1 ? `${groupName} (${index + 1}/${chunks.length})` : groupName; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: groupName, height: icCalculateGroupHeight(chunk.length) }); }); }); // Добавляем иконки без группы как "Прочие" if (noGroup.length > 0) { const noGroupChunks = icSplitLargeGroup(noGroup, 15); noGroupChunks.forEach((chunk, index) => { const chunkName = noGroupChunks.length > 1 ? `Прочие иконки (${index + 1}/${noGroupChunks.length})` : 'Прочие иконки'; allGroupChunks.push({ name: chunkName, icons: chunk, originalGroup: 'no_group', height: icCalculateGroupHeight(chunk.length) }); }); } // Сортируем по высоте (от высоких к низким для лучшего заполнения) allGroupChunks.sort((a, b) => b.height - a.height); // Создаем адаптивную masonry компоновку const columns = icCreateMasonryLayout(allGroupChunks); // Очищаем контейнер и добавляем колонки container.innerHTML = ''; const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-masonry-columns'; columns.forEach(col => { if (col.children.length > 0) { columnsContainer.appendChild(col); } }); container.appendChild(columnsContainer); icUpdateStats(); }
// Функция для создания секции группы - ВСЕ ЭЛЕМЕНТЫ В ОДНОМ КОНТЕЙНЕРЕ function icCreateGroupSection(groupName, icons) { const section = document.createElement('div'); section.className = 'ic-group-section'; // Заголовок группы с ФИКСИРОВАННОЙ шириной как у панели const header = document.createElement('div'); header.className = 'ic-group-title'; header.textContent = groupName; header.title = groupName; section.appendChild(header); // Контейнер для панели const content = document.createElement('div'); content.className = 'ic-group-content'; const panel = icCreateIconColumn(icons); content.appendChild(panel); section.appendChild(content); return section; }
// Функция для создания колонки с иконками function icCreateIconColumn(icons) { const column = document.createElement('div'); column.className = 'ic-panel'; const header = document.createElement('div'); header.className = 'ic-panel-header'; const beforeLabel = document.createElement('div'); beforeLabel.className = 'ic-header-label'; beforeLabel.textContent = 'ДО'; const arrowSpace = document.createElement('div'); arrowSpace.className = 'ic-arrow'; arrowSpace.textContent = '→'; const afterLabel = document.createElement('div'); afterLabel.className = 'ic-header-label'; afterLabel.textContent = 'ПОСЛЕ'; const nameLabel = document.createElement('div'); nameLabel.className = 'ic-header-label'; nameLabel.textContent = 'Название'; nameLabel.style.textAlign = 'center'; header.appendChild(beforeLabel); header.appendChild(arrowSpace); header.appendChild(afterLabel); header.appendChild(nameLabel); column.appendChild(header); icons.forEach(icon => { column.appendChild(icCreateIconRow(icon)); }); return column; }
// Обработчик изменения размера окна function icSetupResizeHandler() { let resizeTimeout; window.addEventListener('resize', function() { clearTimeout(resizeTimeout); resizeTimeout = setTimeout(() => { if (icShowGroups && icFilteredIcons.length > 0) { icRenderWithGroups(); } }, 250); }); }
// Упрощенная функция для обработки BMP function icProcessBMP(img) { return new Promise((resolve) => { try { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = img.naturalWidth || 32; canvas.height = img.naturalHeight || 32; ctx.drawImage(img, 0, 0); // Убираем только magenta (#FF00FF) const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height); const data = imageData.data; for (let i = 0; i < data.length; i += 4) { if (data[i] === 255 && data[i + 1] === 0 && data[i + 2] === 255) { data[i + 3] = 0; } } ctx.putImageData(imageData, 0, 0); const processedImg = new Image(); processedImg.onload = () => resolve({ element: processedImg, processed: true }); processedImg.src = canvas.toDataURL('image/png'); } catch (error) { resolve({ element: img, processed: false }); } }); }
// Функция загрузки изображений function icLoadImage(src, alt) { return new Promise((resolve) => { const img = new Image(); img.onload = async function() { try { if (src.toLowerCase().endsWith('.bmp')) { const result = await icProcessBMP(img); result.element.alt = alt; result.element.width = 32; result.element.height = 32; resolve(result); } else { img.alt = alt; img.width = 32; img.height = 32; resolve({ element: img, processed: false }); } } catch (error) { img.alt = alt; img.width = 32; img.height = 32; resolve({ element: img, processed: false }); } }; img.onerror = function() { const placeholder = new Image(); placeholder.width = 32; placeholder.height = 32; placeholder.src = ''; placeholder.alt = alt; resolve({ element: placeholder, processed: false }); }; img.src = src; }); }
function icGroupIcons(icons) { const groups = {}; const noGroup = []; icons.forEach(icon => { if (icon.group && icon.group.trim() !== '') { if (!groups[icon.group]) { groups[icon.group] = []; } groups[icon.group].push(icon); } else { noGroup.push(icon); } }); return { groups, noGroup }; }
function icCreateIconRow(icon) { const row = document.createElement('div'); row.className = 'ic-row'; const beforeContainer = document.createElement('div'); const afterContainer = document.createElement('div'); Promise.all([ icLoadImage(icon.before, `${icon.name} (до)`), icLoadImage(icon.after, `${icon.name} (после)`) ]).then(([beforeResult, afterResult]) => { const beforeImg = beforeResult.element; const afterImg = afterResult.element; beforeImg.className = 'ic-image'; afterImg.className = 'ic-image'; if (beforeResult.processed) beforeImg.classList.add('ic-bmp-processed'); if (afterResult.processed) afterImg.classList.add('ic-bmp-processed'); beforeImg.width = 32; beforeImg.height = 32; beforeImg.style.width = '32px'; beforeImg.style.height = '32px'; afterImg.width = 32; afterImg.height = 32; afterImg.style.width = '32px'; afterImg.style.height = '32px'; beforeContainer.appendChild(beforeImg); afterContainer.appendChild(afterImg); }); const arrow = document.createElement('div'); arrow.className = 'ic-arrow'; arrow.textContent = '→'; const infoDiv = document.createElement('div'); infoDiv.className = 'ic-info'; const name = document.createElement('div'); name.className = 'ic-name'; name.textContent = icon.name; name.title = icon.name; infoDiv.appendChild(name); row.appendChild(beforeContainer); row.appendChild(arrow); row.appendChild(afterContainer); row.appendChild(infoDiv); return row; }
function icRenderWithoutGroups() { const container = document.getElementById('icContainer'); container.innerHTML = ''; if (icFilteredIcons.length === 0) { container.innerHTML = '<div class="ic-no-results">Ничего не найдено. Попробуйте изменить поисковый запрос или выбрать другую группу.</div>'; icUpdateStats(); return; } const sortedIcons = [...icFilteredIcons].sort((a, b) => { const groupA = a.group || 'zzz'; const groupB = b.group || 'zzz'; if (groupA !== groupB) { return groupA.localeCompare(groupB); } return a.name.localeCompare(b.name); }); const itemsPerColumn = 20; const columnCount = Math.ceil(sortedIcons.length / itemsPerColumn); const columnsContainer = document.createElement('div'); columnsContainer.className = 'ic-columns'; for (let i = 0; i < columnCount; i++) { const startIndex = i * itemsPerColumn; const endIndex = startIndex + itemsPerColumn; const columnIcons = sortedIcons.slice(startIndex, endIndex); const column = icCreateIconColumn(columnIcons); columnsContainer.appendChild(column); } container.appendChild(columnsContainer); icUpdateStats(); }
function icFilterIcons(searchTerm, selectedGroup = '') { let filtered = [...icAllIcons]; if (searchTerm) { const term = searchTerm.toLowerCase(); filtered = filtered.filter(icon => icon.name.toLowerCase().includes(term) || (icon.group && icon.group.toLowerCase().includes(term)) ); } if (selectedGroup) { if (selectedGroup === 'no_group') { filtered = filtered.filter(icon => !icon.group || icon.group.trim() === ''); } else { filtered = filtered.filter(icon => icon.group === selectedGroup); } } icFilteredIcons = filtered; icRenderFilteredIcons(); }
function icRenderFilteredIcons() { if (icShowGroups) { icRenderWithGroups(); } else { icRenderWithoutGroups(); } }
function icUpdateStats() { const totalCount = document.getElementById('icTotalCount'); const shownCount = document.getElementById('icShownCount'); if (totalCount) totalCount.textContent = icAllIcons.length; if (shownCount) shownCount.textContent = icFilteredIcons.length; }
function icInitGroupSelect(groups) { const groupSelect = document.getElementById('icGroupSelect'); if (!groupSelect) return; groupSelect.innerHTML = '<option value="">Все группы</option>'; groupSelect.innerHTML += '<option value="no_group">Прочие</option>'; groups.forEach(group => { const option = document.createElement('option'); option.value = group; option.textContent = group; groupSelect.appendChild(option); }); groupSelect.style.display = (groups.length > 0) ? 'block' : 'none'; groupSelect.addEventListener('change', function() { const searchTerm = document.getElementById('icSearch').value; icFilterIcons(searchTerm, this.value); }); }
function icRenderIcons(images, groups) { icAllIcons = images; icFilteredIcons = [...icAllIcons]; icAvailableGroups = groups; icInitGroupSelect(groups); icRenderFilteredIcons(); }
async function icLoadIcons() { try { const script = document.createElement('script'); script.src = 'image_list.js'; document.head.appendChild(script); script.onload = function() { try { const data = loadBeforeAfterImages(); if (data && data.images) { const processedImages = data.images.map(icon => { const displayName = icon.name.replace(/^32x32_/, ''); return { ...icon, name: displayName }; }); icRenderIcons(processedImages, data.availableGroups || []); } else { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки данных</div>'; } } catch (error) { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка обработки данных</div>'; } }; script.onerror = function() { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки файла image_list.js</div>'; }; } catch (error) { const container = document.getElementById('icContainer'); if (container) container.innerHTML = '<div class="ic-loading">Ошибка загрузки иконок</div>'; } }
function icSetupSearch() { const searchInput = document.getElementById('icSearch'); if (!searchInput) return; let searchTimeout; searchInput.addEventListener('input', function(e) { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons(e.target.value, selectedGroup); }, 300); }); searchInput.addEventListener('keydown', function(e) { if (e.key === 'Escape') { this.value = ''; const selectedGroup = document.getElementById('icGroupSelect').value; icFilterIcons('', selectedGroup); } }); }
function icSetupToggle() { const toggle = document.getElementById('icShowGroups'); if (!toggle) return; toggle.addEventListener('change', function() { icShowGroups = this.checked; icRenderFilteredIcons(); }); }
// Очистка при разгрузке function icCleanup() { if (icThemeCheckInterval) { clearInterval(icThemeCheckInterval); icThemeCheckInterval = null; } clearTimeout(icResizeTimeout); clearTimeout(icSearchTimeout); }
// Инициализация document.addEventListener('DOMContentLoaded', function() { try { icSetupThemeObserver(); icLoadIcons(); icSetupSearch(); icSetupToggle(); icSetupResizeHandler(); window.addEventListener('beforeunload', icCleanup); window.addEventListener('pagehide', icCleanup); } catch (error) { console.error('Ошибка инициализации:', error); } }); })(); </script> </body> </html>
|