Add favicon and logo generation scripts

- Implemented `generate-favicons.js` to create various favicon formats from a single SVG file.
- Created `update-data.js` to scan logo directories, generate PNG and JPG images from SVGs, and update the logos JSON data.
- Added a `.gitignore` file in the `flags_gen` directory to exclude all files except the `.gitignore` itself.
- Updated `logos.json` to ensure proper formatting.
This commit is contained in:
sHa
2025-06-17 16:15:15 +03:00
parent cff94aacb8
commit 8b2adef1af
8 changed files with 2078 additions and 35 deletions

View File

@@ -32,7 +32,7 @@ jobs:
run: npm run build run: npm run build
- name: Generate logos.json - name: Generate logos.json
run: node scripts/scanLogos.js run: node scripts/update-data.js
- name: Generate SVG variants - name: Generate SVG variants
run: node scripts/generate-svg-variants.js run: node scripts/generate-svg-variants.js

View File

@@ -6,7 +6,7 @@ CONTAINER_NAME = slogos-dev
DEV_PORT = 5006 DEV_PORT = 5006
# Main targets # Main targets
.PHONY: all build start stop restart logs clean scan-logos dev rebuild favicon build-with-favicons generate-svg-variants pwa-cache-list run update-lock .PHONY: all build start stop restart logs clean update-data dev rebuild favicon build-with-favicons generate-svg-variants pwa-cache-list run update-lock
all: build start all: build start
@@ -42,11 +42,10 @@ run:
@echo "Running command in container: $(CMD)" @echo "Running command in container: $(CMD)"
$(DOCKER_COMPOSE) -f compose.dev.yml run --rm $(CONTAINER_NAME) $(CMD) $(DOCKER_COMPOSE) -f compose.dev.yml run --rm $(CONTAINER_NAME) $(CMD)
# Scan logos.json from files in the logos directory (for dev mode) update-data:
scan-logos:
@echo "Scanning logos directory and updating logos.json for development..." @echo "Scanning logos directory and updating logos.json for development..."
$(DOCKER_COMPOSE) -f compose.dev.yml run --rm $(CONTAINER_NAME) npm run scan-logos $(DOCKER_COMPOSE) -f compose.dev.yml run --rm $(CONTAINER_NAME) npm run update-data
@echo "Logos have been updated - refresh the browser to see changes" @echo "Logos and Flags have been updated - refresh the browser to see changes"
# Clean up build artifacts and temporary files # Clean up build artifacts and temporary files
clean: clean:

View File

@@ -6,8 +6,8 @@
"build": "rollup -c", "build": "rollup -c",
"dev": "rollup -c -w", "dev": "rollup -c -w",
"start": "sirv public --host 0.0.0.0 --dev --single", "start": "sirv public --host 0.0.0.0 --dev --single",
"scan-logos": "node scripts/scanLogos.js", "update-data": "node scripts/update-data.js",
"generate-favicons": "node scripts/generateFavicons.js", "generate-favicons": "node scripts/generate-favicons.js",
"generate-variants": "node scripts/generate-svg-variants.js", "generate-variants": "node scripts/generate-svg-variants.js",
"variants": "npm run generate-variants", "variants": "npm run generate-variants",
"pwa-cache-list": "node scripts/generate-pwa-cache-list.js" "pwa-cache-list": "node scripts/generate-pwa-cache-list.js"

1912
public/data/flags.json Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -3589,4 +3589,4 @@
], ],
"brand": "Bvr" "brand": "Bvr"
} }
] ]

2
public/flags_gen/.gitignore vendored Normal file
View File

@@ -0,0 +1,2 @@
*
!.gitignore

View File

