| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 | <?php/** * Lithium: the most rad php framework * * @copyright     Copyright 2013, Union of RAD (http://union-of-rad.org) * @license       http://opensource.org/licenses/bsd-license.php The BSD License */namespace lithium\g11n;use lithium\core\Environment;use lithium\util\String;use lithium\g11n\Catalog;/** * The `Message` class is concerned with an aspect of globalizing static message strings * throughout the framework and applications.  When referring to message globalization the * phrase of ""translating a message" is widely used. This leads to the assumption that it's * a single step process whereas it' a multi step one. A short description of each step is * given here in order to help understanding the purpose of this class through the context * of the process as a whole. * *  1. Marking messages as translatable.  `$t()` and `$tn()` (implemented in `aliases()`) *     are recognized as message marking and picked up by the extraction parser. * *  2. Extracting marked messages.  Messages can be extracted through the `g11n` *     command which in turn utilizes the `Catalog` class with the built-in `Code` *     adapter or other custom adapters which are concerned with extracting *     translatable content. * *  3. Creating a message template from extracted messages.  Templates are created *     by the `g11n` command using the `Catalog` class with an adapter for a format *     you prefer. * *  4. Translating messages.  The actual translation of messages by translators *     happens outside using external applications. * *  5. Storing translated messages.  Translations are most often stored by the external *     applications itself. * *  6. Retrieving the translation for a message. See description for `Message::translate()`. * * @see lithium\g11n\Catalog * @see lithium\console\command\G11n * @see lithium\g11n\catalog\adapter\Code */class Message extends \lithium\core\StaticObject {	/**	 * Holds cached message pages generated and used	 * by `lithium\g11n\Message::_translated()`.	 *	 * @var array	 * @see lithium\g11n\Message::_translated()	 */	protected static $_cachedPages = array();	/**	 * Translates a message according to the current or provided locale	 * and into it's correct plural form.	 *	 * Usage:	 * {{{	 * Message::translate('Mind the gap.');	 * Message::translate('house', array('count' => 23));	 * }}}	 *	 * `String::insert()`-style placeholders may be used within the message	 * and replacements provided directly within the `options`  argument.	 *	 * Example:	 * {{{	 * Message::translate('I can see {:count} bike.');	 * Message::translate('This painting is {:color}.', array(	 * 	'color' => Message::translate('silver'),	 * ));	 * }}}	 *	 * @see lithium\util\String::insert()	 * @param string $id The id to use when looking up the translation.	 * @param array $options Valid options are:	 *              - `'count'`: Used to determine the correct plural form. You can either pass	 *                           a signed or unsigned integer, the behavior when passing other types	 *                           is yet undefined.	 *                           The count is made absolute before being passed to the pluralization	 *                           function. This has the effect that that with i.e. an English	 *                           pluralization function passing `-1` results in a singular	 *                           translation.	 *              - `'locale'`: The target locale, defaults to current locale.	 *              - `'scope'`: The scope of the message.	 *              - `'default'`: Is used as a fall back if `_translated()` returns	 *                             without a result.	 *              - `'noop'`: If `true` no whatsoever lookup takes place.	 * @return string The translation or the value of the `'default'` option if none	 *                     could be found.	 */	public static function translate($id, array $options = array()) {		$defaults = array(			'count' => 1,			'locale' => Environment::get('locale'),			'scope' => null,			'default' => null,			'noop' => false		);		$options += $defaults;		if ($options['noop']) {			$result = null;		} else {			$result = static::_translated($id, abs($options['count']), $options['locale'], array(				'scope' => $options['scope']			));		}		if ($result || $options['default']) {			return String::insert($result ?: $options['default'], $options);		}	}	/**	 * Returns an array containing named closures which are aliases for `translate()`.	 * They can be embedded as content filters in the template layer using a filter for	 * `Media::_handle()` or be used in other places where needed.	 *	 * Usage:	 * {{{	 * 	$t('bike');	 * 	$tn('bike', 'bikes', 3);	 * }}}	 *	 * Using in a method:	 * {{{	 * 	public function index() {	 * 		extract(Message::aliases());	 * 		$notice = $t('look');	 * 	}	 * }}}	 *	 * @see lithium\net\http\Media::_handle()	 * @return array Named aliases (`'t'` and `'tn'`) for translation functions.	 */	public static function aliases() {		$t = function($message, array $options = array()) {			return Message::translate($message, $options + array('default' => $message));		};		$tn = function($message1, $message2, $count, array $options = array()) {			return Message::translate($message1, $options + compact('count') + array(				'default' => $count == 1 ? $message1 : $message2			));		};		return compact('t', 'tn');	}	/**	 * Returns or sets the page cache used for mapping message ids to translations.	 *	 * @param array $cache A multidimensional array to use when pre-populating the cache. The	 *              structure of the array is `scope/locale/id`. If `false`, the cache is cleared.	 * @return array Returns an array of cached pages, formatted per the description for `$cache`.	 */	public static function cache($cache = null) {		if ($cache === false) {			static::$_cachedPages = array();		}		if (is_array($cache)) {			static::$_cachedPages += $cache;		}		return static::$_cachedPages;	}	/**	 * Retrieves translations through the `Catalog` class by using `$id` as the lookup	 * key and taking the current or - if specified - the provided locale as well as the	 * scope into account.  Hereupon the correct plural form is determined by passing the	 * value of the `'count'` option to a closure.	 *	 * @see lithium\g11n\Catalog	 * @param string $id The lookup key.	 * @param integer $count Used to determine the correct plural form.	 * @param string $locale The target locale.	 * @param array $options Passed through to `Catalog::read()`. Valid options are:	 *              - `'scope'`: The scope of the message.	 * @return string The translation or `null` if none could be found or the plural	 *         form could not be determined.	 * @filter	 */	protected static function _translated($id, $count, $locale, array $options = array()) {		$params = compact('id', 'count', 'locale', 'options');		$cache =& static::$_cachedPages;		return static::_filter(__FUNCTION__, $params, function($self, $params) use (&$cache) {			extract($params);			if (!isset($cache[$options['scope']][$locale])) {				$cache[$options['scope']][$locale] = Catalog::read(					true, 'message', $locale, $options				);			}			$page = $cache[$options['scope']][$locale];			if (!isset($page[$id])) {				return null;			}			if (!is_array($page[$id])) {				return $page[$id];			}			if (!isset($page['pluralRule']) || !is_callable($page['pluralRule'])) {				return null;			}			$key = $page['pluralRule']($count);			if (isset($page[$id][$key])) {				return $page[$id][$key];			}		});	}}?>
 |