vendor/symfony/string/AbstractString.php line 386

  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 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 doesn't care about the exact variant it deals with.
  20.  *
  21.  * @author Nicolas Grekas <p@tchwork.com>
  22.  * @author Hugo Hamon <hugohamon@neuf.fr>
  23.  *
  24.  * @throws ExceptionInterface
  25.  */
  26. abstract class AbstractString implements \Stringable\JsonSerializable
  27. {
  28.     public const PREG_PATTERN_ORDER \PREG_PATTERN_ORDER;
  29.     public const PREG_SET_ORDER \PREG_SET_ORDER;
  30.     public const PREG_OFFSET_CAPTURE \PREG_OFFSET_CAPTURE;
  31.     public const PREG_UNMATCHED_AS_NULL \PREG_UNMATCHED_AS_NULL;
  32.     public const PREG_SPLIT 0;
  33.     public const PREG_SPLIT_NO_EMPTY \PREG_SPLIT_NO_EMPTY;
  34.     public const PREG_SPLIT_DELIM_CAPTURE \PREG_SPLIT_DELIM_CAPTURE;
  35.     public const PREG_SPLIT_OFFSET_CAPTURE \PREG_SPLIT_OFFSET_CAPTURE;
  36.     protected $string '';
  37.     protected $ignoreCase false;
  38.     abstract public function __construct(string $string '');
  39.     /**
  40.      * Unwraps instances of AbstractString back to strings.
  41.      *
  42.      * @return string[]|array
  43.      */
  44.     public static function unwrap(array $values): array
  45.     {
  46.         foreach ($values as $k => $v) {
  47.             if ($v instanceof self) {
  48.                 $values[$k] = $v->__toString();
  49.             } elseif (\is_array($v) && $values[$k] !== $v = static::unwrap($v)) {
  50.                 $values[$k] = $v;
  51.             }
  52.         }
  53.         return $values;
  54.     }
  55.     /**
  56.      * Wraps (and normalizes) strings in instances of AbstractString.
  57.      *
  58.      * @return static[]|array
  59.      */
  60.     public static function wrap(array $values): array
  61.     {
  62.         $i 0;
  63.         $keys null;
  64.         foreach ($values as $k => $v) {
  65.             if (\is_string($k) && '' !== $k && $k !== $j = (string) new static($k)) {
  66.                 $keys ??= array_keys($values);
  67.                 $keys[$i] = $j;
  68.             }
  69.             if (\is_string($v)) {
  70.                 $values[$k] = new static($v);
  71.             } elseif (\is_array($v) && $values[$k] !== $v = static::wrap($v)) {
  72.                 $values[$k] = $v;
  73.             }
  74.             ++$i;
  75.         }
  76.         return null !== $keys array_combine($keys$values) : $values;
  77.     }
  78.     /**
  79.      * @param string|string[] $needle
  80.      */
  81.     public function after(string|iterable $needlebool $includeNeedle falseint $offset 0): static
  82.     {
  83.         $str = clone $this;
  84.         $i \PHP_INT_MAX;
  85.         if (\is_string($needle)) {
  86.             $needle = [$needle];
  87.         }
  88.         foreach ($needle as $n) {
  89.             $n = (string) $n;
  90.             $j $this->indexOf($n$offset);
  91.             if (null !== $j && $j $i) {
  92.                 $i $j;
  93.                 $str->string $n;
  94.             }
  95.         }
  96.         if (\PHP_INT_MAX === $i) {
  97.             return $str;
  98.         }
  99.         if (!$includeNeedle) {
  100.             $i += $str->length();
  101.         }
  102.         return $this->slice($i);
  103.     }
  104.     /**
  105.      * @param string|string[] $needle
  106.      */
  107.     public function afterLast(string|iterable $needlebool $includeNeedle falseint $offset 0): static
  108.     {
  109.         $str = clone $this;
  110.         $i null;
  111.         if (\is_string($needle)) {
  112.             $needle = [$needle];
  113.         }
  114.         foreach ($needle as $n) {
  115.             $n = (string) $n;
  116.             $j $this->indexOfLast($n$offset);
  117.             if (null !== $j && $j >= $i) {
  118.                 $i $offset $j;
  119.                 $str->string $n;
  120.             }
  121.         }
  122.         if (null === $i) {
  123.             return $str;
  124.         }
  125.         if (!$includeNeedle) {
  126.             $i += $str->length();
  127.         }
  128.         return $this->slice($i);
  129.     }
  130.     abstract public function append(string ...$suffix): static;
  131.     /**
  132.      * @param string|string[] $needle
  133.      */
  134.     public function before(string|iterable $needlebool $includeNeedle falseint $offset 0): static
  135.     {
  136.         $str = clone $this;
  137.         $i \PHP_INT_MAX;
  138.         if (\is_string($needle)) {
  139.             $needle = [$needle];
  140.         }
  141.         foreach ($needle as $n) {
  142.             $n = (string) $n;
  143.             $j $this->indexOf($n$offset);
  144.             if (null !== $j && $j $i) {
  145.                 $i $j;
  146.                 $str->string $n;
  147.             }
  148.         }
  149.         if (\PHP_INT_MAX === $i) {
  150.             return $str;
  151.         }
  152.         if ($includeNeedle) {
  153.             $i += $str->length();
  154.         }
  155.         return $this->slice(0$i);
  156.     }
  157.     /**
  158.      * @param string|string[] $needle
  159.      */
  160.     public function beforeLast(string|iterable $needlebool $includeNeedle falseint $offset 0): static
  161.     {
  162.         $str = clone $this;
  163.         $i null;
  164.         if (\is_string($needle)) {
  165.             $needle = [$needle];
  166.         }
  167.         foreach ($needle as $n) {
  168.             $n = (string) $n;
  169.             $j $this->indexOfLast($n$offset);
  170.             if (null !== $j && $j >= $i) {
  171.                 $i $offset $j;
  172.                 $str->string $n;
  173.             }
  174.         }
  175.         if (null === $i) {
  176.             return $str;
  177.         }
  178.         if ($includeNeedle) {
  179.             $i += $str->length();
  180.         }
  181.         return $this->slice(0$i);
  182.     }
  183.     /**
  184.      * @return int[]
  185.      */
  186.     public function bytesAt(int $offset): array
  187.     {
  188.         $str $this->slice($offset1);
  189.         return '' === $str->string ? [] : array_values(unpack('C*'$str->string));
  190.     }
  191.     abstract public function camel(): static;
  192.     /**
  193.      * @return static[]
  194.      */
  195.     abstract public function chunk(int $length 1): array;
  196.     public function collapseWhitespace(): static
  197.     {
  198.         $str = clone $this;
  199.         $str->string trim(preg_replace("/(?:[ \n\r\t\x0C]{2,}+|[\n\r\t\x0C])/"' '$str->string), " \n\r\t\x0C");
  200.         return $str;
  201.     }
  202.     /**
  203.      * @param string|string[] $needle
  204.      */
  205.     public function containsAny(string|iterable $needle): bool
  206.     {
  207.         return null !== $this->indexOf($needle);
  208.     }
  209.     /**
  210.      * @param string|string[] $suffix
  211.      */
  212.     public function endsWith(string|iterable $suffix): bool
  213.     {
  214.         if (\is_string($suffix)) {
  215.             throw new \TypeError(sprintf('Method "%s()" must be overridden by class "%s" to deal with non-iterable values.'__FUNCTION__, static::class));
  216.         }
  217.         foreach ($suffix as $s) {
  218.             if ($this->endsWith((string) $s)) {
  219.                 return true;
  220.             }
  221.         }
  222.         return false;
  223.     }
  224.     public function ensureEnd(string $suffix): static
  225.     {
  226.         if (!$this->endsWith($suffix)) {
  227.             return $this->append($suffix);
  228.         }
  229.         $suffix preg_quote($suffix);
  230.         $regex '{('.$suffix.')(?:'.$suffix.')++$}D';
  231.         return $this->replaceMatches($regex.($this->ignoreCase 'i' ''), '$1');
  232.     }
  233.     public function ensureStart(string $prefix): static
  234.     {
  235.         $prefix = new static($prefix);
  236.         if (!$this->startsWith($prefix)) {
  237.             return $this->prepend($prefix);
  238.         }
  239.         $str = clone $this;
  240.         $i $prefixLen $prefix->length();
  241.         while ($this->indexOf($prefix$i) === $i) {
  242.             $str $str->slice($prefixLen);
  243.             $i += $prefixLen;
  244.         }
  245.         return $str;
  246.     }
  247.     /**
  248.      * @param string|string[] $string
  249.      */
  250.     public function equalsTo(string|iterable $string): bool
  251.     {
  252.         if (\is_string($string)) {
  253.             throw new \TypeError(sprintf('Method "%s()" must be overridden by class "%s" to deal with non-iterable values.'__FUNCTION__, static::class));
  254.         }
  255.         foreach ($string as $s) {
  256.             if ($this->equalsTo((string) $s)) {
  257.                 return true;
  258.             }
  259.         }
  260.         return false;
  261.     }
  262.     abstract public function folded(): static;
  263.     public function ignoreCase(): static
  264.     {
  265.         $str = clone $this;
  266.         $str->ignoreCase true;
  267.         return $str;
  268.     }
  269.     /**
  270.      * @param string|string[] $needle
  271.      */
  272.     public function indexOf(string|iterable $needleint $offset 0): ?int
  273.     {
  274.         if (\is_string($needle)) {
  275.             throw new \TypeError(sprintf('Method "%s()" must be overridden by class "%s" to deal with non-iterable values.'__FUNCTION__, static::class));
  276.         }
  277.         $i \PHP_INT_MAX;
  278.         foreach ($needle as $n) {
  279.             $j $this->indexOf((string) $n$offset);
  280.             if (null !== $j && $j $i) {
  281.                 $i $j;
  282.             }
  283.         }
  284.         return \PHP_INT_MAX === $i null $i;
  285.     }
  286.     /**
  287.      * @param string|string[] $needle
  288.      */
  289.     public function indexOfLast(string|iterable $needleint $offset 0): ?int
  290.     {
  291.         if (\is_string($needle)) {
  292.             throw new \TypeError(sprintf('Method "%s()" must be overridden by class "%s" to deal with non-iterable values.'__FUNCTION__, static::class));
  293.         }
  294.         $i null;
  295.         foreach ($needle as $n) {
  296.             $j $this->indexOfLast((string) $n$offset);
  297.             if (null !== $j && $j >= $i) {
  298.                 $i $offset $j;
  299.             }
  300.         }
  301.         return $i;
  302.     }
  303.     public function isEmpty(): bool
  304.     {
  305.         return '' === $this->string;
  306.     }
  307.     abstract public function join(array $stringsstring $lastGlue null): static;
  308.     public function jsonSerialize(): string
  309.     {
  310.         return $this->string;
  311.     }
  312.     abstract public function length(): int;
  313.     abstract public function lower(): static;
  314.     /**
  315.      * Matches the string using a regular expression.
  316.      *
  317.      * Pass PREG_PATTERN_ORDER or PREG_SET_ORDER as $flags to get all occurrences matching the regular expression.
  318.      *
  319.      * @return array All matches in a multi-dimensional array ordered according to flags
  320.      */
  321.     abstract public function match(string $regexpint $flags 0int $offset 0): array;
  322.     abstract public function padBoth(int $lengthstring $padStr ' '): static;
  323.     abstract public function padEnd(int $lengthstring $padStr ' '): static;
  324.     abstract public function padStart(int $lengthstring $padStr ' '): static;
  325.     abstract public function prepend(string ...$prefix): static;
  326.     public function repeat(int $multiplier): static
  327.     {
  328.         if ($multiplier) {
  329.             throw new InvalidArgumentException(sprintf('Multiplier must be positive, %d given.'$multiplier));
  330.         }
  331.         $str = clone $this;
  332.         $str->string str_repeat($str->string$multiplier);
  333.         return $str;
  334.     }
  335.     abstract public function replace(string $fromstring $to): static;
  336.     abstract public function replaceMatches(string $fromRegexpstring|callable $to): static;
  337.     abstract public function reverse(): static;
  338.     abstract public function slice(int $start 0int $length null): static;
  339.     abstract public function snake(): static;
  340.     abstract public function splice(string $replacementint $start 0int $length null): static;
  341.     /**
  342.      * @return static[]
  343.      */
  344.     public function split(string $delimiterint $limit nullint $flags null): array
  345.     {
  346.         if (null === $flags) {
  347.             throw new \TypeError('Split behavior when $flags is null must be implemented by child classes.');
  348.         }
  349.         if ($this->ignoreCase) {
  350.             $delimiter .= 'i';
  351.         }
  352.         set_error_handler(static function ($t$m) { throw new InvalidArgumentException($m); });
  353.         try {
  354.             if (false === $chunks preg_split($delimiter$this->string$limit$flags)) {
  355.                 throw new RuntimeException('Splitting failed with error: '.preg_last_error_msg());
  356.             }
  357.         } finally {
  358.             restore_error_handler();
  359.         }
  360.         $str = clone $this;
  361.         if (self::PREG_SPLIT_OFFSET_CAPTURE $flags) {
  362.             foreach ($chunks as &$chunk) {
  363.                 $str->string $chunk[0];
  364.                 $chunk[0] = clone $str;
  365.             }
  366.         } else {
  367.             foreach ($chunks as &$chunk) {
  368.                 $str->string $chunk;
  369.                 $chunk = clone $str;
  370.             }
  371.         }
  372.         return $chunks;
  373.     }
  374.     /**
  375.      * @param string|string[] $prefix
  376.      */
  377.     public function startsWith(string|iterable $prefix): bool
  378.     {
  379.         if (\is_string($prefix)) {
  380.             throw new \TypeError(sprintf('Method "%s()" must be overridden by class "%s" to deal with non-iterable values.'__FUNCTION__, static::class));
  381.         }
  382.         foreach ($prefix as $prefix) {
  383.             if ($this->startsWith((string) $prefix)) {
  384.                 return true;
  385.             }
  386.         }
  387.         return false;
  388.     }
  389.     abstract public function title(bool $allWords false): static;
  390.     public function toByteString(string $toEncoding null): ByteString
  391.     {
  392.         $b = new ByteString();
  393.         $toEncoding \in_array($toEncoding, ['utf8''utf-8''UTF8'], true) ? 'UTF-8' $toEncoding;
  394.         if (null === $toEncoding || $toEncoding === $fromEncoding $this instanceof AbstractUnicodeString || preg_match('//u'$b->string) ? 'UTF-8' 'Windows-1252') {
  395.             $b->string $this->string;
  396.             return $b;
  397.         }
  398.         set_error_handler(static function ($t$m) { throw new InvalidArgumentException($m); });
  399.         try {
  400.             try {
  401.                 $b->string mb_convert_encoding($this->string$toEncoding'UTF-8');
  402.             } catch (InvalidArgumentException $e) {
  403.                 if (!\function_exists('iconv')) {
  404.                     throw $e;
  405.                 }
  406.                 $b->string iconv('UTF-8'$toEncoding$this->string);
  407.             }
  408.         } finally {
  409.             restore_error_handler();
  410.         }
  411.         return $b;
  412.     }
  413.     public function toCodePointString(): CodePointString
  414.     {
  415.         return new CodePointString($this->string);
  416.     }
  417.     public function toString(): string
  418.     {
  419.         return $this->string;
  420.     }
  421.     public function toUnicodeString(): UnicodeString
  422.     {
  423.         return new UnicodeString($this->string);
  424.     }
  425.     abstract public function trim(string $chars " \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}"): static;
  426.     abstract public function trimEnd(string $chars " \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}"): static;
  427.     /**
  428.      * @param string|string[] $prefix
  429.      */
  430.     public function trimPrefix($prefix): static
  431.     {
  432.         if (\is_array($prefix) || $prefix instanceof \Traversable) { // don't use is_iterable(), it's slow
  433.             foreach ($prefix as $s) {
  434.                 $t $this->trimPrefix($s);
  435.                 if ($t->string !== $this->string) {
  436.                     return $t;
  437.                 }
  438.             }
  439.             return clone $this;
  440.         }
  441.         $str = clone $this;
  442.         if ($prefix instanceof self) {
  443.             $prefix $prefix->string;
  444.         } else {
  445.             $prefix = (string) $prefix;
  446.         }
  447.         if ('' !== $prefix && \strlen($this->string) >= \strlen($prefix) && === substr_compare($this->string$prefix0\strlen($prefix), $this->ignoreCase)) {
  448.             $str->string substr($this->string\strlen($prefix));
  449.         }
  450.         return $str;
  451.     }
  452.     abstract public function trimStart(string $chars " \t\n\r\0\x0B\x0C\u{A0}\u{FEFF}"): static;
  453.     /**
  454.      * @param string|string[] $suffix
  455.      */
  456.     public function trimSuffix($suffix): static
  457.     {
  458.         if (\is_array($suffix) || $suffix instanceof \Traversable) { // don't use is_iterable(), it's slow
  459.             foreach ($suffix as $s) {
  460.                 $t $this->trimSuffix($s);
  461.                 if ($t->string !== $this->string) {
  462.                     return $t;
  463.                 }
  464.             }
  465.             return clone $this;
  466.         }
  467.         $str = clone $this;
  468.         if ($suffix instanceof self) {
  469.             $suffix $suffix->string;
  470.         } else {
  471.             $suffix = (string) $suffix;
  472.         }
  473.         if ('' !== $suffix && \strlen($this->string) >= \strlen($suffix) && === substr_compare($this->string$suffix, -\strlen($suffix), null$this->ignoreCase)) {
  474.             $str->string substr($this->string0, -\strlen($suffix));
  475.         }
  476.         return $str;
  477.     }
  478.     public function truncate(int $lengthstring $ellipsis ''bool $cut true): static
  479.     {
  480.         $stringLength $this->length();
  481.         if ($stringLength <= $length) {
  482.             return clone $this;
  483.         }
  484.         $ellipsisLength '' !== $ellipsis ? (new static($ellipsis))->length() : 0;
  485.         if ($length $ellipsisLength) {
  486.             $ellipsisLength 0;
  487.         }
  488.         if (!$cut) {
  489.             if (null === $length $this->indexOf([' '"\r""\n""\t"], ($length ?: 1) - 1)) {
  490.                 return clone $this;
  491.             }
  492.             $length += $ellipsisLength;
  493.         }
  494.         $str $this->slice(0$length $ellipsisLength);
  495.         return $ellipsisLength $str->trimEnd()->append($ellipsis) : $str;
  496.     }
  497.     abstract public function upper(): static;
  498.     /**
  499.      * Returns the printable length on a terminal.
  500.      */
  501.     abstract public function width(bool $ignoreAnsiDecoration true): int;
  502.     public function wordwrap(int $width 75string $break "\n"bool $cut false): static
  503.     {
  504.         $lines '' !== $break $this->split($break) : [clone $this];
  505.         $chars = [];
  506.         $mask '';
  507.         if (=== \count($lines) && '' === $lines[0]->string) {
  508.             return $lines[0];
  509.         }
  510.         foreach ($lines as $i => $line) {
  511.             if ($i) {
  512.                 $chars[] = $break;
  513.                 $mask .= '#';
  514.             }
  515.             foreach ($line->chunk() as $char) {
  516.                 $chars[] = $char->string;
  517.                 $mask .= ' ' === $char->string ' ' '?';
  518.             }
  519.         }
  520.         $string '';
  521.         $j 0;
  522.         $b $i = -1;
  523.         $mask wordwrap($mask$width'#'$cut);
  524.         while (false !== $b strpos($mask'#'$b 1)) {
  525.             for (++$i$i $b; ++$i) {
  526.                 $string .= $chars[$j];
  527.                 unset($chars[$j++]);
  528.             }
  529.             if ($break === $chars[$j] || ' ' === $chars[$j]) {
  530.                 unset($chars[$j++]);
  531.             }
  532.             $string .= $break;
  533.         }
  534.         $str = clone $this;
  535.         $str->string $string.implode(''$chars);
  536.         return $str;
  537.     }
  538.     public function __sleep(): array
  539.     {
  540.         return ['string'];
  541.     }
  542.     public function __clone()
  543.     {
  544.         $this->ignoreCase false;
  545.     }
  546.     public function __toString(): string
  547.     {
  548.         return $this->string;
  549.     }
  550. }