| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628 | <?php/** * Internationalization * * PHP 5 * * CakePHP(tm) : Rapid Development Framework (http://cakephp.org) * Copyright 2005-2012, Cake Software Foundation, Inc. (http://cakefoundation.org) * * Licensed under The MIT License * Redistributions of files must retain the above copyright notice. * * @copyright     Copyright 2005-2012, Cake Software Foundation, Inc. (http://cakefoundation.org) * @link          http://cakephp.org CakePHP(tm) Project * @package       Cake.I18n * @since         CakePHP(tm) v 1.2.0.4116 * @license       MIT License (http://www.opensource.org/licenses/mit-license.php) */App::uses('CakePlugin', 'Core');App::uses('L10n', 'I18n');App::uses('Multibyte', 'I18n');App::uses('CakeSession', 'Model/Datasource');/** * I18n handles translation of Text and time format strings. * * @package       Cake.I18n */class I18n {/** * Instance of the L10n class for localization * * @var L10n */	public $l10n = null;/** * Default domain of translation * * @var string */	public static $defaultDomain = 'default';/** * Current domain of translation * * @var string */	public $domain = null;/** * Current category of translation * * @var string */	public $category = 'LC_MESSAGES';/** * Current language used for translations * * @var string */	protected $_lang = null;/** * Translation strings for a specific domain read from the .mo or .po files * * @var array */	protected $_domains = array();/** * Set to true when I18N::_bindTextDomain() is called for the first time. * If a translation file is found it is set to false again * * @var boolean */	protected $_noLocale = false;/** * Translation categories * * @var array */	protected $_categories = array(		'LC_ALL', 'LC_COLLATE', 'LC_CTYPE', 'LC_MONETARY', 'LC_NUMERIC', 'LC_TIME', 'LC_MESSAGES'	);/** * Escape string * * @var string */	protected $_escape = null;/** * Constructor, use I18n::getInstance() to get the i18n translation object. * * @return void */	public function __construct() {		$this->l10n = new L10n();	}/** * Return a static instance of the I18n class * * @return I18n */	public static function &getInstance() {		static $instance = array();		if (!$instance) {			$instance[0] = new I18n();		}		return $instance[0];	}/** * Used by the translation functions in basics.php * Returns a translated string based on current language and translation files stored in locale folder * * @param string $singular String to translate * @param string $plural Plural string (if any) * @param string $domain Domain The domain of the translation. Domains are often used by plugin translations * @param string $category Category The integer value of the category to use. * @param integer $count Count Count is used with $plural to choose the correct plural form. * @param string $language Language to translate string to. * 	If null it checks for language in session followed by Config.language configuration variable. * @return string translated string. */	public static function translate($singular, $plural = null, $domain = null, $category = 6, $count = null, $language = null) {		$_this = I18n::getInstance();		if (strpos($singular, "\r\n") !== false) {			$singular = str_replace("\r\n", "\n", $singular);		}		if ($plural !== null && strpos($plural, "\r\n") !== false) {			$plural = str_replace("\r\n", "\n", $plural);		}		if (is_numeric($category)) {			$_this->category = $_this->_categories[$category];		}		if (empty($language)) {			if (CakeSession::started()) {				$language = CakeSession::read('Config.language');			}			if (empty($language)) {				$language = Configure::read('Config.language');			}		}		if (($_this->_lang && $_this->_lang !== $language) || !$_this->_lang) {			$lang = $_this->l10n->get($language);			$_this->_lang = $lang;		}		if (is_null($domain)) {			$domain = self::$defaultDomain;		}		$_this->domain = $domain . '_' . $_this->l10n->lang;		if (!isset($_this->_domains[$domain][$_this->_lang])) {			$_this->_domains[$domain][$_this->_lang] = Cache::read($_this->domain, '_cake_core_');		}		if (!isset($_this->_domains[$domain][$_this->_lang][$_this->category])) {			$_this->_bindTextDomain($domain);			Cache::write($_this->domain, $_this->_domains[$domain][$_this->_lang], '_cake_core_');		}		if ($_this->category == 'LC_TIME') {			return $_this->_translateTime($singular, $domain);		}		if (!isset($count)) {			$plurals = 0;		} elseif (!empty($_this->_domains[$domain][$_this->_lang][$_this->category]["%plural-c"]) && $_this->_noLocale === false) {			$header = $_this->_domains[$domain][$_this->_lang][$_this->category]["%plural-c"];			$plurals = $_this->_pluralGuess($header, $count);		} else {			if ($count != 1) {				$plurals = 1;			} else {				$plurals = 0;			}		}		if (!empty($_this->_domains[$domain][$_this->_lang][$_this->category][$singular])) {			if (($trans = $_this->_domains[$domain][$_this->_lang][$_this->category][$singular]) || ($plurals) && ($trans = $_this->_domains[$domain][$_this->_lang][$_this->category][$plural])) {				if (is_array($trans)) {					if (isset($trans[$plurals])) {						$trans = $trans[$plurals];					} else {						trigger_error(							__d('cake_dev',								'Missing plural form translation for "%s" in "%s" domain, "%s" locale. ' .								' Check your po file for correct plurals and valid Plural-Forms header.',								$singular,								$domain,								$_this->_lang							),							E_USER_WARNING						);						$trans = $trans[0];					}				}				if (strlen($trans)) {					return $trans;				}			}		}		if (!empty($plurals)) {			return $plural;		}		return $singular;	}/** * Clears the domains internal data array. Useful for testing i18n. * * @return void */	public static function clear() {		$self = I18n::getInstance();		$self->_domains = array();	}/** * Get the loaded domains cache. * * @return array */	public static function domains() {		$self = I18n::getInstance();		return $self->_domains;	}/** * Attempts to find the plural form of a string. * * @param string $header Type * @param integer $n Number * @return integer plural match */	protected function _pluralGuess($header, $n) {		if (!is_string($header) || $header === "nplurals=1;plural=0;" || !isset($header[0])) {			return 0;		}		if ($header === "nplurals=2;plural=n!=1;") {			return $n != 1 ? 1 : 0;		} elseif ($header === "nplurals=2;plural=n>1;") {			return $n > 1 ? 1 : 0;		}		if (strpos($header, "plurals=3")) {			if (strpos($header, "100!=11")) {				if (strpos($header, "10<=4")) {					return $n % 10 == 1 && $n % 100 != 11 ? 0 : ($n % 10 >= 2 && $n % 10 <= 4 && ($n % 100 < 10 || $n % 100 >= 20) ? 1 : 2);				} elseif (strpos($header, "100<10")) {					return $n % 10 == 1 && $n % 100 != 11 ? 0 : ($n % 10 >= 2 && ($n % 100 < 10 || $n % 100 >= 20) ? 1 : 2);				}				return $n % 10 == 1 && $n % 100 != 11 ? 0 : ($n != 0 ? 1 : 2);			} elseif (strpos($header, "n==2")) {				return $n == 1 ? 0 : ($n == 2 ? 1 : 2);			} elseif (strpos($header, "n==0")) {				return $n == 1 ? 0 : ($n == 0 || ($n % 100 > 0 && $n % 100 < 20) ? 1 : 2);			} elseif (strpos($header, "n>=2")) {				return $n == 1 ? 0 : ($n >= 2 && $n <= 4 ? 1 : 2);			} elseif (strpos($header, "10>=2")) {				return $n == 1 ? 0 : ($n % 10 >= 2 && $n % 10 <= 4 && ($n % 100 < 10 || $n % 100 >= 20) ? 1 : 2);			}			return $n % 10 == 1 ? 0 : ($n % 10 == 2 ? 1 : 2);		} elseif (strpos($header, "plurals=4")) {			if (strpos($header, "100==2")) {				return $n % 100 == 1 ? 0 : ($n % 100 == 2 ? 1 : ($n % 100 == 3 || $n % 100 == 4 ? 2 : 3));			} elseif (strpos($header, "n>=3")) {				return $n == 1 ? 0 : ($n == 2 ? 1 : ($n == 0 || ($n >= 3 && $n <= 10) ? 2 : 3));			} elseif (strpos($header, "100>=1")) {				return $n == 1 ? 0 : ($n == 0 || ($n % 100 >= 1 && $n % 100 <= 10) ? 1 : ($n % 100 >= 11 && $n % 100 <= 20 ? 2 : 3));			}		} elseif (strpos($header, "plurals=5")) {			return $n == 1 ? 0 : ($n == 2 ? 1 : ($n >= 3 && $n <= 6 ? 2 : ($n >= 7 && $n <= 10 ? 3 : 4)));		}	}/** * Binds the given domain to a file in the specified directory. * * @param string $domain Domain to bind * @return string Domain binded */	protected function _bindTextDomain($domain) {		$this->_noLocale = true;		$core = true;		$merge = array();		$searchPaths = App::path('locales');		$plugins = CakePlugin::loaded();		if (!empty($plugins)) {			foreach ($plugins as $plugin) {				$pluginDomain = Inflector::underscore($plugin);				if ($pluginDomain === $domain) {					$searchPaths[] = CakePlugin::path($plugin) . 'Locale' . DS;					$searchPaths = array_reverse($searchPaths);					break;				}			}		}		foreach ($searchPaths as $directory) {			foreach ($this->l10n->languagePath as $lang) {				$localeDef = $directory . $lang . DS . $this->category;				if (is_file($localeDef)) {					$definitions = self::loadLocaleDefinition($localeDef);					if ($definitions !== false) {						$this->_domains[$domain][$this->_lang][$this->category] = self::loadLocaleDefinition($localeDef);						$this->_noLocale = false;						return $domain;					}				}				if ($core) {					$app = $directory . $lang . DS . $this->category . DS . 'core';					$translations = false;					if (is_file($app . '.mo')) {						$translations = self::loadMo($app . '.mo');					}					if ($translations === false && is_file($app . '.po')) {						$translations = self::loadPo($app . '.po');					}					if ($translations !== false) {						$this->_domains[$domain][$this->_lang][$this->category] = $translations;						$merge[$domain][$this->_lang][$this->category] = $this->_domains[$domain][$this->_lang][$this->category];						$this->_noLocale = false;						$core = null;					}				}				$file = $directory . $lang . DS . $this->category . DS . $domain;				$translations = false;				if (is_file($file . '.mo')) {					$translations = self::loadMo($file . '.mo');				}				if ($translations === false && is_file($file . '.po')) {					$translations = self::loadPo($file . '.po');				}				if ($translations !== false) {					$this->_domains[$domain][$this->_lang][$this->category] = $translations;					$this->_noLocale = false;					break 2;				}			}		}		if (empty($this->_domains[$domain][$this->_lang][$this->category])) {			$this->_domains[$domain][$this->_lang][$this->category] = array();			return $domain;		}		if (isset($this->_domains[$domain][$this->_lang][$this->category][""])) {			$head = $this->_domains[$domain][$this->_lang][$this->category][""];			foreach (explode("\n", $head) as $line) {				$header = strtok($line,":");				$line = trim(strtok("\n"));				$this->_domains[$domain][$this->_lang][$this->category]["%po-header"][strtolower($header)] = $line;			}			if (isset($this->_domains[$domain][$this->_lang][$this->category]["%po-header"]["plural-forms"])) {				$switch = preg_replace("/(?:[() {}\\[\\]^\\s*\\]]+)/", "", $this->_domains[$domain][$this->_lang][$this->category]["%po-header"]["plural-forms"]);				$this->_domains[$domain][$this->_lang][$this->category]["%plural-c"] = $switch;				unset($this->_domains[$domain][$this->_lang][$this->category]["%po-header"]);			}			$this->_domains = Hash::mergeDiff($this->_domains, $merge);			if (isset($this->_domains[$domain][$this->_lang][$this->category][null])) {				unset($this->_domains[$domain][$this->_lang][$this->category][null]);			}		}		return $domain;	}/** * Loads the binary .mo file and returns array of translations * * @param string $filename Binary .mo file to load * @return mixed Array of translations on success or false on failure */	public static function loadMo($filename) {		$translations = false;		// @codingStandardsIgnoreStart		// Binary files extracted makes non-standard local variables		if ($data = file_get_contents($filename)) {			$translations = array();			$header = substr($data, 0, 20);			$header = unpack("L1magic/L1version/L1count/L1o_msg/L1o_trn", $header);			extract($header);			if ((dechex($magic) == '950412de' || dechex($magic) == 'ffffffff950412de') && !$version) {				for ($n = 0; $n < $count; $n++) {					$r = unpack("L1len/L1offs", substr($data, $o_msg + $n * 8, 8));					$msgid = substr($data, $r["offs"], $r["len"]);					unset($msgid_plural);					if (strpos($msgid, "\000")) {						list($msgid, $msgid_plural) = explode("\000", $msgid);					}					$r = unpack("L1len/L1offs", substr($data, $o_trn + $n * 8, 8));					$msgstr = substr($data, $r["offs"], $r["len"]);					if (strpos($msgstr, "\000")) {						$msgstr = explode("\000", $msgstr);					}					$translations[$msgid] = $msgstr;					if (isset($msgid_plural)) {						$translations[$msgid_plural] =& $translations[$msgid];					}				}			}		}		// @codingStandardsIgnoreEnd		return $translations;	}/** * Loads the text .po file and returns array of translations * * @param string $filename Text .po file to load * @return mixed Array of translations on success or false on failure */	public static function loadPo($filename) {		if (!$file = fopen($filename, "r")) {			return false;		}		$type = 0;		$translations = array();		$translationKey = "";		$plural = 0;		$header = "";		do {			$line = trim(fgets($file));			if ($line === "" || $line[0] === "#") {				continue;			}			if (preg_match("/msgid[[:space:]]+\"(.+)\"$/i", $line, $regs)) {				$type = 1;				$translationKey = stripcslashes($regs[1]);			} elseif (preg_match("/msgid[[:space:]]+\"\"$/i", $line, $regs)) {				$type = 2;				$translationKey = "";			} elseif (preg_match("/^\"(.*)\"$/i", $line, $regs) && ($type == 1 || $type == 2 || $type == 3)) {				$type = 3;				$translationKey .= stripcslashes($regs[1]);			} elseif (preg_match("/msgstr[[:space:]]+\"(.+)\"$/i", $line, $regs) && ($type == 1 || $type == 3) && $translationKey) {				$translations[$translationKey] = stripcslashes($regs[1]);				$type = 4;			} elseif (preg_match("/msgstr[[:space:]]+\"\"$/i", $line, $regs) && ($type == 1 || $type == 3) && $translationKey) {				$type = 4;				$translations[$translationKey] = "";			} elseif (preg_match("/^\"(.*)\"$/i", $line, $regs) && $type == 4 && $translationKey) {				$translations[$translationKey] .= stripcslashes($regs[1]);			} elseif (preg_match("/msgid_plural[[:space:]]+\".*\"$/i", $line, $regs)) {				$type = 6;			} elseif (preg_match("/^\"(.*)\"$/i", $line, $regs) && $type == 6 && $translationKey) {				$type = 6;			} elseif (preg_match("/msgstr\[(\d+)\][[:space:]]+\"(.+)\"$/i", $line, $regs) && ($type == 6 || $type == 7) && $translationKey) {				$plural = $regs[1];				$translations[$translationKey][$plural] = stripcslashes($regs[2]);				$type = 7;			} elseif (preg_match("/msgstr\[(\d+)\][[:space:]]+\"\"$/i", $line, $regs) && ($type == 6 || $type == 7) && $translationKey) {				$plural = $regs[1];				$translations[$translationKey][$plural] = "";				$type = 7;			} elseif (preg_match("/^\"(.*)\"$/i", $line, $regs) && $type == 7 && $translationKey) {				$translations[$translationKey][$plural] .= stripcslashes($regs[1]);			} elseif (preg_match("/msgstr[[:space:]]+\"(.+)\"$/i", $line, $regs) && $type == 2 && !$translationKey) {				$header .= stripcslashes($regs[1]);				$type = 5;			} elseif (preg_match("/msgstr[[:space:]]+\"\"$/i", $line, $regs) && !$translationKey) {				$header = "";				$type = 5;			} elseif (preg_match("/^\"(.*)\"$/i", $line, $regs) && $type == 5) {				$header .= stripcslashes($regs[1]);			} else {				unset($translations[$translationKey]);				$type = 0;				$translationKey = "";				$plural = 0;			}		} while (!feof($file));		fclose($file);		$merge[""] = $header;		return array_merge($merge, $translations);	}/** * Parses a locale definition file following the POSIX standard * * @param string $filename Locale definition filename * @return mixed Array of definitions on success or false on failure */	public static function loadLocaleDefinition($filename) {		if (!$file = fopen($filename, "r")) {			return false;		}		$definitions = array();		$comment = '#';		$escape = '\\';		$currentToken = false;		$value = '';		$_this = I18n::getInstance();		while ($line = fgets($file)) {			$line = trim($line);			if (empty($line) || $line[0] === $comment) {				continue;			}			$parts = preg_split("/[[:space:]]+/", $line);			if ($parts[0] === 'comment_char') {				$comment = $parts[1];				continue;			}			if ($parts[0] === 'escape_char') {				$escape = $parts[1];				continue;			}			$count = count($parts);			if ($count == 2) {				$currentToken = $parts[0];				$value = $parts[1];			} elseif ($count == 1) {				$value .= $parts[0];			} else {				continue;			}			$len = strlen($value) - 1;			if ($value[$len] === $escape) {				$value = substr($value, 0, $len);				continue;			}			$mustEscape = array($escape . ',', $escape . ';', $escape . '<', $escape . '>', $escape . $escape);			$replacements = array_map('crc32', $mustEscape);			$value = str_replace($mustEscape, $replacements, $value);			$value = explode(';', $value);			$_this->_escape = $escape;			foreach ($value as $i => $val) {				$val = trim($val, '"');				$val = preg_replace_callback('/(?:<)?(.[^>]*)(?:>)?/', array(&$_this, '_parseLiteralValue'), $val);				$val = str_replace($replacements, $mustEscape, $val);				$value[$i] = $val;			}			if (count($value) == 1) {				$definitions[$currentToken] = array_pop($value);			} else {				$definitions[$currentToken] = $value;			}		}		return $definitions;	}/** * Auxiliary function to parse a symbol from a locale definition file * * @param string $string Symbol to be parsed * @return string parsed symbol */	protected function _parseLiteralValue($string) {		$string = $string[1];		if (substr($string, 0, 2) === $this->_escape . 'x') {			$delimiter = $this->_escape . 'x';			return implode('', array_map('chr', array_map('hexdec',array_filter(explode($delimiter, $string)))));		}		if (substr($string, 0, 2) === $this->_escape . 'd') {			$delimiter = $this->_escape . 'd';			return implode('', array_map('chr', array_filter(explode($delimiter, $string))));		}		if ($string[0] === $this->_escape && isset($string[1]) && is_numeric($string[1])) {			$delimiter = $this->_escape;			return implode('', array_map('chr', array_filter(explode($delimiter, $string))));		}		if (substr($string, 0, 3) === 'U00') {			$delimiter = 'U00';			return implode('', array_map('chr', array_map('hexdec', array_filter(explode($delimiter, $string)))));		}		if (preg_match('/U([0-9a-fA-F]{4})/', $string, $match)) {			return Multibyte::ascii(array(hexdec($match[1])));		}		return $string;	}/** * Returns a Time format definition from corresponding domain * * @param string $format Format to be translated * @param string $domain Domain where format is stored * @return mixed translated format string if only value or array of translated strings for corresponding format. */	protected function _translateTime($format, $domain) {		if (!empty($this->_domains[$domain][$this->_lang]['LC_TIME'][$format])) {			if (($trans = $this->_domains[$domain][$this->_lang][$this->category][$format])) {				return $trans;			}		}		return $format;	}}
 |