Files
trainingstagebuch/scripts/apply-cognate-overrides.js
Torsten Schulz (local) eb54b4f7cf
All checks were successful
Deploy tt-tagebuch / deploy (push) Successful in 45s
feat(i18n): add scripts for locale translation and patching
- Implemented `fill-de-extended-gaps.js` to fill missing billing/orders keys in de-extended from de.
- Created `fill-i18n-deep.py` for deep translation of locale JSONs using deep-translator with fallback options.
- Added `fill-i18n-locales.js` to translate locale JSONs and write overrides for untranslated keys.
- Introduced `fix-en-leaks.py` to translate keys that still match the en-US merge, addressing English leaks.
- Developed `patch-de-ch-swiss.js` to replace 'ß' with 'ss' in de-CH.json without deleting existing entries.
- Created `patch-en-gb-au.js` to apply UK/AU spelling corrections in en-GB and en-AU locales.
- Added shell scripts `run-fix-en-leaks.sh` and `run-i18n-deep-fill.sh` for sequential execution of translation tasks.
- Implemented `update-i18n-todo-stats.js` to update statistics in the I18N_TODO.md file based on translation completeness.
2026-05-15 15:52:54 +02:00

125 lines
4.6 KiB
JavaScript

#!/usr/bin/env node
/**
* Setzt bewusste Cognates/Produktnamen explizit in allen Locales (Wert = en-US oder de).
*/
const fs = require('fs');
const path = require('path');
const ROOT = path.resolve(__dirname, '..');
const LOCALES_DIR = path.join(ROOT, 'frontend', 'src', 'i18n', 'locales');
const LOCALES = [
'fr', 'es', 'it', 'pl', 'ja', 'zh', 'th', 'tl', 'fil',
'en-US', 'en-GB', 'en-AU',
];
// Keys, die in Nicht-DE-Locales explizit gesetzt werden (Wert aus en-US-Merge)
const COGNATE_KEYS = [
'common.filter', 'common.details', 'common.name', 'common.status',
'common.optional', 'common.in', 'common.ok',
'navigation.clickTtBrowser',
'members.rowTooltipSeparator', 'members.ttrQttr', 'members.status',
'members.clickTtRequestShort',
'diary.standardDurationShort', 'diary.min',
'trainingStats.name', 'trainingStats.ttr', 'trainingStats.qttr',
'tournaments.knockoutLabel', 'tournaments.genderMixed', 'tournaments.optional',
'tournaments.index', 'tournaments.diff',
'billing.iban',
'schedule.code', 'schedule.team', 'schedule.teams', 'schedule.vs',
'teamManagement.team', 'teamManagement.teams',
'courtDrawingTool.strokeTypeTopspin', 'courtDrawingTool.strokeTypeFlip',
'courtDrawingTool.strokeTypeBlock',
'matchReportApi.vs', 'courtDrawing.ok',
'pdfGenerator.diff', 'pdfGenerator.lineupQttr',
'pendingApprovals.email', 'permissions.email',
'logs.details', 'logs.logDetailsLabels.ip',
'dialogs.info.ok', 'dialogs.confirm.ok',
'memberTransferDialog.server', 'memberTransferDialog.endpoint', 'memberTransferDialog.format',
'myTischtennis.title', 'myTischtennisAccount.title', 'myTischtennisAccount.teams',
'memberNotes.tags', 'nuscoreAnalyzer.title',
'predefinedActivities.min',
];
const EXTRA = {
fr: { 'members.address': 'Adresse' },
es: { 'members.address': 'Dirección' },
it: { 'members.address': 'Indirizzo' },
pl: { 'members.address': 'Adres' },
ja: { 'members.address': '住所' },
zh: { 'members.address': '地址' },
th: { 'members.address': 'ที่อยู่' },
tl: { 'members.address': 'Address', 'languages.th': 'Thai (ไทย)' },
fil: { 'members.address': 'Address', 'languages.th': 'Thai (ไทย)' },
};
function flatten(obj, prefix = '', out = {}) {
for (const [key, value] of Object.entries(obj || {})) {
const nextKey = prefix ? `${prefix}.${key}` : key;
if (value && typeof value === 'object' && !Array.isArray(value)) {
flatten(value, nextKey, out);
} else if (typeof value === 'string') {
out[nextKey] = value;
}
}
return out;
}
function deepMerge(base, override) {
if (!base || typeof base !== 'object' || Array.isArray(base)) return override ?? base;
const result = { ...base };
for (const [key, value] of Object.entries(override || {})) {
if (
value && typeof value === 'object' && !Array.isArray(value) &&
result[key] && typeof result[key] === 'object' && !Array.isArray(result[key])
) {
result[key] = deepMerge(result[key], value);
} else {
result[key] = value;
}
}
return result;
}
function setByPath(obj, dotPath, value) {
const parts = dotPath.split('.');
let cur = obj;
for (let i = 0; i < parts.length - 1; i++) {
if (!cur[parts[i]] || typeof cur[parts[i]] !== 'object') cur[parts[i]] = {};
cur = cur[parts[i]];
}
cur[parts[parts.length - 1]] = value;
}
function buildOverrides(deFlat, targetFlat) {
const out = {};
for (const [key, value] of Object.entries(targetFlat)) {
if (value !== deFlat[key]) setByPath(out, key, value);
}
return out;
}
const de = JSON.parse(fs.readFileSync(path.join(LOCALES_DIR, 'de.json'), 'utf8'));
const enUs = JSON.parse(fs.readFileSync(path.join(LOCALES_DIR, 'en-US.json'), 'utf8'));
const deFlat = flatten(de);
const enMerged = flatten(deepMerge(JSON.parse(JSON.stringify(de)), enUs));
for (const locale of LOCALES) {
const localePath = path.join(LOCALES_DIR, `${locale}.json`);
if (!fs.existsSync(localePath)) continue;
const localeJson = JSON.parse(fs.readFileSync(localePath, 'utf8'));
const merged = flatten(deepMerge(JSON.parse(JSON.stringify(de)), localeJson));
for (const key of COGNATE_KEYS) {
const val = enMerged[key] ?? deFlat[key];
if (val !== undefined) merged[key] = val;
}
for (const [key, val] of Object.entries(EXTRA[locale] || {})) {
merged[key] = val;
}
const overrides = buildOverrides(deFlat, merged);
fs.writeFileSync(localePath, `${JSON.stringify(overrides, null, 2)}\n`, 'utf8');
const stillDe = Object.keys(deFlat).filter((k) => merged[k] === deFlat[k]).length;
console.log(`${locale}: overrides=${Object.keys(flatten(overrides)).length}, stillDe=${stillDe}`);
}