68c6afc56f
Fixes #1077 Take a look at the docs update to see how it works
182 lines
6 KiB
PHP
182 lines
6 KiB
PHP
<?php
|
|
|
|
namespace Shaarli;
|
|
|
|
use Gettext\GettextTranslator;
|
|
use Gettext\Merge;
|
|
use Gettext\Translations;
|
|
use Gettext\Translator;
|
|
use Gettext\TranslatorInterface;
|
|
use Shaarli\Config\ConfigManager;
|
|
|
|
/**
|
|
* Class Languages
|
|
*
|
|
* Load Shaarli translations using 'gettext/gettext'.
|
|
* This class allows to either use PHP gettext extension, or a PHP implementation of gettext,
|
|
* with a fixed language, or dynamically using autoLocale().
|
|
*
|
|
* Translation files PO/MO files follow gettext standard and must be placed under:
|
|
* <translation path>/<language>/LC_MESSAGES/<domain>.[po|mo]
|
|
*
|
|
* Pros/cons:
|
|
* - gettext extension is faster
|
|
* - gettext is very system dependent (PHP extension, the locale must be installed, and web server reloaded)
|
|
*
|
|
* Settings:
|
|
* - translation.mode:
|
|
* - auto: use default setting (PHP implementation)
|
|
* - php: use PHP implementation
|
|
* - gettext: use gettext wrapper
|
|
* - translation.language:
|
|
* - auto: use autoLocale() and the language change according to user HTTP headers
|
|
* - fixed language: e.g. 'fr'
|
|
* - translation.extensions:
|
|
* - domain => translation_path: allow plugins and themes to extend the defaut extension
|
|
* The domain must be unique, and translation path must be relative, and contains the tree mentioned above.
|
|
*
|
|
* @package Shaarli
|
|
*/
|
|
class Languages
|
|
{
|
|
/**
|
|
* Core translations domain
|
|
*/
|
|
const DEFAULT_DOMAIN = 'shaarli';
|
|
|
|
/**
|
|
* @var TranslatorInterface
|
|
*/
|
|
protected $translator;
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
protected $language;
|
|
|
|
/**
|
|
* @var ConfigManager
|
|
*/
|
|
protected $conf;
|
|
|
|
/**
|
|
* Languages constructor.
|
|
*
|
|
* @param string $language lang determined by autoLocale(), can be overridden.
|
|
* @param ConfigManager $conf instance.
|
|
*/
|
|
public function __construct($language, $conf)
|
|
{
|
|
$this->conf = $conf;
|
|
$confLanguage = $this->conf->get('translation.language', 'auto');
|
|
// Auto mode or invalid parameter, use the detected language.
|
|
// If the detected language is invalid, it doesn't matter, it will use English.
|
|
if ($confLanguage === 'auto' || ! $this->isValidLanguage($confLanguage)) {
|
|
$this->language = substr($language, 0, 5);
|
|
} else {
|
|
$this->language = $confLanguage;
|
|
}
|
|
|
|
if (! extension_loaded('gettext')
|
|
|| in_array($this->conf->get('translation.mode', 'auto'), ['auto', 'php'])
|
|
) {
|
|
$this->initPhpTranslator();
|
|
} else {
|
|
$this->initGettextTranslator();
|
|
}
|
|
|
|
// Register default functions (e.g. '__()') to use our Translator
|
|
$this->translator->register();
|
|
}
|
|
|
|
/**
|
|
* Initialize the translator using php gettext extension (gettext dependency act as a wrapper).
|
|
*/
|
|
protected function initGettextTranslator ()
|
|
{
|
|
$this->translator = new GettextTranslator();
|
|
$this->translator->setLanguage($this->language);
|
|
$this->translator->loadDomain(self::DEFAULT_DOMAIN, 'inc/languages');
|
|
|
|
// Default extension translation from the current theme
|
|
$themeTransFolder = rtrim($this->conf->get('raintpl_tpl'), '/') .'/'. $this->conf->get('theme') .'/language';
|
|
if (is_dir($themeTransFolder)) {
|
|
$this->translator->loadDomain($this->conf->get('theme'), $themeTransFolder, false);
|
|
}
|
|
|
|
foreach ($this->conf->get('translation.extensions', []) as $domain => $translationPath) {
|
|
if ($domain !== self::DEFAULT_DOMAIN) {
|
|
$this->translator->loadDomain($domain, $translationPath, false);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Initialize the translator using a PHP implementation of gettext.
|
|
*
|
|
* Note that if language po file doesn't exist, errors are ignored (e.g. not installed language).
|
|
*/
|
|
protected function initPhpTranslator()
|
|
{
|
|
$this->translator = new Translator();
|
|
$translations = new Translations();
|
|
// Core translations
|
|
try {
|
|
$translations = $translations->addFromPoFile('inc/languages/'. $this->language .'/LC_MESSAGES/shaarli.po');
|
|
$translations->setDomain('shaarli');
|
|
$this->translator->loadTranslations($translations);
|
|
} catch (\InvalidArgumentException $e) {}
|
|
|
|
// Default extension translation from the current theme
|
|
$theme = $this->conf->get('theme');
|
|
$themeTransFolder = rtrim($this->conf->get('raintpl_tpl'), '/') .'/'. $theme .'/language';
|
|
if (is_dir($themeTransFolder)) {
|
|
try {
|
|
$translations = Translations::fromPoFile(
|
|
$themeTransFolder .'/'. $this->language .'/LC_MESSAGES/'. $theme .'.po'
|
|
);
|
|
$translations->setDomain($theme);
|
|
$this->translator->loadTranslations($translations);
|
|
} catch (\InvalidArgumentException $e) {}
|
|
}
|
|
|
|
// Extension translations (plugins, themes, etc.).
|
|
foreach ($this->conf->get('translation.extensions', []) as $domain => $translationPath) {
|
|
if ($domain === self::DEFAULT_DOMAIN) {
|
|
continue;
|
|
}
|
|
|
|
try {
|
|
$extension = Translations::fromPoFile($translationPath . $this->language .'/LC_MESSAGES/'. $domain .'.po');
|
|
$extension->setDomain($domain);
|
|
$this->translator->loadTranslations($extension);
|
|
} catch (\InvalidArgumentException $e) {}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Checks if a language string is valid.
|
|
*
|
|
* @param string $language e.g. 'fr' or 'en_US'
|
|
*
|
|
* @return bool true if valid, false otherwise
|
|
*/
|
|
protected function isValidLanguage($language)
|
|
{
|
|
return preg_match('/^[a-z]{2}(_[A-Z]{2})?/', $language) === 1;
|
|
}
|
|
|
|
/**
|
|
* Get the list of available languages for Shaarli.
|
|
*
|
|
* @return array List of available languages, with their label.
|
|
*/
|
|
public static function getAvailableLanguages()
|
|
{
|
|
return [
|
|
'auto' => t('Automatic'),
|
|
'en' => t('English'),
|
|
'fr' => t('French'),
|
|
];
|
|
}
|
|
}
|