2016-09-30 01:58:09 +00:00
< ? php
declare ( strict_types = 1 );
namespace JKingWeb\NewsSync ;
class Lang {
const PATH = BASE . " locale " . DIRECTORY_SEPARATOR ;
const DEFAULT = " en " ;
const REQUIRED = [
2016-09-30 23:03:30 +00:00
" Exception.JKingWeb/NewsSync/Lang/Exception.defaultFileMissing " => " Default language file \" { 0} \" missing " ,
" Exception.JKingWeb/NewsSync/Lang/Exception.fileMissing " => " Language file \" { 0} \" is not available " ,
" Exception.JKingWeb/NewsSync/Lang/Exception.fileUnreadable " => " Insufficient permissions to read language file \" { 0} \" " ,
" Exception.JKingWeb/NewsSync/Lang/Exception.fileCorrupt " => " Language file \" { 0} \" is corrupt or does not conform to expected format " ,
" Exception.JKingWeb/NewsSync/Lang/Exception.stringMissing " => " Message string \" { msgID} \" missing from all loaded language files ( { fileList}) " ,
2016-10-15 13:45:23 +00:00
" Exception.JKingWeb/NewsSync/Lang/Exception.stringInvalid " => " Message string \" { msgID} \" is not a valid ICU message string (language files loaded: { fileList}) " ,
2016-09-30 01:58:09 +00:00
];
static protected $requirementsMet = false ;
static protected $synched = false ;
2016-09-30 23:03:30 +00:00
static protected $wanted = self :: DEFAULT ;
2016-09-30 01:58:09 +00:00
static protected $locale = " " ;
static protected $loaded = [];
static protected $strings = self :: REQUIRED ;
protected function __construct () {}
static public function set ( string $locale = " " , bool $immediate = false ) : string {
if ( ! self :: $requirementsMet ) self :: checkRequirements ();
if ( $locale == " " ) $locale = self :: DEFAULT ;
if ( $locale == self :: $wanted ) return $locale ;
$list = self :: listFiles ();
if ( ! in_array ( self :: DEFAULT , $list )) throw new Lang\Exception ( " defaultFileMissing " , self :: DEFAULT );
self :: $wanted = self :: match ( $locale , $list );
self :: $synched = false ;
if ( $immediate ) self :: load ();
return self :: $wanted ;
}
static public function get () : string {
return ( self :: $locale == " " ) ? self :: DEFAULT : self :: $locale ;
}
static public function msg ( string $msgID , $vars = null ) : string {
// if we're trying to load the system default language and it fails, we have a chicken and egg problem, so we catch the exception and load no language file instead
if ( ! self :: $synched ) try { self :: load ();} catch ( Lang\Exception $e ) {
if ( self :: $wanted == self :: DEFAULT ) {
self :: set ();
self :: load ();
} else {
throw $e ;
}
}
if ( ! array_key_exists ( $msgID , self :: $strings )) throw new Lang\Exception ( " stringMissing " , [ 'msgID' => $msgID , 'fileList' => implode ( " , " , self :: $loaded )]);
// variables fed to MessageFormatter must be contained in array
if ( $vars !== null && ! is_array ( $vars )) $vars = [ $vars ];
2016-10-15 13:45:23 +00:00
$msg = \MessageFormatter :: formatMessage ( self :: $locale , self :: $strings [ $msgID ], $vars );
if ( $msg === false ) throw new Lang\Exception ( " stringInvalid " , [ 'msgID' => $msgID , 'fileList' => implode ( " , " , self :: $loaded )]);
return $msg ;
2016-09-30 01:58:09 +00:00
}
static public function list ( string $locale = " " ) : array {
$out = [];
$files = self :: listFiles ();
foreach ( $files as $tag ) {
$out [ $tag ] = \Locale :: getDisplayName ( $tag , ( $locale == " " ) ? $tag : $locale );
}
return $out ;
}
static public function match ( string $locale , array $list = null ) : string {
if ( $list === null ) $list = self :: listFiles ();
$default = ( self :: $locale == " " ) ? self :: DEFAULT : self :: $locale ;
return \Locale :: lookup ( $list , $locale , true , $default );
}
static protected function checkRequirements () : bool {
if ( ! extension_loaded ( " intl " )) throw new FatalException ( " The \" Intl \" extension is required, but not loaded " );
self :: $requirementsMet = true ;
return true ;
}
static protected function listFiles ( string $path = self :: PATH ) : array {
$out = glob ( $path . " *.php " );
return array_map ( function ( $file ) {
$file = substr ( $file , strrpos ( $file , DIRECTORY_SEPARATOR ) + 1 );
return strtolower ( substr ( $file , 0 , strrpos ( $file , " . " )));
}, $out );
}
static protected function load () : bool {
self :: $synched = true ;
if ( ! self :: $requirementsMet ) self :: checkRequirements ();
// if we've yet to request a locale, just load the fallback strings and return
if ( self :: $wanted == " " ) {
self :: $strings = self :: REQUIRED ;
self :: $locale = self :: $wanted ;
return true ;
}
// decompose the requested locale from specific to general, building a list of files to load
$tags = \Locale :: parseLocale ( self :: $wanted );
$files = [];
$loaded = [];
$strings = [];
while ( sizeof ( $tags ) > 0 ) {
$files [] = \Locale :: composeLocale ( $tags );
$tag = array_pop ( $tags );
}
// include the default locale as the base if the most general locale requested is not the default
if ( $tag != self :: DEFAULT ) $files [] = self :: DEFAULT ;
// save the list of files to be loaded for later reference
$loaded = $files ;
// reduce the list of files to be loaded to the minimum necessary (e.g. if we go from "fr" to "fr_ca", we don't need to load "fr" or "en")
$files = [];
foreach ( $loaded as $file ) {
if ( $file == self :: $locale ) break ;
$files [] = $file ;
}
// if we need to load all files, start with the fallback strings
if ( $files == $loaded ) $strings [] = self :: REQUIRED ;
// read files in reverse order
$files = array_reverse ( $files );
foreach ( $files as $file ) {
if ( ! file_exists ( self :: PATH . " $file .php " )) throw new Lang\Exception ( " fileMissing " , $file );
if ( ! is_readable ( self :: PATH . " $file .php " )) throw new Lang\Exception ( " fileUnreadable " , $file );
2016-09-30 23:03:30 +00:00
if ( ! $strings [] = ( @ include self :: PATH . " $file .php " )) throw new Lang\Exception ( " fileCorrupt " , $file );
2016-09-30 01:58:09 +00:00
}
// apply the results and return
self :: $strings = call_user_func_array ( " array_replace_recursive " , $strings );
self :: $loaded = $loaded ;
self :: $locale = self :: $wanted ;
return true ;
}
}