Passed
Pull Request — main (#104)
by Andrey
14:17
created

Str::snake()   A

Complexity

Conditions 3
Paths 3

Size

Total Lines 15
Code Lines 7

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 3

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 3
eloc 7
c 1
b 0
f 0
nc 3
nop 2
dl 0
loc 15
ccs 2
cts 2
cp 1
crap 3
rs 10
1
<?php
2
3
namespace Helldar\Support\Helpers;
4
5
use Helldar\Support\Facades\Helpers\Call as CallHelper;
6
use Helldar\Support\Facades\Tools\Replace;
7
use Illuminate\Contracts\Support\DeferringDisplayableValue;
0 ignored issues
show
Bug introduced by
The type Illuminate\Contracts\Sup...ferringDisplayableValue was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
8
use Illuminate\Contracts\Support\Htmlable;
0 ignored issues
show
Bug introduced by
The type Illuminate\Contracts\Support\Htmlable was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
9
use voku\helper\ASCII;
10
11
final class Str
12
{
13
    /**
14
     * The cache of snake-cased words.
15
     *
16
     * @var array
17
     */
18
    protected static $snakeCache = [];
19
20
    /**
21
     * The cache of camel-cased words.
22
     *
23
     * @var array
24
     */
25
    protected static $camelCache = [];
26
27
    /**
28
     * The cache of studly-cased words.
29
     *
30
     * @var array
31
     */
32
    protected static $studlyCache = [];
33
34
    protected $escaping_methods = [
35
        DeferringDisplayableValue::class => 'resolveDisplayableValue',
36
        Htmlable::class                  => 'toHtml',
37
    ];
38
39
    /**
40
     * Get a new stringable object from the given string.
41
     *
42
     * @see https://github.com/illuminate/support/blob/master/Str.php
43
     *
44
     * @param  string|null  $value
45
     *
46
     * @return \Helldar\Support\Helpers\Stringable
47 2
     */
48
    public function of(?string $value): Stringable
49 2
    {
50
        return new Stringable($value);
51
    }
52
53 2
    /**
54
     * Escape HTML special characters in a string.
55
     *
56
     * @param  string|null  $value
57
     * @param  bool  $double
58
     *
59
     * @return string|null
60
     */
61
    public function e(?string $value, bool $double = true): ?string
62
    {
63 2
        if ($escaped = CallHelper::runOf($this->escaping_methods, $value)) {
64
            return $escaped;
65 2
        }
66
67
        return htmlspecialchars($value, ENT_QUOTES, 'UTF-8', $double);
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of htmlspecialchars() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

67
        return htmlspecialchars(/** @scrutinizer ignore-type */ $value, ENT_QUOTES, 'UTF-8', $double);
Loading history...
68
    }
69
70
    /**
71
     * Convert special HTML entities back to characters.
72
     *
73
     * @param  string|null  $value
74
     *
75 2
     * @return string|null
76
     */
77 2
    public function de(?string $value): ?string
78
    {
79
        return htmlspecialchars_decode($value, ENT_QUOTES);
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of htmlspecialchars_decode() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

79
        return htmlspecialchars_decode(/** @scrutinizer ignore-type */ $value, ENT_QUOTES);
Loading history...
80
    }
81
82
    /**
83
     * Replacing multiple spaces with a single space.
84
     *
85
     * @param  string|null  $value
86
     *
87
     * @return string|null
88
     */
89 2
    public function removeSpaces(?string $value): ?string
90
    {
91 2
        return preg_replace('!\s+!', ' ', $value);
92 2
    }
93
94
    /**
95 2
     * Get a string according to an integer value.
96 2
     *
97 2
     * @param  float  $number
98 2
     * @param  array  $choice
99 2
     * @param  string|null  $extra
100
     *
101 2
     * @return string
102
     */
103
    public function choice(float $number, array $choice = [], string $extra = null): string
104
    {
105
        $number = (int) $number;
106 2
        $mod    = $number % 10;
107
108
        switch (true) {
109 2
            case $mod === 0:
110 2
            case $mod >= 5 && $mod <= 9:
111
            case ($number % 100 >= 11) && ($number % 100 <= 20):
112
                $result = $choice[2] ?? '';
113 2
                break;
114
115
            case $mod >= 2 && $mod <= 4:
116
                $result = $choice[1] ?? '';
117
                break;
118
119
            default:
120
                $result = $choice[0] ?? '';
121
        }
122
123
        if (empty($extra)) {
124
            return trim($result);
125
        }
126 2
127
        return implode(' ', [trim($result), trim($extra)]);
128 2
    }
129
130 2
    /**
131
     * Begin a string with a single instance of a given value.
132
     *
133
     * @see https://github.com/illuminate/support/blob/master/Str.php
134
     *
135
     * @param  string|null  $value
136
     * @param  string  $prefix
137
     *
138
     * @return string
139
     */
140
    public function start(?string $value, string $prefix): string
141
    {
142
        $quoted = preg_quote($prefix, '/');
143 2
144
        return $prefix . preg_replace('/^(?:' . $quoted . ')+/u', '', $value);
145 2
    }
146
147 2
    /**
148
     * Cap a string with a single instance of a given value.
149
     *
150
     * @see https://github.com/illuminate/support/blob/master/Str.php
151
     *
152
     * @param  string  $value
153
     * @param  string  $cap
154
     *
155
     * @return string
156
     */
157
    public function finish(string $value, string $cap = '/'): string
158 94
    {
159
        $quoted = preg_quote($cap, '/');
160 94
161 94
        return preg_replace('/(?:' . $quoted . ')+$/u', '', $value) . $cap;
162 94
    }
163
164
    /**
165
     *  Determine if a given string starts with a given substring.
166 92
     *
167
     * @param  string  $haystack
168
     * @param  string|string[]  $needles
169
     *
170
     * @return bool
171
     */
172
    public function startsWith(string $haystack, $needles): bool
173
    {
174
        foreach ((array) $needles as $needle) {
175
            if ((string) $needle !== '' && str_starts_with($haystack, $needle)) {
176
                return true;
177 6
            }
178
        }
179 6
180 6
        return false;
181 6
    }
182
183
    /**
184
     * Determine if a given string ends with a given substring.
185 6
     *
186
     * @param  string  $haystack
187
     * @param  string|string[]  $needles
188
     *
189
     * @return bool
190
     */
191
    public function endsWith(string $haystack, $needles): bool
192
    {
193
        foreach ((array) $needles as $needle) {
194
            if ((string) $needle !== '' && str_ends_with($haystack, $needle)) {
195
                return true;
196
            }
197 127
        }
198
199 127
        return false;
200
    }
201
202
    /**
203
     * Convert the given string to lower-case.
204
     *
205
     * @see https://github.com/illuminate/support/blob/master/Str.php
206
     *
207
     * @param  string|null  $value
208
     *
209
     * @return string
210
     */
211 2
    public function lower(?string $value): string
212
    {
213 2
        return mb_strtolower($value, 'UTF-8');
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of mb_strtolower() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

213
        return mb_strtolower(/** @scrutinizer ignore-type */ $value, 'UTF-8');
Loading history...
214
    }
215
216
    /**
217
     * Convert the given string to upper-case.
218
     *
219
     * @see https://github.com/illuminate/support/blob/master/Str.php
220
     *
221
     * @param  string|null  $value
222
     *
223
     * @return string
224
     */
225 39
    public function upper(?string $value): ?string
226
    {
227 39
        return mb_strtoupper($value, 'UTF-8');
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of mb_strtoupper() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

227
        return mb_strtoupper(/** @scrutinizer ignore-type */ $value, 'UTF-8');
Loading history...
228
    }
229 39
230 5
    /**
231
     * Convert a value to studly caps case.
232
     *
233 35
     * @see https://github.com/illuminate/support/blob/master/Str.php
234
     *
235 35
     * @param  string|null  $value
236
     *
237
     * @return string|null
238
     */
239
    public function studly(?string $value): ?string
240
    {
241
        $key = $value;
242
243
        if (isset(self::$studlyCache[$key])) {
244
            return self::$studlyCache[$key];
245
        }
246
247 34
        $value = ucwords(str_replace(['-', '_'], ' ', $value));
248
249 34
        return self::$studlyCache[$key] = str_replace(' ', '', $value);
250 1
    }
251
252
    /**
253 33
     * Convert a value to camel case.
254
     *
255
     * @see https://github.com/illuminate/support/blob/master/Str.php
256
     *
257
     * @param  string|null  $value
258
     *
259
     * @return string|null
260
     */
261
    public function camel(?string $value): ?string
262
    {
263
        if (isset(self::$camelCache[$value])) {
264
            return self::$camelCache[$value];
265
        }
266 2
267
        return self::$camelCache[$value] = lcfirst($this->studly($value));
268 2
    }
269
270 2
    /**
271 1
     * Convert a string to snake case.
272
     *
273
     * @see https://github.com/illuminate/support/blob/master/Str.php
274 1
     *
275 1
     * @param  string|null  $value
276
     * @param  string  $delimiter
277 1
     *
278
     * @return string|null
279
     */
280 1
    public function snake(?string $value, string $delimiter = '_'): ?string
281
    {
282
        $key = $value;
283
284
        if (isset(self::$snakeCache[$key][$delimiter])) {
285
            return self::$snakeCache[$key][$delimiter];
286
        }
287
288
        if (! ctype_lower($value)) {
289
            $value = preg_replace('/\s+/u', '', ucwords($value));
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of ucwords() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

289
            $value = preg_replace('/\s+/u', '', ucwords(/** @scrutinizer ignore-type */ $value));
Loading history...
290
291
            $value = $this->lower(preg_replace('/(.)(?=[A-Z])/u', '$1' . $delimiter, $value));
292
        }
293
294 2
        return self::$snakeCache[$key][$delimiter] = $value;
295
    }
296 2
297
    /**
298
     * Generate a URL friendly "slug" from a given string.
299 2
     *
300
     * @see https://github.com/illuminate/support/blob/master/Str.php
301 2
     *
302
     * @param  string  $title
303
     * @param  string  $separator
304 2
     * @param  string|null  $language
305
     *
306
     * @return string
307 2
     */
308
    public function slug(string $title, string $separator = '-', ?string $language = 'en')
309
    {
310 2
        $title = $language ? $this->ascii($title, $language) : $title;
311
312 2
        // Convert all dashes/underscores into separator
313
        $flip = $separator === '-' ? '_' : '-';
314
315
        $title = preg_replace('![' . preg_quote($flip) . ']+!u', $separator, $title);
316
317
        // Replace @ with the word 'at'
318
        $title = str_replace('@', $separator . 'at' . $separator, $title);
319
320
        // Remove all characters that are not the separator, letters, numbers, or whitespace.
321
        $title = preg_replace('![^' . preg_quote($separator) . '\pL\pN\s]+!u', '', $this->lower($title));
322
323
        // Replace all separator characters and whitespace by a single separator
324 2
        $title = preg_replace('![' . preg_quote($separator) . '\s]+!u', $separator, $title);
325
326 2
        return trim($title, $separator);
327 2
    }
328
329
    /**
330 2
     * Convert the given string to title case.
331
     *
332
     * @see https://github.com/illuminate/support/blob/master/Str.php
333
     *
334
     * @param  string|null  $value
335
     *
336
     * @return string|null
337
     */
338
    public function title(?string $value): ?string
339
    {
340
        if (is_numeric($value)) {
341
            return $value;
342
        }
343 2
344
        return mb_convert_case($value, MB_CASE_TITLE, 'UTF-8') ?: null;
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of mb_convert_case() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

344
        return mb_convert_case(/** @scrutinizer ignore-type */ $value, MB_CASE_TITLE, 'UTF-8') ?: null;
Loading history...
345 2
    }
346 2
347 2
    /**
348
     * Return the length of the given string.
349
     *
350
     * @see https://github.com/illuminate/support/blob/master/Str.php
351
     *
352
     * @param  string|null  $value
353
     * @param  string|null  $encoding
354
     *
355
     * @return int
356
     */
357
    public function length(?string $value, string $encoding = null): int
358
    {
359
        return $encoding
360
            ? mb_strlen($value, $encoding)
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type null; however, parameter $string of mb_strlen() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

360
            ? mb_strlen(/** @scrutinizer ignore-type */ $value, $encoding)
Loading history...
361 2
            : mb_strlen($value);
362
    }
363 2
364
    /**
365
     * Returns the portion of string specified by the start and length parameters.
366
     *
367
     * @see https://github.com/illuminate/support/blob/master/Str.php
368
     *
369
     * @param  string  $string
370
     * @param  int  $start
371
     * @param  int|null  $length
372
     *
373
     * @return string|null
374
     */
375 18
    public function substr(string $string, int $start, int $length = null): ?string
376
    {
377 18
        return mb_substr($string, $start, $length, 'UTF-8');
378
    }
379 18
380
    /**
381
     * Replace all occurrences of the search string with the replacement string.
382
     *
383
     * @param  string  $template
384
     * @param  array  $values
385
     * @param  string|null  $key_format
386
     *
387
     * @return string
388
     */
389
    public function replace(string $template, array $values, string $key_format = null): string
390
    {
391
        $keys = Replace::toFormatArray(array_keys($values), $key_format);
392
393
        return str_replace($keys, array_values($values), $template);
394
    }
395
396
    /**
397
     * Get the portion of a string before the first occurrence of a given value.
398
     *
399
     * @see https://github.com/illuminate/support/blob/master/Str.php
400
     *
401
     * @param  string  $subject
402
     * @param  string  $search
403
     *
404
     * @return string
405
     */
406
    public function before(string $subject, string $search): ?string
407 88
    {
408
        return ! empty($search) ? explode($search, $subject)[0] : null;
409 88
    }
410
411
    /**
412
     * Return the remainder of a string after the first occurrence of a given value.
413
     *
414
     * @see https://github.com/illuminate/support/blob/master/Str.php
415
     *
416
     * @param  string  $subject
417
     * @param  string  $search
418
     *
419
     * @return string
420 26
     */
421
    public function after(string $subject, string $search): ?string
422 26
    {
423 26
        return ! empty($search) ? array_reverse(explode($search, $subject, 2))[0] : null;
424 24
    }
425
426
    /**
427
     * Determine if a given string contains a given substring.
428 4
     *
429
     * @param  string  $haystack
430
     * @param  string|string[]  $needles
431
     *
432
     * @return bool
433
     */
434
    public function contains(string $haystack, $needles): bool
435
    {
436
        foreach ((array) $needles as $needle) {
437
            if ((string) $needle !== '' && str_contains($haystack, $needle)) {
438
                return true;
439
            }
440
        }
441
442 2
        return false;
443
    }
444 2
445
    /**
446 2
     * Generate a more truly "random" alpha-numeric string.
447 2
     *
448
     * @see https://github.com/illuminate/support/blob/master/Str.php
449 2
     *
450
     * @param  int  $length
451 2
     *
452
     * @throws \Exception
453
     *
454 2
     * @return string
455
     */
456
    public function random(int $length = 16): string
457
    {
458
        $string = '';
459
460
        while (($len = strlen($string)) < $length) {
461
            $size = $length - $len;
462
463
            $bytes = random_bytes($size);
464 8
465
            $string .= substr(str_replace(['/', '+', '='], '', base64_encode($bytes)), 0, $size);
466 8
        }
467
468 8
        return $string;
469
    }
470
471
    /**
472
     * Determines if the value is empty.
473
     *
474
     * @param  mixed  $value
475
     *
476
     * @return bool
477
     */
478 2
    public function isEmpty($value): bool
479
    {
480 2
        $value = is_string($value) ? trim($value) : $value;
481
482
        return empty($value) && ! is_numeric($value) && (is_string($value) || is_null($value));
483
    }
484
485
    /**
486
     * Determines if the value is doesn't empty.
487
     *
488
     * @param  mixed  $value
489
     *
490
     * @return bool
491
     */
492
    public function doesntEmpty($value): bool
493 4
    {
494
        return ! $this->isEmpty($value);
495 4
    }
496
497
    /**
498
     * Transliterate a UTF-8 value to ASCII.
499
     *
500
     * @see https://github.com/illuminate/support/blob/master/Str.php
501
     *
502
     * @param  string|null  $value
503
     * @param  string|null  $language
504
     *
505
     * @return string
506
     */
507
    public function ascii(?string $value, ?string $language = 'en'): string
508
    {
509
        return ASCII::to_ascii((string) $value, $language);
0 ignored issues
show
Bug introduced by
It seems like $language can also be of type null; however, parameter $language of voku\helper\ASCII::to_ascii() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

509
        return ASCII::to_ascii((string) $value, /** @scrutinizer ignore-type */ $language);
Loading history...
510
    }
511
}
512