| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368 | 
							- <?php
 
- /*
 
-  * This file is part of the Symfony package.
 
-  *
 
-  * (c) Fabien Potencier <fabien@symfony.com>
 
-  *
 
-  * For the full copyright and license information, please view the LICENSE
 
-  * file that was distributed with this source code.
 
-  */
 
- namespace Symfony\Component\Translation;
 
- use Symfony\Contracts\Translation\TranslatorInterface;
 
- /**
 
-  * This translator should only be used in a development environment.
 
-  */
 
- final class PseudoLocalizationTranslator implements TranslatorInterface
 
- {
 
-     private const EXPANSION_CHARACTER = '~';
 
-     private $translator;
 
-     private $accents;
 
-     private $expansionFactor;
 
-     private $brackets;
 
-     private $parseHTML;
 
-     /**
 
-      * @var string[]
 
-      */
 
-     private $localizableHTMLAttributes;
 
-     /**
 
-      * Available options:
 
-      *  * accents:
 
-      *      type: boolean
 
-      *      default: true
 
-      *      description: replace ASCII characters of the translated string with accented versions or similar characters
 
-      *      example: if true, "foo" => "ƒöö".
 
-      *
 
-      *  * expansion_factor:
 
-      *      type: float
 
-      *      default: 1
 
-      *      validation: it must be greater than or equal to 1
 
-      *      description: expand the translated string by the given factor with spaces and tildes
 
-      *      example: if 2, "foo" => "~foo ~"
 
-      *
 
-      *  * brackets:
 
-      *      type: boolean
 
-      *      default: true
 
-      *      description: wrap the translated string with brackets
 
-      *      example: if true, "foo" => "[foo]"
 
-      *
 
-      *  * parse_html:
 
-      *      type: boolean
 
-      *      default: false
 
-      *      description: parse the translated string as HTML - looking for HTML tags has a performance impact but allows to preserve them from alterations - it also allows to compute the visible translated string length which is useful to correctly expand ot when it contains HTML
 
-      *      warning: unclosed tags are unsupported, they will be fixed (closed) by the parser - eg, "foo <div>bar" => "foo <div>bar</div>"
 
-      *
 
-      *  * localizable_html_attributes:
 
-      *      type: string[]
 
-      *      default: []
 
-      *      description: the list of HTML attributes whose values can be altered - it is only useful when the "parse_html" option is set to true
 
-      *      example: if ["title"], and with the "accents" option set to true, "<a href="#" title="Go to your profile">Profile</a>" => "<a href="#" title="Ĝö ţö ýöûŕ þŕöƒîļé">Þŕöƒîļé</a>" - if "title" was not in the "localizable_html_attributes" list, the title attribute data would be left unchanged.
 
-      */
 
-     public function __construct(TranslatorInterface $translator, array $options = [])
 
-     {
 
-         $this->translator = $translator;
 
-         $this->accents = $options['accents'] ?? true;
 
-         if (1.0 > ($this->expansionFactor = $options['expansion_factor'] ?? 1.0)) {
 
-             throw new \InvalidArgumentException('The expansion factor must be greater than or equal to 1.');
 
-         }
 
-         $this->brackets = $options['brackets'] ?? true;
 
-         $this->parseHTML = $options['parse_html'] ?? false;
 
-         if ($this->parseHTML && !$this->accents && 1.0 === $this->expansionFactor) {
 
-             $this->parseHTML = false;
 
-         }
 
-         $this->localizableHTMLAttributes = $options['localizable_html_attributes'] ?? [];
 
-     }
 
-     /**
 
-      * {@inheritdoc}
 
-      */
 
-     public function trans(string $id, array $parameters = [], ?string $domain = null, ?string $locale = null): string
 
-     {
 
-         $trans = '';
 
-         $visibleText = '';
 
-         foreach ($this->getParts($this->translator->trans($id, $parameters, $domain, $locale)) as [$visible, $localizable, $text]) {
 
-             if ($visible) {
 
-                 $visibleText .= $text;
 
-             }
 
-             if (!$localizable) {
 
-                 $trans .= $text;
 
-                 continue;
 
-             }
 
-             $this->addAccents($trans, $text);
 
-         }
 
-         $this->expand($trans, $visibleText);
 
-         $this->addBrackets($trans);
 
-         return $trans;
 
-     }
 
-     public function getLocale(): string
 
-     {
 
-         return $this->translator->getLocale();
 
-     }
 
-     private function getParts(string $originalTrans): array
 
-     {
 
-         if (!$this->parseHTML) {
 
-             return [[true, true, $originalTrans]];
 
-         }
 
-         $html = mb_encode_numericentity($originalTrans, [0x80, 0x10FFFF, 0, 0x1FFFFF], mb_detect_encoding($originalTrans, null, true) ?: 'UTF-8');
 
-         $useInternalErrors = libxml_use_internal_errors(true);
 
-         $dom = new \DOMDocument();
 
-         $dom->loadHTML('<trans>'.$html.'</trans>');
 
-         libxml_clear_errors();
 
-         libxml_use_internal_errors($useInternalErrors);
 
-         return $this->parseNode($dom->childNodes->item(1)->childNodes->item(0)->childNodes->item(0));
 
-     }
 
-     private function parseNode(\DOMNode $node): array
 
-     {
 
-         $parts = [];
 
-         foreach ($node->childNodes as $childNode) {
 
-             if (!$childNode instanceof \DOMElement) {
 
-                 $parts[] = [true, true, $childNode->nodeValue];
 
-                 continue;
 
-             }
 
-             $parts[] = [false, false, '<'.$childNode->tagName];
 
-             /** @var \DOMAttr $attribute */
 
-             foreach ($childNode->attributes as $attribute) {
 
-                 $parts[] = [false, false, ' '.$attribute->nodeName.'="'];
 
-                 $localizableAttribute = \in_array($attribute->nodeName, $this->localizableHTMLAttributes, true);
 
-                 foreach (preg_split('/(&(?:amp|quot|#039|lt|gt);+)/', htmlspecialchars($attribute->nodeValue, \ENT_QUOTES, 'UTF-8'), -1, \PREG_SPLIT_DELIM_CAPTURE) as $i => $match) {
 
-                     if ('' === $match) {
 
-                         continue;
 
-                     }
 
-                     $parts[] = [false, $localizableAttribute && 0 === $i % 2, $match];
 
-                 }
 
-                 $parts[] = [false, false, '"'];
 
-             }
 
-             $parts[] = [false, false, '>'];
 
-             $parts = array_merge($parts, $this->parseNode($childNode, $parts));
 
-             $parts[] = [false, false, '</'.$childNode->tagName.'>'];
 
-         }
 
-         return $parts;
 
-     }
 
-     private function addAccents(string &$trans, string $text): void
 
-     {
 
-         $trans .= $this->accents ? strtr($text, [
 
-             ' ' => ' ',
 
-             '!' => '¡',
 
-             '"' => '″',
 
-             '#' => '♯',
 
-             '$' => '€',
 
-             '%' => '‰',
 
-             '&' => '⅋',
 
-             '\'' => '´',
 
-             '(' => '{',
 
-             ')' => '}',
 
-             '*' => '⁎',
 
-             '+' => '⁺',
 
-             ',' => '،',
 
-             '-' => '‐',
 
-             '.' => '·',
 
-             '/' => '⁄',
 
-             '0' => '⓪',
 
-             '1' => '①',
 
-             '2' => '②',
 
-             '3' => '③',
 
-             '4' => '④',
 
-             '5' => '⑤',
 
-             '6' => '⑥',
 
-             '7' => '⑦',
 
-             '8' => '⑧',
 
-             '9' => '⑨',
 
-             ':' => '∶',
 
-             ';' => '⁏',
 
-             '<' => '≤',
 
-             '=' => '≂',
 
-             '>' => '≥',
 
-             '?' => '¿',
 
-             '@' => '՞',
 
-             'A' => 'Å',
 
-             'B' => 'Ɓ',
 
-             'C' => 'Ç',
 
-             'D' => 'Ð',
 
-             'E' => 'É',
 
-             'F' => 'Ƒ',
 
-             'G' => 'Ĝ',
 
-             'H' => 'Ĥ',
 
-             'I' => 'Î',
 
-             'J' => 'Ĵ',
 
-             'K' => 'Ķ',
 
-             'L' => 'Ļ',
 
-             'M' => 'Ṁ',
 
-             'N' => 'Ñ',
 
-             'O' => 'Ö',
 
-             'P' => 'Þ',
 
-             'Q' => 'Ǫ',
 
-             'R' => 'Ŕ',
 
-             'S' => 'Š',
 
-             'T' => 'Ţ',
 
-             'U' => 'Û',
 
-             'V' => 'Ṽ',
 
-             'W' => 'Ŵ',
 
-             'X' => 'Ẋ',
 
-             'Y' => 'Ý',
 
-             'Z' => 'Ž',
 
-             '[' => '⁅',
 
-             '\\' => '∖',
 
-             ']' => '⁆',
 
-             '^' => '˄',
 
-             '_' => '‿',
 
-             '`' => '‵',
 
-             'a' => 'å',
 
-             'b' => 'ƀ',
 
-             'c' => 'ç',
 
-             'd' => 'ð',
 
-             'e' => 'é',
 
-             'f' => 'ƒ',
 
-             'g' => 'ĝ',
 
-             'h' => 'ĥ',
 
-             'i' => 'î',
 
-             'j' => 'ĵ',
 
-             'k' => 'ķ',
 
-             'l' => 'ļ',
 
-             'm' => 'ɱ',
 
-             'n' => 'ñ',
 
-             'o' => 'ö',
 
-             'p' => 'þ',
 
-             'q' => 'ǫ',
 
-             'r' => 'ŕ',
 
-             's' => 'š',
 
-             't' => 'ţ',
 
-             'u' => 'û',
 
-             'v' => 'ṽ',
 
-             'w' => 'ŵ',
 
-             'x' => 'ẋ',
 
-             'y' => 'ý',
 
-             'z' => 'ž',
 
-             '{' => '(',
 
-             '|' => '¦',
 
-             '}' => ')',
 
-             '~' => '˞',
 
-         ]) : $text;
 
-     }
 
-     private function expand(string &$trans, string $visibleText): void
 
-     {
 
-         if (1.0 >= $this->expansionFactor) {
 
-             return;
 
-         }
 
-         $visibleLength = $this->strlen($visibleText);
 
-         $missingLength = (int) ceil($visibleLength * $this->expansionFactor) - $visibleLength;
 
-         if ($this->brackets) {
 
-             $missingLength -= 2;
 
-         }
 
-         if (0 >= $missingLength) {
 
-             return;
 
-         }
 
-         $words = [];
 
-         $wordsCount = 0;
 
-         foreach (preg_split('/ +/', $visibleText, -1, \PREG_SPLIT_NO_EMPTY) as $word) {
 
-             $wordLength = $this->strlen($word);
 
-             if ($wordLength >= $missingLength) {
 
-                 continue;
 
-             }
 
-             if (!isset($words[$wordLength])) {
 
-                 $words[$wordLength] = 0;
 
-             }
 
-             ++$words[$wordLength];
 
-             ++$wordsCount;
 
-         }
 
-         if (!$words) {
 
-             $trans .= 1 === $missingLength ? self::EXPANSION_CHARACTER : ' '.str_repeat(self::EXPANSION_CHARACTER, $missingLength - 1);
 
-             return;
 
-         }
 
-         arsort($words, \SORT_NUMERIC);
 
-         $longestWordLength = max(array_keys($words));
 
-         while (true) {
 
-             $r = mt_rand(1, $wordsCount);
 
-             foreach ($words as $length => $count) {
 
-                 $r -= $count;
 
-                 if ($r <= 0) {
 
-                     break;
 
-                 }
 
-             }
 
-             $trans .= ' '.str_repeat(self::EXPANSION_CHARACTER, $length);
 
-             $missingLength -= $length + 1;
 
-             if (0 === $missingLength) {
 
-                 return;
 
-             }
 
-             while ($longestWordLength >= $missingLength) {
 
-                 $wordsCount -= $words[$longestWordLength];
 
-                 unset($words[$longestWordLength]);
 
-                 if (!$words) {
 
-                     $trans .= 1 === $missingLength ? self::EXPANSION_CHARACTER : ' '.str_repeat(self::EXPANSION_CHARACTER, $missingLength - 1);
 
-                     return;
 
-                 }
 
-                 $longestWordLength = max(array_keys($words));
 
-             }
 
-         }
 
-     }
 
-     private function addBrackets(string &$trans): void
 
-     {
 
-         if (!$this->brackets) {
 
-             return;
 
-         }
 
-         $trans = '['.$trans.']';
 
-     }
 
-     private function strlen(string $s): int
 
-     {
 
-         return false === ($encoding = mb_detect_encoding($s, null, true)) ? \strlen($s) : mb_strlen($s, $encoding);
 
-     }
 
- }
 
 
  |