@@ -5,13 +5,31 @@ const path = require('path');
const { Resvg } = require('@resvg/resvg-js'); const { Resvg } = require('@resvg/resvg-js');
// Configuration // Configuration
const logosDir = path.join(__dirname, '../public/logos'); const collections = [
const outputFile = path.join(__dirname, '../public/data/logos.json'); { name: 'logos', label: 'Logos',
const genDir = path.join(__dirname, '../public/logos_gen'); baseDir: 'logos',
genDir: 'logos_gen',
dataFile: 'data/logos.json'
},
{ name: 'flags', label: 'Flags',
baseDir: 'flags',
genDir: 'flags_gen',
dataFile: 'data/flags.json'
}
];
// Accept collection as a CLI arg or env var
const collectionArg = process.argv.find(arg => arg.startsWith('--collection='));
const collectionName = collectionArg ? collectionArg.split('=')[1] : (process.env.COLLECTION || 'logos');
const collection = collections.find(c => c.name === collectionName) || collections[0];
const logosDir = path.join(__dirname, '..', 'public', collection.baseDir);
const outputFile = path.join(__dirname, '..', 'public', collection.dataFile);
const genDir = path.join(__dirname, '..', 'public', collection.genDir);
// Remove old PNG/JPG folders if they exist // Remove old PNG/JPG folders if they exist
const pngDir = path.join(__dirname, '../public/logos-png'); const pngDir = path.join(__dirname, '..', 'public', collection.baseDir + '-png');
const jpgDir = path.join(__dirname, '../public/logos-jpg'); const jpgDir = path.join(__dirname, '..', 'public', collection.baseDir + '-jpg');
if (fs.existsSync(pngDir)) fs.rmSync(pngDir, { recursive: true, force: true }); if (fs.existsSync(pngDir)) fs.rmSync(pngDir, { recursive: true, force: true });
if (fs.existsSync(jpgDir)) fs.rmSync(jpgDir, { recursive: true, force: true }); if (fs.existsSync(jpgDir)) fs.rmSync(jpgDir, { recursive: true, force: true });
@@ -73,23 +91,23 @@ function svgToJpg(svgBuffer, width, height) {
} }
// Pregenerate PNG and JPG images for SVG files // Pregenerate PNG and JPG images for SVG files
function pregenerateImages(logoFiles) { function pregenerateImages(logoFiles, logosDir, genDir) {
cleanDir(genDir); cleanDir(genDir);
for (const file of logoFiles) { // Only process SVG files
if (/\.svg$/i.test(file)) { const svgFiles = logoFiles.filter(file => /\.svg$/i.test(file));
const base = getBaseName(file); for (const file of svgFiles) {
const svgPath = path.join(logosDir, file); const base = getBaseName(file);
const pngPath = path.join(genDir, base + '.png'); const svgPath = path.join(logosDir, file);
const jpgPath = path.join(genDir, base + '.jpg'); const pngPath = path.join(genDir, base + '.png');
try { const jpgPath = path.join(genDir, base + '.jpg');
const svgBuffer = fs.readFileSync(svgPath); try {
const pngBuffer = svgToPng(svgBuffer, 256, 256); const svgBuffer = fs.readFileSync(svgPath);
fs.writeFileSync(pngPath, pngBuffer); const pngBuffer = svgToPng(svgBuffer, 256, 256);
const jpgBuffer = svgToJpg(svgBuffer); fs.writeFileSync(pngPath, pngBuffer);
fs.writeFileSync(jpgPath, jpgBuffer); const jpgBuffer = svgToJpg(svgBuffer);
} catch (e) { fs.writeFileSync(jpgPath, jpgBuffer);
console.error('Error generating PNG/JPG for', file, e); } catch (e) {
} console.error('Error generating PNG/JPG for', file, e);
} }
} }
} }
@@ -235,11 +253,123 @@ function saveLogosToJson(logos) {
// Main function // Main function
function main() { function main() {
const logos = scanLogos(); // If no collection is specified, process all collections
// Pregenerate PNG/JPG for all SVGs if (!collectionArg && !process.env.COLLECTION) {
const files = fs.readdirSync(logosDir); for (const col of collections) {
pregenerateImages(files); const logosDir = path.join(__dirname, '..', 'public', col.baseDir);
saveLogosToJson(logos); const outputFile = path.join(__dirname, '..', 'public', col.dataFile);
const genDir = path.join(__dirname, '..', 'public', col.genDir);
if (!fs.existsSync(logosDir)) {
fs.mkdirSync(logosDir, { recursive: true });
}
const files = fs.readdirSync(logosDir);
const logos = (function scanLogosForCol() {
let existing = [];
if (fs.existsSync(outputFile)) {
try {
existing = JSON.parse(fs.readFileSync(outputFile, 'utf8'));
} catch (e) {
console.error('Could not parse existing', col.dataFile + ':', e);
}
}
try {
if (!fs.existsSync(logosDir)) {
console.error(`Directory does not exist: ${logosDir}`);
return [];
}
// Filter for image files (svg, png, jpg, jpeg)
const logoFiles = files.filter(file =>
/\.(svg|png|jpg|jpeg)$/i.test(file)
);
// Create a Set of all logo paths in the directory
const logoPathsSet = new Set(logoFiles.map(file => `${col.baseDir}/${file}`));
// Mark existing records as disabled if they are not found in the directory
for (const logo of existing) {
if (!logoPathsSet.has(logo.path)) {
logo.disable = true;
}
}
// Create a Set of existing paths to avoid duplication
const existingPathsSet = new Set(existing.map(logo => logo.path));
// Create new minimal logo objects for files that don't have records yet
const newLogos = logoFiles
.filter(file => !existingPathsSet.has(`${col.baseDir}/${file}`))
.map(file => {
const format = getFileExtension(file);
const logoPath = `${col.baseDir}/${file}`;
return {
name: formatName(file),
path: logoPath,
format: format,
disable: false,
brand: formatName(file)
};
})
.sort((a, b) => a.name.localeCompare(b.name));
let merged = [...existing];
for (const newLogo of newLogos) {
let insertIdx = merged.findIndex(l => newLogo.name.localeCompare(l.name) < 0);
if (insertIdx === -1) {
merged.push(newLogo);
} else {
merged.splice(insertIdx, 0, newLogo);
}
}
for (const logoObj of merged) {
if (logoObj.format?.toLowerCase() === 'svg' &&
logoObj.colors &&
Object.keys(logoObj.colors).length > 0 &&
logoObj.colorConfig &&
!logoObj.sets) {
if (!logoObj.targets) {
logoObj.targets = {};
if (logoObj.colorConfig.selector) {
const selectors = logoObj.colorConfig.selector.split(',').map(s => s.trim());
selectors.forEach((selector, index) => {
logoObj.targets[`selector_${index + 1}`] = selector;
});
} else if (logoObj.colorConfig.target) {
logoObj.targets.main = logoObj.colorConfig.target;
}
}
if (logoObj.targets && Object.keys(logoObj.targets).length > 0) {
logoObj.sets = {};
let setIndex = 1;
for (const [colorName, colorValue] of Object.entries(logoObj.colors)) {
const setName = `set_${setIndex}`;
logoObj.sets[setName] = {};
Object.keys(logoObj.targets).forEach(targetName => {
logoObj.sets[setName][targetName] = colorName;
});
setIndex++;
}
}
}
}
return merged;
} catch (error) {
console.error('Error scanning directory:', logosDir, error);
return [];
}
})();
// Pregenerate PNG/JPG for all SVGs
pregenerateImages(files, logosDir, genDir);
// Save to JSON
try {
const data = JSON.stringify(logos, null, 2);
fs.writeFileSync(outputFile, data);
console.log(`Successfully wrote ${logos.length} items to ${outputFile}`);
} catch (error) {
console.error('Error writing data to file:', outputFile, error);
}
}
} else {
// Single collection mode (as before)
const logos = scanLogos();
const files = fs.readdirSync(logosDir);
pregenerateImages(files, logosDir, genDir);
saveLogosToJson(logos);
}
} }
// Run the script // Run the script