330 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			330 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
#!/usr/bin/env php
 | 
						|
<?php
 | 
						|
 | 
						|
/**
 | 
						|
 * Format a language file in the same way as the EN equivalent.
 | 
						|
 * Matches the line numbers of translated content.
 | 
						|
 * Potentially destructive, Ensure you have a backup of your translation content before running.
 | 
						|
 */
 | 
						|
 | 
						|
$args = array_slice($argv, 1);
 | 
						|
 | 
						|
if (count($args) < 2) {
 | 
						|
    errorOut("Please provide a language code as the first argument and a translation file name, or '--all', as the second (./format.php fr activities)");
 | 
						|
}
 | 
						|
 | 
						|
$lang = formatLocale($args[0]);
 | 
						|
$fileName = explode('.', $args[1])[0];
 | 
						|
$fileNames = [$fileName];
 | 
						|
if ($fileName === '--all') {
 | 
						|
    $fileNames = getTranslationFileNames();
 | 
						|
}
 | 
						|
 | 
						|
foreach ($fileNames as $fileName) {
 | 
						|
    $formatted = formatFileContents($lang, $fileName);
 | 
						|
    writeLangFile($lang, $fileName, $formatted);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
 * Format the contents of a single translation file in the given language.
 | 
						|
 * @param string $lang
 | 
						|
 * @param string $fileName
 | 
						|
 * @return string
 | 
						|
 */
 | 
						|
function formatFileContents(string $lang, string $fileName) : string {
 | 
						|
    $enLines = loadLangFileLines('en', $fileName);
 | 
						|
    $langContent = loadLang($lang, $fileName);
 | 
						|
    $enContent = loadLang('en', $fileName);
 | 
						|
 | 
						|
    // Calculate the longest top-level key length
 | 
						|
    $longestKeyLength = calculateKeyPadding($enContent);
 | 
						|
 | 
						|
    // Start formatted content
 | 
						|
    $formatted = [];
 | 
						|
    $mode = 'header';
 | 
						|
    $skipArray = false;
 | 
						|
    $arrayKeys = [];
 | 
						|
 | 
						|
    foreach ($enLines as $index => $line) {
 | 
						|
        $trimLine = trim($line);
 | 
						|
        if ($mode === 'header') {
 | 
						|
            $formatted[$index] = $line;
 | 
						|
            if (str_replace(' ', '', $trimLine) === 'return[') $mode = 'body';
 | 
						|
        }
 | 
						|
 | 
						|
        if ($mode === 'body') {
 | 
						|
            $matches = [];
 | 
						|
            $arrayEndMatch = preg_match('/]\s*,\s*$/', $trimLine);
 | 
						|
 | 
						|
            if ($skipArray) {
 | 
						|
                if ($arrayEndMatch) $skipArray = false;
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            // Comment to ignore
 | 
						|
            if (strpos($trimLine, '//!') === 0) {
 | 
						|
                $formatted[$index] = "";
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            // Comment
 | 
						|
            if (strpos($trimLine, '//') === 0) {
 | 
						|
                $formatted[$index] = "\t" . $trimLine;
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            // Arrays
 | 
						|
            $arrayStartMatch = preg_match('/^\'(.*)\'\s+?=>\s+?\[(\],)?\s*?$/', $trimLine, $matches);
 | 
						|
 | 
						|
            $indent = count($arrayKeys) + 1;
 | 
						|
            if ($arrayStartMatch === 1) {
 | 
						|
                if ($fileName === 'settings' && $matches[1] === 'language_select') {
 | 
						|
                    $skipArray = true;
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
                $arrayKeys[] = $matches[1];
 | 
						|
                $formatted[$index] = str_repeat(" ", $indent * 4) . str_pad("'{$matches[1]}'", $longestKeyLength) . "=> [";
 | 
						|
                if ($arrayEndMatch !== 1) continue;
 | 
						|
            }
 | 
						|
            if ($arrayEndMatch === 1) {
 | 
						|
                unsetArrayByKeys($langContent, $arrayKeys);
 | 
						|
                array_pop($arrayKeys);
 | 
						|
                if (isset($formatted[$index])) {
 | 
						|
                    $formatted[$index] .= '],';
 | 
						|
                } else {
 | 
						|
                    $formatted[$index] = str_repeat(" ", ($indent-1) * 4) . "],";
 | 
						|
                }
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            // Translation
 | 
						|
            $translationMatch = preg_match('/^\'(.*)\'\s+?=>\s+?[\'"](.*)?[\'"].+?$/', $trimLine, $matches);
 | 
						|
            if ($translationMatch === 1) {
 | 
						|
                $key = $matches[1];
 | 
						|
                $keys = array_merge($arrayKeys, [$key]);
 | 
						|
                $langVal = getTranslationByKeys($langContent, $keys);
 | 
						|
                if (empty($langVal)) continue;
 | 
						|
 | 
						|
                $keyPad = $longestKeyLength;
 | 
						|
                if (count($arrayKeys) === 0) {
 | 
						|
                    unset($langContent[$key]);
 | 
						|
                } else {
 | 
						|
                    $keyPad = calculateKeyPadding(getTranslationByKeys($enContent, $arrayKeys));
 | 
						|
                }
 | 
						|
 | 
						|
                $formatted[$index] = formatTranslationLine($key, $langVal, $indent, $keyPad);
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
    }
 | 
						|
 | 
						|
    // Fill missing lines
 | 
						|
    $arraySize = max(array_keys($formatted));
 | 
						|
    $formatted = array_replace(array_fill(0, $arraySize, ''), $formatted);
 | 
						|
 | 
						|
    // Add remaining translations
 | 
						|
    $langContent = array_filter($langContent, function($item) {
 | 
						|
        return !is_null($item) && !empty($item);
 | 
						|
    });
 | 
						|
    if (count($langContent) > 0) {
 | 
						|
        $formatted[] = '';
 | 
						|
        $formatted[] = "\t// Unmatched";
 | 
						|
    }
 | 
						|
    foreach ($langContent as $key => $value) {
 | 
						|
        if (is_array($value)) {
 | 
						|
            $formatted[] = formatTranslationArray($key, $value);
 | 
						|
        } else {
 | 
						|
            $formatted[] = formatTranslationLine($key, $value);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // Add end line
 | 
						|
    $formatted[] = '];';
 | 
						|
    return implode("\n", $formatted);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Format a translation line.
 | 
						|
 * @param string $key
 | 
						|
 * @param string $value
 | 
						|
 * @param int $indent
 | 
						|
 * @param int $keyPad
 | 
						|
 * @return string
 | 
						|
 */
 | 
						|
function formatTranslationLine(string $key, string $value, int $indent = 1, int $keyPad = 1) : string {
 | 
						|
    $start = str_repeat(" ", $indent * 4) . str_pad("'{$key}'", $keyPad, ' ');
 | 
						|
    if (strpos($value, "\n") !== false) {
 | 
						|
        $escapedValue = '"' .  str_replace("\n", '\n', $value)  . '"';
 | 
						|
        $escapedValue = '"' .  str_replace('"', '\"', $escapedValue)  . '"';
 | 
						|
    } else {
 | 
						|
        $escapedValue = "'" . str_replace("'", "\\'", $value) . "'";
 | 
						|
    }
 | 
						|
    return "{$start} => {$escapedValue},";
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Find the longest key in the array and provide the length
 | 
						|
 * for all keys to be used when printed.
 | 
						|
 * @param array $array
 | 
						|
 * @return int
 | 
						|
 */
 | 
						|
function calculateKeyPadding(array $array) : int {
 | 
						|
    $top = 0;
 | 
						|
    foreach ($array as $key => $value) {
 | 
						|
        $keyLen = strlen($key);
 | 
						|
        $top = max($top, $keyLen);
 | 
						|
    }
 | 
						|
    return min(35, $top + 2);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Format an translation array with the given key.
 | 
						|
 * Simply prints as an old-school php array.
 | 
						|
 * Used as a last-resort backup to save unused translations.
 | 
						|
 * @param string $key
 | 
						|
 * @param array $array
 | 
						|
 * @return string
 | 
						|
 */
 | 
						|
function formatTranslationArray(string $key, array $array) : string {
 | 
						|
    $arrayPHP = var_export($array, true);
 | 
						|
    return "    '{$key}' => {$arrayPHP},";
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Find a string translation value within a multi-dimensional array
 | 
						|
 * by traversing the given array of keys.
 | 
						|
 * @param array $translations
 | 
						|
 * @param array $keys
 | 
						|
 * @return string|array
 | 
						|
 */
 | 
						|
function getTranslationByKeys(array $translations, array $keys)  {
 | 
						|
    $val = $translations;
 | 
						|
    foreach ($keys as $key) {
 | 
						|
        $val = $val[$key] ?? '';
 | 
						|
        if ($val === '') return '';
 | 
						|
    }
 | 
						|
    return $val;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Unset an inner item of a multi-dimensional array by
 | 
						|
 * traversing the given array of keys.
 | 
						|
 * @param array $input
 | 
						|
 * @param array $keys
 | 
						|
 */
 | 
						|
function unsetArrayByKeys(array &$input, array $keys) {
 | 
						|
    $val = &$input;
 | 
						|
    $lastIndex = count($keys) - 1;
 | 
						|
    foreach ($keys as $index => &$key) {
 | 
						|
        if ($index === $lastIndex && is_array($val)) {
 | 
						|
            unset($val[$key]);
 | 
						|
        }
 | 
						|
        if (!is_array($val)) return;
 | 
						|
        $val = &$val[$key] ?? [];
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Write the given content to a translation file.
 | 
						|
 * @param string $lang
 | 
						|
 * @param string $fileName
 | 
						|
 * @param string $content
 | 
						|
 */
 | 
						|
function writeLangFile(string $lang, string $fileName, string $content) {
 | 
						|
    $path = __DIR__ . "/{$lang}/{$fileName}.php";
 | 
						|
    if (!file_exists($path)) {
 | 
						|
        errorOut("Expected translation file '{$path}' does not exist");
 | 
						|
    }
 | 
						|
    file_put_contents($path, $content);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Load the contents of a language file as an array of text lines.
 | 
						|
 * @param string $lang
 | 
						|
 * @param string $fileName
 | 
						|
 * @return array
 | 
						|
 */
 | 
						|
function loadLangFileLines(string $lang, string $fileName) : array {
 | 
						|
    $path = __DIR__ . "/{$lang}/{$fileName}.php";
 | 
						|
    if (!file_exists($path)) {
 | 
						|
        errorOut("Expected translation file '{$path}' does not exist");
 | 
						|
    }
 | 
						|
    $lines = explode("\n", file_get_contents($path));
 | 
						|
    return array_map(function($line) {
 | 
						|
        return trim($line, "\r");
 | 
						|
    }, $lines);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Load the contents of a language file
 | 
						|
 * @param string $lang
 | 
						|
 * @param string $fileName
 | 
						|
 * @return array
 | 
						|
 */
 | 
						|
function loadLang(string $lang, string $fileName) : array {
 | 
						|
    $path = __DIR__ . "/{$lang}/{$fileName}.php";
 | 
						|
    if (!file_exists($path)) {
 | 
						|
        errorOut("Expected translation file '{$path}' does not exist");
 | 
						|
    }
 | 
						|
 | 
						|
    $fileData = include($path);
 | 
						|
    return $fileData;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Fetch an array containing the names of all translation files without the extension.
 | 
						|
 * @return array
 | 
						|
 */
 | 
						|
function getTranslationFileNames() : array {
 | 
						|
    $dir = __DIR__ . "/en";
 | 
						|
    if (!file_exists($dir)) {
 | 
						|
        errorOut("Expected directory '{$dir}' does not exist");
 | 
						|
    }
 | 
						|
    $files = scandir($dir);
 | 
						|
    $fileNames = [];
 | 
						|
    foreach ($files as $file) {
 | 
						|
        if (substr($file, -4) === '.php') {
 | 
						|
            $fileNames[] = substr($file, 0, strlen($file) - 4);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return $fileNames;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Format a locale to follow the lowercase_UPERCASE standard
 | 
						|
 * @param string $lang
 | 
						|
 * @return string
 | 
						|
 */
 | 
						|
function formatLocale(string $lang) : string {
 | 
						|
    $langParts = explode('_', strtoupper($lang));
 | 
						|
    $langParts[0] = strtolower($langParts[0]);
 | 
						|
    return implode('_', $langParts);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Dump a variable then die.
 | 
						|
 * @param $content
 | 
						|
 */
 | 
						|
function dd($content) {
 | 
						|
    print_r($content);
 | 
						|
    exit(1);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Log out some information text in blue
 | 
						|
 * @param $text
 | 
						|
 */
 | 
						|
function info($text) {
 | 
						|
    echo "\e[34m" . $text . "\e[0m\n";
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Log out an error in red and exit.
 | 
						|
 * @param $text
 | 
						|
 */
 | 
						|
function errorOut($text) {
 | 
						|
    echo "\e[31m" . $text . "\e[0m\n";
 | 
						|
    exit(1);
 | 
						|
} |