Use an i18n tool
This commit is contained in:
parent
e487438606
commit
16dec853f9
12 changed files with 662 additions and 97 deletions
377
i18n.php
Normal file
377
i18n.php
Normal file
|
@ -0,0 +1,377 @@
|
|||
<?php
|
||||
|
||||
/*
|
||||
* Fork this project on GitHub!
|
||||
* https://github.com/Philipp15b/php-i18n
|
||||
*
|
||||
* License: MIT
|
||||
*/
|
||||
|
||||
class i18n {
|
||||
|
||||
/**
|
||||
* Language file path
|
||||
*
|
||||
* This is the path for the language files.
|
||||
* You must use the '{LANGUAGE}' placeholder for the language or
|
||||
* the script wont find any language files.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $filePath = './lang/lang_{LANGUAGE}.ini';
|
||||
|
||||
/**
|
||||
* Cache file path
|
||||
*
|
||||
* This is the path for all the cache files.
|
||||
* Best is an empty directory with no other files in it.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $cachePath = './cache/lang/';
|
||||
|
||||
/**
|
||||
* Fallback language
|
||||
*
|
||||
* This is the language which is used when there is no language file for
|
||||
* all other user languages.
|
||||
* It has the lowest priority.
|
||||
* Remember to create a language file for the fallback!!
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $fallbackLang = 'en';
|
||||
|
||||
/**
|
||||
* Merge in fallback language
|
||||
*
|
||||
* Whether to merge current language's strings with the strings of
|
||||
* the fallback language ($fallbackLang).
|
||||
*
|
||||
* @var bool
|
||||
*/
|
||||
protected $mergeFallback = true;
|
||||
|
||||
/**
|
||||
* The class name of the compiled class that contains the translated texts.
|
||||
* @var string
|
||||
*/
|
||||
protected $prefix = 'L';
|
||||
|
||||
/**
|
||||
* Forced language
|
||||
*
|
||||
* If you want to force a specific language define it here.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $forcedLang = NULL;
|
||||
|
||||
/**
|
||||
* This is the separator used if you use sections in your ini-file
|
||||
*
|
||||
* For example, if you have a string 'greeting' in a section 'welcomepage'
|
||||
* you will can access it via 'L::welcomepage_greeting'.
|
||||
* If you changed it to 'ABC' you could access your string
|
||||
* via 'L::welcomepageABCgreeting'
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $sectionSeparator = '_';
|
||||
|
||||
|
||||
/*
|
||||
* The following properties are only available after calling init().
|
||||
*/
|
||||
|
||||
/**
|
||||
* User languages
|
||||
*
|
||||
* These are the languages the user uses.
|
||||
* Normally, if you use the getUserLangs-method this array will be
|
||||
* filled in like this:
|
||||
* 1. Forced language
|
||||
* 2. Language in $_GET['lang']
|
||||
* 3. Language in $_SESSION['lang']
|
||||
* 4. Fallback language
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $userLangs = array();
|
||||
|
||||
protected $appliedLang = NULL;
|
||||
protected $langFilePath = NULL;
|
||||
protected $cacheFilePath = NULL;
|
||||
protected $isInitialized = false;
|
||||
|
||||
|
||||
/**
|
||||
* Constructor
|
||||
*
|
||||
* The constructor sets all important settings.
|
||||
* All params are optional, you can set the options via extra functions too.
|
||||
*
|
||||
* @param string [$filePath] This is the path for the language files.
|
||||
* You must use the '{LANGUAGE}' placeholder for the language.
|
||||
* @param string [$cachePath] This is the path for all the cache files.
|
||||
* Best is an empty directory with no other files in it. No placeholders.
|
||||
* @param string [$fallbackLang] This is the language which is used
|
||||
* when there is no language file for all other user languages.
|
||||
* It has the lowest priority.
|
||||
* @param string [$prefix] The class name of the compiled class
|
||||
* that contains the translated texts. Defaults to 'L'.
|
||||
*/
|
||||
public function __construct(
|
||||
$filePath = NULL,
|
||||
$cachePath = NULL,
|
||||
$fallbackLang = NULL,
|
||||
$prefix = NULL)
|
||||
{
|
||||
// Apply settings
|
||||
if ($filePath != NULL) {
|
||||
$this->filePath = $filePath;
|
||||
}
|
||||
|
||||
if ($cachePath != NULL) {
|
||||
$this->cachePath = $cachePath;
|
||||
}
|
||||
|
||||
if ($fallbackLang != NULL) {
|
||||
$this->fallbackLang = $fallbackLang;
|
||||
}
|
||||
|
||||
if ($prefix != NULL) {
|
||||
$this->prefix = $prefix;
|
||||
}
|
||||
}
|
||||
|
||||
public function init() {
|
||||
if ($this->isInitialized()) {
|
||||
throw new BadMethodCallException(
|
||||
'This object from class ' . __CLASS__ .
|
||||
' is already initialized. ' .
|
||||
'It is not possible to init one object twice!');
|
||||
}
|
||||
|
||||
$this->isInitialized = true;
|
||||
|
||||
$this->userLangs = $this->getUserLangs();
|
||||
|
||||
// search for language file
|
||||
$this->appliedLang = NULL;
|
||||
foreach ($this->userLangs as $priority => $langcode) {
|
||||
$this->langFilePath = $this->getConfigFilename($langcode);
|
||||
if (file_exists($this->langFilePath)) {
|
||||
$this->appliedLang = $langcode;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if ($this->appliedLang == NULL) {
|
||||
throw new RuntimeException('No language file was found.');
|
||||
}
|
||||
|
||||
// search for cache file
|
||||
$this->cacheFilePath =
|
||||
$this->cachePath . '/php_i18n_' . md5_file(__FILE__) . '_' .
|
||||
$this->prefix . '_' . $this->appliedLang . '.cache.php';
|
||||
|
||||
// whether we need to create a new cache file
|
||||
$outdated = !file_exists($this->cacheFilePath) ||
|
||||
filemtime($this->cacheFilePath) < filemtime($this->langFilePath) || // the language config was updated
|
||||
($this->mergeFallback && filemtime($this->cacheFilePath) < filemtime($this->getConfigFilename($this->fallbackLang))); // the fallback language config was updated
|
||||
|
||||
if ($outdated || true) {
|
||||
$config = $this->load($this->langFilePath);
|
||||
if ($this->mergeFallback)
|
||||
$config = array_replace_recursive($this->load($this->getConfigFilename($this->fallbackLang)), $config);
|
||||
|
||||
$compiled =
|
||||
"<?php\n"
|
||||
. "class " . $this->prefix . " {\n"
|
||||
. $this->compile($config)
|
||||
. ' public static function __callStatic($string, $args) {' . "\n"
|
||||
. ' if (defined("self::" . $string)) {' . "\n"
|
||||
. ' return vsprintf(constant("self::" . $string), $args);' . "\n"
|
||||
. " }\n"
|
||||
. " }\n"
|
||||
. " function ".$this->prefix .'($string, $args=NULL) {' . "\n"
|
||||
. ' $return = constant("'.$this->prefix.'::".$string);' . "\n"
|
||||
. ' return $args ? vsprintf($return,$args) : $return;' . "\n"
|
||||
. " }\n"
|
||||
. '}';
|
||||
|
||||
if( ! is_dir($this->cachePath)) {
|
||||
mkdir($this->cachePath, 0755, true);
|
||||
}
|
||||
if (file_put_contents($this->cacheFilePath, $compiled) === FALSE) {
|
||||
throw new Exception("Could not write cache file to path '" . $this->cacheFilePath . "'. Is it writable?");
|
||||
}
|
||||
chmod($this->cacheFilePath, 0755);
|
||||
|
||||
}
|
||||
|
||||
require_once $this->cacheFilePath;
|
||||
}
|
||||
|
||||
public function isInitialized() {
|
||||
return $this->isInitialized;
|
||||
}
|
||||
|
||||
public function getAppliedLang() {
|
||||
return $this->appliedLang;
|
||||
}
|
||||
|
||||
public function getCachePath() {
|
||||
return $this->cachePath;
|
||||
}
|
||||
|
||||
public function getFallbackLang() {
|
||||
return $this->fallbackLang;
|
||||
}
|
||||
|
||||
public function setFilePath($filePath) {
|
||||
$this->fail_after_init();
|
||||
$this->filePath = $filePath;
|
||||
}
|
||||
|
||||
public function setCachePath($cachePath) {
|
||||
$this->fail_after_init();
|
||||
$this->cachePath = $cachePath;
|
||||
}
|
||||
|
||||
public function setFallbackLang($fallbackLang) {
|
||||
$this->fail_after_init();
|
||||
$this->fallbackLang = $fallbackLang;
|
||||
}
|
||||
|
||||
public function setMergeFallback($mergeFallback) {
|
||||
$this->fail_after_init();
|
||||
$this->mergeFallback = $mergeFallback;
|
||||
}
|
||||
|
||||
public function setPrefix($prefix) {
|
||||
$this->fail_after_init();
|
||||
$this->prefix = $prefix;
|
||||
}
|
||||
|
||||
public function setForcedLang($forcedLang) {
|
||||
$this->fail_after_init();
|
||||
$this->forcedLang = $forcedLang;
|
||||
}
|
||||
|
||||
public function setSectionSeparator($sectionSeparator) {
|
||||
$this->fail_after_init();
|
||||
$this->sectionSeparator = $sectionSeparator;
|
||||
}
|
||||
|
||||
/**
|
||||
* @deprecated Use setSectionSeparator.
|
||||
*/
|
||||
public function setSectionSeperator($sectionSeparator) {
|
||||
$this->setSectionSeparator($sectionSeparator);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the user languages
|
||||
*
|
||||
* Normally it returns an array like this:
|
||||
* 1. Forced language
|
||||
* 2. Language in $_GET['lang']
|
||||
* 3. Language in $_SESSION['lang']
|
||||
* 4. HTTP_ACCEPT_LANGUAGE
|
||||
* 5. Fallback language
|
||||
* Note: duplicate values are deleted.
|
||||
*
|
||||
* @return array with the user languages sorted by priority.
|
||||
*/
|
||||
public function getUserLangs() {
|
||||
$userLangs = array();
|
||||
|
||||
// Highest priority: forced language
|
||||
if ($this->forcedLang != NULL) {
|
||||
$userLangs[] = $this->forcedLang;
|
||||
}
|
||||
|
||||
// 2nd highest priority: GET parameter 'lang'
|
||||
if (isset($_GET['lang']) && is_string($_GET['lang'])) {
|
||||
$userLangs[] = $_GET['lang'];
|
||||
}
|
||||
|
||||
// 3rd highest priority: SESSION parameter 'lang'
|
||||
if (isset($_SESSION['lang']) && is_string($_SESSION['lang'])) {
|
||||
$userLangs[] = $_SESSION['lang'];
|
||||
}
|
||||
|
||||
// 4th highest priority: HTTP_ACCEPT_LANGUAGE
|
||||
if (isset($_SERVER['HTTP_ACCEPT_LANGUAGE'])) {
|
||||
foreach (explode(',', $_SERVER['HTTP_ACCEPT_LANGUAGE']) as $part) {
|
||||
$userLangs[] = strtolower(substr($part, 0, 2));
|
||||
}
|
||||
}
|
||||
|
||||
// Lowest priority: fallback
|
||||
$userLangs[] = $this->fallbackLang;
|
||||
|
||||
// remove duplicate elements
|
||||
$userLangs = array_unique($userLangs);
|
||||
|
||||
// remove illegal userLangs
|
||||
$userLangs2 = array();
|
||||
foreach ($userLangs as $key => $value) {
|
||||
// only allow a-z, A-Z and 0-9 and _ and -
|
||||
if (preg_match('/^[a-zA-Z0-9_-]*$/', $value) === 1)
|
||||
$userLangs2[$key] = $value;
|
||||
}
|
||||
|
||||
return $userLangs2;
|
||||
}
|
||||
|
||||
protected function getConfigFilename($langcode) {
|
||||
return str_replace('{LANGUAGE}', $langcode, $this->filePath);
|
||||
}
|
||||
|
||||
protected function load($filename) {
|
||||
$ext = substr(strrchr($filename, '.'), 1);
|
||||
switch ($ext) {
|
||||
case 'properties':
|
||||
case 'ini':
|
||||
$config = parse_ini_file($filename, true);
|
||||
break;
|
||||
case 'yml':
|
||||
case 'yaml':
|
||||
$config = spyc_load_file($filename);
|
||||
break;
|
||||
case 'json':
|
||||
$config = json_decode(file_get_contents($filename), true);
|
||||
break;
|
||||
default:
|
||||
throw new InvalidArgumentException($ext . " is not a valid extension!");
|
||||
}
|
||||
return $config;
|
||||
}
|
||||
|
||||
/**
|
||||
* Recursively compile an associative array to PHP code.
|
||||
*/
|
||||
protected function compile($config, $prefix = '') {
|
||||
$code = '';
|
||||
foreach ($config as $key => $value) {
|
||||
if (is_array($value)) {
|
||||
$code .= $this->compile($value, $prefix . $key . $this->sectionSeparator);
|
||||
} else {
|
||||
$fullName = $prefix . $key;
|
||||
if (!preg_match('/^[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*$/', $fullName)) {
|
||||
throw new InvalidArgumentException(__CLASS__ . ": Cannot compile translation key " . $fullName . " because it is not a valid PHP identifier.");
|
||||
}
|
||||
$code .= ' const ' . $fullName . ' = \'' . str_replace('\'', '\\\'', $value) . "';\n";
|
||||
}
|
||||
}
|
||||
return $code;
|
||||
}
|
||||
|
||||
protected function fail_after_init() {
|
||||
if ($this->isInitialized()) {
|
||||
throw new BadMethodCallException('This ' . __CLASS__ . ' object is already initalized, so you can not change any settings.');
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue