vendor/symfony/string/AbstractUnicodeString.php line 29

  1. <?php
  2. /*
  3.  * This file is part of the Symfony package.
  4.  *
  5.  * (c) Fabien Potencier <fabien@symfony.com>
  6.  *
  7.  * For the full copyright and license information, please view the LICENSE
  8.  * file that was distributed with this source code.
  9.  */
  10. namespace Symfony\Component\String;
  11. use Symfony\Component\String\Exception\ExceptionInterface;
  12. use Symfony\Component\String\Exception\InvalidArgumentException;
  13. use Symfony\Component\String\Exception\RuntimeException;
  14. /**
  15.  * Represents a string of abstract Unicode characters.
  16.  *
  17.  * Unicode defines 3 types of "characters" (bytes, code points and grapheme clusters).
  18.  * This class is the abstract type to use as a type-hint when the logic you want to
  19.  * implement is Unicode-aware but doesn't care about code points vs grapheme clusters.
  20.  *
  21.  * @author Nicolas Grekas <p@tchwork.com>
  22.  *
  23.  * @throws ExceptionInterface
  24.  */
  25. abstract class AbstractUnicodeString extends AbstractString
  26. {
  27.     public const NFC \Normalizer::NFC;
  28.     public const NFD \Normalizer::NFD;
  29.     public const NFKC \Normalizer::NFKC;
  30.     public const NFKD \Normalizer::NFKD;
  31.     // all ASCII letters sorted by typical frequency of occurrence
  32.     private const ASCII "\x20\x65\x69\x61\x73\x6E\x74\x72\x6F\x6C\x75\x64\x5D\x5B\x63\x6D\x70\x27\x0A\x67\x7C\x68\x76\x2E\x66\x62\x2C\x3A\x3D\x2D\x71\x31\x30\x43\x32\x2A\x79\x78\x29\x28\x4C\x39\x41\x53\x2F\x50\x22\x45\x6A\x4D\x49\x6B\x33\x3E\x35\x54\x3C\x44\x34\x7D\x42\x7B\x38\x46\x77\x52\x36\x37\x55\x47\x4E\x3B\x4A\x7A\x56\x23\x48\x4F\x57\x5F\x26\x21\x4B\x3F\x58\x51\x25\x59\x5C\x09\x5A\x2B\x7E\x5E\x24\x40\x60\x7F\x00\x01\x02\x03\x04\x05\x06\x07\x08\x0B\x0C\x0D\x0E\x0F\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1A\x1B\x1C\x1D\x1E\x1F";
  33.     // the subset of folded case mappings that is not in lower case mappings
  34.     private const FOLD_FROM = ['İ''µ''ſ'"\xCD\x85"'ς''ϐ''ϑ''ϕ''ϖ''ϰ''ϱ''ϵ''ẛ'"\xE1\xBE\xBE"'ß''ʼn''ǰ''ΐ''ΰ''և''ẖ''ẗ''ẘ''ẙ''ẚ''ẞ''ὐ''ὒ''ὔ''ὖ''ᾀ''ᾁ''ᾂ''ᾃ''ᾄ''ᾅ''ᾆ''ᾇ''ᾈ''ᾉ''ᾊ''ᾋ''ᾌ''ᾍ''ᾎ''ᾏ''ᾐ''ᾑ''ᾒ''ᾓ''ᾔ''ᾕ''ᾖ''ᾗ''ᾘ''ᾙ''ᾚ''ᾛ''ᾜ''ᾝ''ᾞ''ᾟ''ᾠ''ᾡ''ᾢ''ᾣ''ᾤ''ᾥ''ᾦ''ᾧ''ᾨ''ᾩ''ᾪ''ᾫ''ᾬ''ᾭ''ᾮ''ᾯ''ᾲ''ᾳ''ᾴ''ᾶ''ᾷ''ᾼ''ῂ''ῃ''ῄ''ῆ''ῇ''ῌ''ῒ''ῖ''ῗ''ῢ''ῤ''ῦ''ῧ''ῲ''ῳ''ῴ''ῶ''ῷ''ῼ''ff''fi''fl''ffi''ffl''ſt''st''ﬓ''ﬔ''ﬕ''ﬖ''ﬗ'];
  35.     private const FOLD_TO = ['i̇''μ''s''ι''σ''β''θ''φ''π''κ''ρ''ε''ṡ''ι''ss''ʼn''ǰ''ΐ''ΰ''եւ''ẖ''ẗ''ẘ''ẙ''aʾ''ss''ὐ''ὒ''ὔ''ὖ''ἀι''ἁι''ἂι''ἃι''ἄι''ἅι''ἆι''ἇι''ἀι''ἁι''ἂι''ἃι''ἄι''ἅι''ἆι''ἇι''ἠι''ἡι''ἢι''ἣι''ἤι''ἥι''ἦι''ἧι''ἠι''ἡι''ἢι''ἣι''ἤι''ἥι''ἦι''ἧι''ὠι''ὡι''ὢι''ὣι''ὤι''ὥι''ὦι''ὧι''ὠι''ὡι''ὢι''ὣι''ὤι''ὥι''ὦι''ὧι''ὰι''αι''άι''ᾶ''ᾶι''αι''ὴι''ηι''ήι''ῆ''ῆι''ηι''ῒ''ῖ''ῗ''ῢ''ῤ''ῦ''ῧ''ὼι''ωι''ώι''ῶ''ῶι''ωι''ff''fi''fl''ffi''ffl''st''st''մն''մե''մի''վն''մխ'];
  36.     // the subset of upper case mappings that map one code point to many code points
  37.     private const UPPER_FROM = ['ß''ff''fi''fl''ffi''ffl''ſt''st''և''ﬓ''ﬔ''ﬕ''ﬖ''ﬗ''ʼn''ΐ''ΰ''ǰ''ẖ''ẗ''ẘ''ẙ''ẚ''ὐ''ὒ''ὔ''ὖ''ᾶ''ῆ''ῒ''ΐ''ῖ''ῗ''ῢ''ΰ''ῤ''ῦ''ῧ''ῶ'];
  38.     private const UPPER_TO = ['SS''FF''FI''FL''FFI''FFL''ST''ST''ԵՒ''ՄՆ''ՄԵ''ՄԻ''ՎՆ''ՄԽ''ʼN''Ϊ́''Ϋ́''J̌''H̱''T̈''W̊''Y̊''Aʾ''Υ̓''Υ̓̀''Υ̓́''Υ̓͂''Α͂''Η͂''Ϊ̀''Ϊ́''Ι͂''Ϊ͂''Ϋ̀''Ϋ́''Ρ̓''Υ͂''Ϋ͂''Ω͂'];
  39.     // the subset of https://github.com/unicode-org/cldr/blob/master/common/transforms/Latin-ASCII.xml that is not in NFKD
  40.     private const TRANSLIT_FROM = ['Æ''Ð''Ø''Þ''ß''æ''ð''ø''þ''Đ''đ''Ħ''ħ''ı''ĸ''Ŀ''ŀ''Ł''ł''ʼn''Ŋ''ŋ''Œ''œ''Ŧ''ŧ''ƀ''Ɓ''Ƃ''ƃ''Ƈ''ƈ''Ɖ''Ɗ''Ƌ''ƌ''Ɛ''Ƒ''ƒ''Ɠ''ƕ''Ɩ''Ɨ''Ƙ''ƙ''ƚ''Ɲ''ƞ''Ƣ''ƣ''Ƥ''ƥ''ƫ''Ƭ''ƭ''Ʈ''Ʋ''Ƴ''ƴ''Ƶ''ƶ''DŽ''Dž''dž''Ǥ''ǥ''ȡ''Ȥ''ȥ''ȴ''ȵ''ȶ''ȷ''ȸ''ȹ''Ⱥ''Ȼ''ȼ''Ƚ''Ⱦ''ȿ''ɀ''Ƀ''Ʉ''Ɇ''ɇ''Ɉ''ɉ''Ɍ''ɍ''Ɏ''ɏ''ɓ''ɕ''ɖ''ɗ''ɛ''ɟ''ɠ''ɡ''ɢ''ɦ''ɧ''ɨ''ɪ''ɫ''ɬ''ɭ''ɱ''ɲ''ɳ''ɴ''ɶ''ɼ''ɽ''ɾ''ʀ''ʂ''ʈ''ʉ''ʋ''ʏ''ʐ''ʑ''ʙ''ʛ''ʜ''ʝ''ʟ''ʠ''ʣ''ʥ''ʦ''ʪ''ʫ''ᴀ''ᴁ''ᴃ''ᴄ''ᴅ''ᴆ''ᴇ''ᴊ''ᴋ''ᴌ''ᴍ''ᴏ''ᴘ''ᴛ''ᴜ''ᴠ''ᴡ''ᴢ''ᵫ''ᵬ''ᵭ''ᵮ''ᵯ''ᵰ''ᵱ''ᵲ''ᵳ''ᵴ''ᵵ''ᵶ''ᵺ''ᵻ''ᵽ''ᵾ''ᶀ''ᶁ''ᶂ''ᶃ''ᶄ''ᶅ''ᶆ''ᶇ''ᶈ''ᶉ''ᶊ''ᶌ''ᶍ''ᶎ''ᶏ''ᶑ''ᶒ''ᶓ''ᶖ''ᶙ''ẚ''ẜ''ẝ''ẞ''Ỻ''ỻ''Ỽ''ỽ''Ỿ''ỿ''©''®''₠''₢''₣''₤''₧''₺''₹''ℌ''℞''㎧''㎮''㏆''㏗''㏞''㏟''¼''½''¾''⅓''⅔''⅕''⅖''⅗''⅘''⅙''⅚''⅛''⅜''⅝''⅞''⅟''〇''‘''’''‚''‛''“''”''„''‟''′''″''〝''〞''«''»''‹''›''‐''‑''‒''–''—''―''︱''︲''﹘''‖''⁄''⁅''⁆''⁎''、''。''〈''〉''《''》''〔''〕''〘''〙''〚''〛''︑''︒''︹''︺''︽''︾''︿''﹀''﹑''﹝''﹞''⦅''⦆''。''、''×''÷''−''∕''∖''∣''∥''≪''≫''⦅''⦆'];
  41.     private const TRANSLIT_TO = ['AE''D''O''TH''ss''ae''d''o''th''D''d''H''h''i''q''L''l''L''l''\'n''N''n''OE''oe''T''t''b''B''B''b''C''c''D''D''D''d''E''F''f''G''hv''I''I''K''k''l''N''n''OI''oi''P''p''t''T''t''T''V''Y''y''Z''z''DZ''Dz''dz''G''g''d''Z''z''l''n''t''j''db''qp''A''C''c''L''T''s''z''B''U''E''e''J''j''R''r''Y''y''b''c''d''d''e''j''g''g''G''h''h''i''I''l''l''l''m''n''n''N''OE''r''r''r''R''s''t''u''v''Y''z''z''B''G''H''j''L''q''dz''dz''ts''ls''lz''A''AE''B''C''D''D''E''J''K''L''M''O''P''T''U''V''W''Z''ue''b''d''f''m''n''p''r''r''s''t''z''th''I''p''U''b''d''f''g''k''l''m''n''p''r''s''v''x''z''a''d''e''e''i''u''a''s''s''SS''LL''ll''V''v''Y''y''(C)''(R)''CE''Cr''Fr.''L.''Pts''TL''Rs''x''Rx''m/s''rad/s''C/kg''pH''V/m''A/m'' 1/4'' 1/2'' 3/4'' 1/3'' 2/3'' 1/5'' 2/5'' 3/5'' 4/5'' 1/6'' 5/6'' 1/8'' 3/8'' 5/8'' 7/8'' 1/''0''\'''\''',''\'''"''"'',,''"''\'''"''"''"''<<''>>''<''>''-''-''-''-''-''-''-''-''-''||''/''['']''*'',''.''<''>''<<''>>''['']''['']''['']'',''.''['']''<<''>>''<''>'',''['']''((''))''.'',''*''/''-''/''\\''|''||''<<''>>''((''))'];
  42.     private static $transliterators = [];
  43.     private static $tableZero;
  44.     private static $tableWide;
  45.     public static function fromCodePoints(int ...$codes): static
  46.     {
  47.         $string '';
  48.         foreach ($codes as $code) {
  49.             if (0x80 $code %= 0x200000) {
  50.                 $string .= \chr($code);
  51.             } elseif (0x800 $code) {
  52.                 $string .= \chr(0xC0 $code >> 6).\chr(0x80 $code 0x3F);
  53.             } elseif (0x10000 $code) {
  54.                 $string .= \chr(0xE0 $code >> 12).\chr(0x80 $code >> 0x3F).\chr(0x80 $code 0x3F);
  55.             } else {
  56.                 $string .= \chr(0xF0 $code >> 18).\chr(0x80 $code >> 12 0x3F).\chr(0x80 $code >> 0x3F).\chr(0x80 $code 0x3F);
  57.             }
  58.         }
  59.         return new static($string);
  60.     }
  61.     /**
  62.      * Generic UTF-8 to ASCII transliteration.
  63.      *
  64.      * Install the intl extension for best results.
  65.      *
  66.      * @param string[]|\Transliterator[]|\Closure[] $rules See "*-Latin" rules from Transliterator::listIDs()
  67.      */
  68.     public function ascii(array $rules = []): self
  69.     {
  70.         $str = clone $this;
  71.         $s $str->string;
  72.         $str->string '';
  73.         array_unshift($rules'nfd');
  74.         $rules[] = 'latin-ascii';
  75.         if (\function_exists('transliterator_transliterate')) {
  76.             $rules[] = 'any-latin/bgn';
  77.         }
  78.         $rules[] = 'nfkd';
  79.         $rules[] = '[:nonspacing mark:] remove';
  80.         while (\strlen($s) - $i strspn($sself::ASCII)) {
  81.             if (< --$i) {
  82.                 $str->string .= substr($s0$i);
  83.                 $s substr($s$i);
  84.             }
  85.             if (!$rule array_shift($rules)) {
  86.                 $rules = []; // An empty rule interrupts the next ones
  87.             }
  88.             if ($rule instanceof \Transliterator) {
  89.                 $s $rule->transliterate($s);
  90.             } elseif ($rule instanceof \Closure) {
  91.                 $s $rule($s);
  92.             } elseif ($rule) {
  93.                 if ('nfd' === $rule strtolower($rule)) {
  94.                     normalizer_is_normalized($sself::NFD) ?: $s normalizer_normalize($sself::NFD);
  95.                 } elseif ('nfkd' === $rule) {
  96.                     normalizer_is_normalized($sself::NFKD) ?: $s normalizer_normalize($sself::NFKD);
  97.                 } elseif ('[:nonspacing mark:] remove' === $rule) {
  98.                     $s preg_replace('/\p{Mn}++/u'''$s);
  99.                 } elseif ('latin-ascii' === $rule) {
  100.                     $s str_replace(self::TRANSLIT_FROMself::TRANSLIT_TO$s);
  101.                 } elseif ('de-ascii' === $rule) {
  102.                     $s preg_replace("/([AUO])\u{0308}(?=\p{Ll})/u"'$1e'$s);
  103.                     $s str_replace(["a\u{0308}""o\u{0308}""u\u{0308}""A\u{0308}""O\u{0308}""U\u{0308}"], ['ae''oe''ue''AE''OE''UE'], $s);
  104.                 } elseif (\function_exists('transliterator_transliterate')) {
  105.                     if (null === $transliterator self::$transliterators[$rule] ??= \Transliterator::create($rule)) {
  106.                         if ('any-latin/bgn' === $rule) {
  107.                             $rule 'any-latin';
  108.                             $transliterator self::$transliterators[$rule] ??= \Transliterator::create($rule);
  109.                         }
  110.                         if (null === $transliterator) {
  111.                             throw new InvalidArgumentException(sprintf('Unknown transliteration rule "%s".'$rule));
  112.                         }
  113.                         self::$transliterators['any-latin/bgn'] = $transliterator;
  114.                     }
  115.                     $s $transliterator->transliterate($s);
  116.                 }
  117.             } elseif (!\function_exists('iconv')) {
  118.                 $s preg_replace('/[^\x00-\x7F]/u''?'$s);
  119.             } else {
  120.                 $s = @preg_replace_callback('/[^\x00-\x7F]/u', static function ($c) {
  121.                     $c = (string) iconv('UTF-8''ASCII//TRANSLIT'$c[0]);
  122.                     if ('' === $c && '' === iconv('UTF-8''ASCII//TRANSLIT''²')) {
  123.                         throw new \LogicException(sprintf('"%s" requires a translit-able iconv implementation, try installing "gnu-libiconv" if you\'re using Alpine Linux.', static::class));
  124.                     }
  125.                     return \strlen($c) ? ltrim($c'\'`"^~') : ('' !== $c $c '?');
  126.                 }, $s);
  127.             }
  128.         }
  129.         $str->string .= $s;
  130.         return $str;
  131.     }
  132.     public function camel(): static
  133.     {
  134.         $str = clone $this;
  135.         $str->string str_replace(' '''preg_replace_callback('/\b.(?![A-Z]{2,})/u', static function ($m) use (&$i) {
  136.             return === ++$i ? ('İ' === $m[0] ? 'i̇' mb_strtolower($m[0], 'UTF-8')) : mb_convert_case($m[0], \MB_CASE_TITLE'UTF-8');
  137.         }, preg_replace('/[^\pL0-9]++/u'' '$this->string)));
  138.         return $str;
  139.     }
  140.     /**
  141.      * @return int[]
  142.      */
  143.     public function codePointsAt(int $offset): array
  144.     {
  145.         $str $this->slice($offset1);
  146.         if ('' === $str->string) {
  147.             return [];
  148.         }
  149.         $codePoints = [];
  150.         foreach (preg_split('//u'$str->string, -1\PREG_SPLIT_NO_EMPTY) as $c) {
  151.             $codePoints[] = mb_ord($c'UTF-8');
  152.         }
  153.         return $codePoints;
  154.     }
  155.     public function folded(bool $compat true): static
  156.     {
  157.         $str = clone $this;
  158.         if (!$compat || !\defined('Normalizer::NFKC_CF')) {
  159.             $str->string normalizer_normalize($str->string$compat \Normalizer::NFKC \Normalizer::NFC);
  160.             $str->string mb_strtolower(str_replace(self::FOLD_FROMself::FOLD_TO$this->string), 'UTF-8');
  161.         } else {
  162.             $str->string normalizer_normalize($str->string\Normalizer::NFKC_CF);
  163.         }
  164.         return $str;
  165.     }
  166.     public function join(array $stringsstring $lastGlue null): static
  167.     {
  168.         $str = clone $this;
  169.         $tail null !== $lastGlue && \count($strings) ? $lastGlue.array_pop($strings) : '';
  170.         $str->string implode($this->string$strings).$tail;
  171.         if (!preg_match('//u'$str->string)) {
  172.             throw new InvalidArgumentException('Invalid UTF-8 string.');
  173.         }
  174.         return $str;
  175.     }
  176.     public function lower(): static
  177.     {
  178.         $str = clone $this;
  179.         $str->string mb_strtolower(str_replace('İ''i̇'$str->string), 'UTF-8');
  180.         return $str;
  181.     }
  182.     public function match(string $regexpint $flags 0int $offset 0): array
  183.     {
  184.         $match = ((\PREG_PATTERN_ORDER \PREG_SET_ORDER) & $flags) ? 'preg_match_all' 'preg_match';
  185.         if ($this->ignoreCase) {
  186.             $regexp .= 'i';
  187.         }
  188.         set_error_handler(static function ($t$m) { throw new InvalidArgumentException($m); });
  189.         try {
  190.             if (false === $match($regexp.'u'$this->string$matches$flags \PREG_UNMATCHED_AS_NULL$offset)) {
  191.                 throw new RuntimeException('Matching failed with error: '.preg_last_error_msg());
  192.             }
  193.         } finally {
  194.             restore_error_handler();
  195.         }
  196.         return $matches;
  197.     }
  198.     public function normalize(int $form self::NFC): static
  199.     {
  200.         if (!\in_array($form, [self::NFCself::NFDself::NFKCself::NFKD])) {
  201.             throw new InvalidArgumentException('Unsupported normalization form.');
  202.         }
  203.         $str = clone $this;
  204.         normalizer_is_normalized($str->string$form) ?: $str->string normalizer_normalize($str->string$form);
  205.         return $str;
  206.     }
  207.     public function padBoth(int $lengthstring $padStr ' '): static
  208.     {
  209.         if ('' === $padStr || !preg_match('//u'$padStr)) {
  210.             throw new InvalidArgumentException('Invalid UTF-8 string.');
  211.         }
  212.         $pad = clone $this;
  213.         $pad->string $padStr;
  214.         return $this->pad($length$pad\STR_PAD_BOTH);
  215.     }
  216.     public function padEnd(int $lengthstring $padStr ' '): static
  217.     {
  218.         if ('' === $padStr || !preg_match('//u'$padStr)) {
  219.             throw new InvalidArgumentException('Invalid UTF-8 string.');
  220.         }
  221.         $pad = clone $this;
  222.         $pad->string $padStr;
  223.         return $this->pad($length$pad\STR_PAD_RIGHT);
  224.     }
  225.     public function padStart(int $lengthstring $padStr ' '): static
  226.     {
  227.         if ('' === $padStr || !preg_match('//u'$padStr)) {
  228.             throw new InvalidArgumentException('Invalid UTF-8 string.');
  229.         }
  230.         $pad = clone $this;
  231.         $pad->string $padStr;
  232.         return $this->pad($length$pad\STR_PAD_LEFT);
  233.     }
  234.     public function replaceMatches(string $fromRegexpstring|callable $to): static
  235.     {
  236.         if ($this->ignoreCase) {
  237.             $fromRegexp .= 'i';
  238.         }
  239.         if (\is_array($to) || $to instanceof \Closure) {
  240.             $replace 'preg_replace_callback';
  241.             $to = static function (array $m) use ($to): string {
  242.                 $to $to($m);
  243.                 if ('' !== $to && (!\is_string($to) || !preg_match('//u'$to))) {
  244.                     throw new InvalidArgumentException('Replace callback must return a valid UTF-8 string.');
  245.                 }
  246.                 return $to;
  247.             };
  248.         } elseif ('' !== $to && !preg_match('//u'$to)) {
  249.             throw new InvalidArgumentException('Invalid UTF-8 string.');
  250.         } else {
  251.             $replace 'preg_replace';
  252.         }
  253.         set_error_handler(static function ($t$m) { throw new InvalidArgumentException($m); });
  254.         try {
  255.             if (null === $string $replace($fromRegexp.'u'$to$this->string)) {
  256.                 $lastError preg_last_error();
  257.                 foreach (get_defined_constants(true)['pcre'] as $k => $v) {
  258.                     if ($lastError === $v && str_ends_with($k'_ERROR')) {
  259.                         throw new RuntimeException('Matching failed with '.$k.'.');
  260.                     }
  261.                 }
  262.                 throw new RuntimeException('Matching failed with unknown error code.');
  263.             }
  264.         } finally {
  265.             restore_error_handler();
  266.         }
  267.         $str = clone $this;
  268.         $str->string $string;
  269.         return $str;
  270.     }
  271.     public function reverse(): static
  272.     {
  273.         $str = clone $this;
  274.         $str->string implode(''array_reverse(preg_split('/(\X)/u'$str->string, -1\PREG_SPLIT_DELIM_CAPTURE \PREG_SPLIT_NO_EMPTY)));
  275.         return $str;
  276.     }
  277.     public function snake(): static
  278.     {
  279.         $str $this->camel();
  280.         $str->string mb_strtolower(preg_replace(['/(\p{Lu}+)(\p{Lu}\p{Ll})/u''/([\p{Ll}0-9])(\p{Lu})/u'], '\1_\2'$str->string), 'UTF-8');
  281.         return $str;
  282.     }
  283.     public function title(bool $allWords false): static
  284.     {
  285.         $str = clone $this;
  286.         $limit $allWords ? -1;
  287.         $str->string preg_replace_callback('/\b./u', static function (array $m): string {
  288.             return mb_convert_case($m[0], \MB_CASE_TITLE'UTF-8');
  289.         }, $str->string$limit);
  290.         return $str;
  291.     }
  292.     public function trim(string $chars " \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}"): static
  293.     {
  294.         if (" \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}" !== $chars && !preg_match('//u'$chars)) {
  295.             throw new InvalidArgumentException('Invalid UTF-8 chars.');
  296.         }
  297.         $chars preg_quote($chars);
  298.         $str = clone $this;
  299.         $str->string preg_replace("{^[$chars]++|[$chars]++$}uD"''$str->string);
  300.         return $str;
  301.     }
  302.     public function trimEnd(string $chars " \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}"): static
  303.     {
  304.         if (" \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}" !== $chars && !preg_match('//u'$chars)) {
  305.             throw new InvalidArgumentException('Invalid UTF-8 chars.');
  306.         }
  307.         $chars preg_quote($chars);
  308.         $str = clone $this;
  309.         $str->string preg_replace("{[$chars]++$}uD"''$str->string);
  310.         return $str;
  311.     }
  312.     public function trimPrefix($prefix): static
  313.     {
  314.         if (!$this->ignoreCase) {
  315.             return parent::trimPrefix($prefix);
  316.         }
  317.         $str = clone $this;
  318.         if ($prefix instanceof \Traversable) {
  319.             $prefix iterator_to_array($prefixfalse);
  320.         } elseif ($prefix instanceof parent) {
  321.             $prefix $prefix->string;
  322.         }
  323.         $prefix implode('|'array_map('preg_quote', (array) $prefix));
  324.         $str->string preg_replace("{^(?:$prefix)}iuD"''$this->string);
  325.         return $str;
  326.     }
  327.     public function trimStart(string $chars " \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}"): static
  328.     {
  329.         if (" \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}" !== $chars && !preg_match('//u'$chars)) {
  330.             throw new InvalidArgumentException('Invalid UTF-8 chars.');
  331.         }
  332.         $chars preg_quote($chars);
  333.         $str = clone $this;
  334.         $str->string preg_replace("{^[$chars]++}uD"''$str->string);
  335.         return $str;
  336.     }
  337.     public function trimSuffix($suffix): static
  338.     {
  339.         if (!$this->ignoreCase) {
  340.             return parent::trimSuffix($suffix);
  341.         }
  342.         $str = clone $this;
  343.         if ($suffix instanceof \Traversable) {
  344.             $suffix iterator_to_array($suffixfalse);
  345.         } elseif ($suffix instanceof parent) {
  346.             $suffix $suffix->string;
  347.         }
  348.         $suffix implode('|'array_map('preg_quote', (array) $suffix));
  349.         $str->string preg_replace("{(?:$suffix)$}iuD"''$this->string);
  350.         return $str;
  351.     }
  352.     public function upper(): static
  353.     {
  354.         $str = clone $this;
  355.         $str->string mb_strtoupper($str->string'UTF-8');
  356.         return $str;
  357.     }
  358.     public function width(bool $ignoreAnsiDecoration true): int
  359.     {
  360.         $width 0;
  361.         $s str_replace(["\x00""\x05""\x07"], ''$this->string);
  362.         if (str_contains($s"\r")) {
  363.             $s str_replace(["\r\n""\r"], "\n"$s);
  364.         }
  365.         if (!$ignoreAnsiDecoration) {
  366.             $s preg_replace('/[\p{Cc}\x7F]++/u'''$s);
  367.         }
  368.         foreach (explode("\n"$s) as $s) {
  369.             if ($ignoreAnsiDecoration) {
  370.                 $s preg_replace('/(?:\x1B(?:
  371.                     \[ [\x30-\x3F]*+ [\x20-\x2F]*+ [\x40-\x7E]
  372.                     | [P\]X^_] .*? \x1B\\\\
  373.                     | [\x41-\x7E]
  374.                 )|[\p{Cc}\x7F]++)/xu'''$s);
  375.             }
  376.             $lineWidth $this->wcswidth($s);
  377.             if ($lineWidth $width) {
  378.                 $width $lineWidth;
  379.             }
  380.         }
  381.         return $width;
  382.     }
  383.     private function pad(int $lenself $padint $type): static
  384.     {
  385.         $sLen $this->length();
  386.         if ($len <= $sLen) {
  387.             return clone $this;
  388.         }
  389.         $padLen $pad->length();
  390.         $freeLen $len $sLen;
  391.         $len $freeLen $padLen;
  392.         switch ($type) {
  393.             case \STR_PAD_RIGHT:
  394.                 return $this->append(str_repeat($pad->stringintdiv($freeLen$padLen)).($len $pad->slice(0$len) : ''));
  395.             case \STR_PAD_LEFT:
  396.                 return $this->prepend(str_repeat($pad->stringintdiv($freeLen$padLen)).($len $pad->slice(0$len) : ''));
  397.             case \STR_PAD_BOTH:
  398.                 $freeLen /= 2;
  399.                 $rightLen ceil($freeLen);
  400.                 $len $rightLen $padLen;
  401.                 $str $this->append(str_repeat($pad->stringintdiv($rightLen$padLen)).($len $pad->slice(0$len) : ''));
  402.                 $leftLen floor($freeLen);
  403.                 $len $leftLen $padLen;
  404.                 return $str->prepend(str_repeat($pad->stringintdiv($leftLen$padLen)).($len $pad->slice(0$len) : ''));
  405.             default:
  406.                 throw new InvalidArgumentException('Invalid padding type.');
  407.         }
  408.     }
  409.     /**
  410.      * Based on https://github.com/jquast/wcwidth, a Python implementation of https://www.cl.cam.ac.uk/~mgk25/ucs/wcwidth.c.
  411.      */
  412.     private function wcswidth(string $string): int
  413.     {
  414.         $width 0;
  415.         foreach (preg_split('//u'$string, -1\PREG_SPLIT_NO_EMPTY) as $c) {
  416.             $codePoint mb_ord($c'UTF-8');
  417.             if (=== $codePoint // NULL
  418.                 || 0x034F === $codePoint // COMBINING GRAPHEME JOINER
  419.                 || (0x200B <= $codePoint && 0x200F >= $codePoint// ZERO WIDTH SPACE to RIGHT-TO-LEFT MARK
  420.                 || 0x2028 === $codePoint // LINE SEPARATOR
  421.                 || 0x2029 === $codePoint // PARAGRAPH SEPARATOR
  422.                 || (0x202A <= $codePoint && 0x202E >= $codePoint// LEFT-TO-RIGHT EMBEDDING to RIGHT-TO-LEFT OVERRIDE
  423.                 || (0x2060 <= $codePoint && 0x2063 >= $codePoint// WORD JOINER to INVISIBLE SEPARATOR
  424.             ) {
  425.                 continue;
  426.             }
  427.             // Non printable characters
  428.             if (32 $codePoint // C0 control characters
  429.                 || (0x07F <= $codePoint && 0x0A0 $codePoint// C1 control characters and DEL
  430.             ) {
  431.                 return -1;
  432.             }
  433.             self::$tableZero ??= require __DIR__.'/Resources/data/wcswidth_table_zero.php';
  434.             if ($codePoint >= self::$tableZero[0][0] && $codePoint <= self::$tableZero[$ubound \count(self::$tableZero) - 1][1]) {
  435.                 $lbound 0;
  436.                 while ($ubound >= $lbound) {
  437.                     $mid floor(($lbound $ubound) / 2);
  438.                     if ($codePoint self::$tableZero[$mid][1]) {
  439.                         $lbound $mid 1;
  440.                     } elseif ($codePoint self::$tableZero[$mid][0]) {
  441.                         $ubound $mid 1;
  442.                     } else {
  443.                         continue 2;
  444.                     }
  445.                 }
  446.             }
  447.             self::$tableWide ??= require __DIR__.'/Resources/data/wcswidth_table_wide.php';
  448.             if ($codePoint >= self::$tableWide[0][0] && $codePoint <= self::$tableWide[$ubound \count(self::$tableWide) - 1][1]) {
  449.                 $lbound 0;
  450.                 while ($ubound >= $lbound) {
  451.                     $mid floor(($lbound $ubound) / 2);
  452.                     if ($codePoint self::$tableWide[$mid][1]) {
  453.                         $lbound $mid 1;
  454.                     } elseif ($codePoint self::$tableWide[$mid][0]) {
  455.                         $ubound $mid 1;
  456.                     } else {
  457.                         $width += 2;
  458.                         continue 2;
  459.                     }
  460.                 }
  461.             }
  462.             ++$width;
  463.         }
  464.         return $width;
  465.     }
  466. }