You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
117 lines
3.8 KiB
117 lines
3.8 KiB
<?php |
|
/** |
|
* @link http://www.yiiframework.com/ |
|
* @copyright Copyright (c) 2008 Yii Software LLC |
|
* @license http://www.yiiframework.com/license/ |
|
*/ |
|
|
|
namespace yii\i18n; |
|
|
|
/** |
|
* MessageFormatter is an enhanced version of PHP intl class that no matter which PHP and ICU versions are used: |
|
* |
|
* - Accepts named arguments and mixed numeric and named arguments. |
|
* - Issues no error when an insufficient number of arguments have been provided. Instead, the placeholders will not be |
|
* substituted. |
|
* - Fixes PHP 5.5 weird placeholder replacement in case no arguments are provided at all (https://bugs.php.net/bug.php?id=65920). |
|
* |
|
* @author Alexander Makarov <sam@rmcreative.ru> |
|
* @author Carsten Brandt <mail@cebe.cc> |
|
* @since 2.0 |
|
*/ |
|
class MessageFormatter extends \MessageFormatter |
|
{ |
|
/** |
|
* Format the message. |
|
* |
|
* @link http://php.net/manual/en/messageformatter.format.php |
|
* @param array $args Arguments to insert into the format string. |
|
* @return string|boolean The formatted string, or false if an error occurred. |
|
*/ |
|
public function format($args) |
|
{ |
|
if ($args === []) { |
|
return $this->getPattern(); |
|
} |
|
|
|
if (version_compare(PHP_VERSION, '5.5.0', '<')) { |
|
$pattern = self::replaceNamedArguments($this->getPattern(), $args); |
|
$this->setPattern($pattern); |
|
$args = array_values($args); |
|
} |
|
return parent::format($args); |
|
} |
|
|
|
/** |
|
* Quick format message. |
|
* |
|
* @link http://php.net/manual/en/messageformatter.formatmessage.php |
|
* @param string $locale The locale to use for formatting locale-dependent parts. |
|
* @param string $pattern The pattern string to insert things into. |
|
* @param array $args The array of values to insert into the format string. |
|
* @return string|boolean The formatted pattern string or false if an error occurred. |
|
*/ |
|
public static function formatMessage($locale, $pattern, $args) |
|
{ |
|
if ($args === []) { |
|
return $pattern; |
|
} |
|
|
|
if (version_compare(PHP_VERSION, '5.5.0', '<')) { |
|
$pattern = self::replaceNamedArguments($pattern, $args); |
|
$args = array_values($args); |
|
} |
|
return parent::formatMessage($locale, $pattern, $args); |
|
} |
|
|
|
/** |
|
* Replace named placeholders with numeric placeholders and quote unused. |
|
* |
|
* @param string $pattern The pattern string to replace things into. |
|
* @param array $args The array of values to insert into the format string. |
|
* @return string The pattern string with placeholders replaced. |
|
*/ |
|
private static function replaceNamedArguments($pattern, $args) |
|
{ |
|
$map = array_flip(array_keys($args)); |
|
|
|
// parsing pattern based on ICU grammar: |
|
// http://icu-project.org/apiref/icu4c/classMessageFormat.html#details |
|
$parts = explode('{', $pattern); |
|
$c = count($parts); |
|
$pattern = $parts[0]; |
|
$d = 0; |
|
$stack = []; |
|
for($i = 1; $i < $c; $i++) { |
|
if (preg_match('~^(\s*)([\d\w]+)(\s*)([},])(\s*)(.*)$~us', $parts[$i], $matches)) { |
|
// if we are not inside a plural or select this is a message |
|
if (!isset($stack[$d]) || $stack[$d] != 'plural' && $stack[$d] != 'select') { |
|
$d++; |
|
// replace normal arg if it is available |
|
if (isset($map[$matches[2]])) { |
|
$q = ''; |
|
$pattern .= '{' . $matches[1] . $map[$matches[2]] . $matches[3]; |
|
} else { |
|
// quote unused args |
|
$q = ($matches[4] == '}') ? "'" : ""; |
|
$pattern .= "$q{" . $matches[1] . $matches[2] . $matches[3]; |
|
} |
|
$pattern .= ($term = $matches[4] . $q . $matches[5] . $matches[6]); |
|
// store type of current level |
|
$stack[$d] = ($matches[4] == ',') ? substr($matches[6], 0, 6) : 'none'; |
|
// if it's plural or select, the next bracket is NOT begin of a message then! |
|
if ($stack[$d] == 'plural' || $stack[$d] == 'select') { |
|
$i++; |
|
$d -= substr_count($term, '}'); |
|
} else { |
|
$d -= substr_count($term, '}'); |
|
continue; |
|
} |
|
} |
|
} |
|
$pattern .= '{' . $parts[$i]; |
|
$d += 1 - substr_count($parts[$i], '}'); |
|
} |
|
return $pattern; |
|
} |
|
}
|
|
|