| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using System.Linq; |
| | 4 | | using System.Text.RegularExpressions; |
| | 5 | | using ICU4N.Text; |
| | 6 | |
|
| | 7 | | namespace Jellyfin.Extensions |
| | 8 | | { |
| | 9 | | /// <summary> |
| | 10 | | /// Provides extensions methods for <see cref="string" />. |
| | 11 | | /// </summary> |
| | 12 | | public static partial class StringExtensions |
| | 13 | | { |
| 0 | 14 | | private static readonly Lazy<string> _transliteratorId = new(() => |
| 0 | 15 | | Environment.GetEnvironmentVariable("JELLYFIN_TRANSLITERATOR_ID") |
| 0 | 16 | | ?? "Any-Latin; Latin-Ascii; Lower; NFD; [:Nonspacing Mark:] Remove; [:Punctuation:] Remove;"); |
| | 17 | |
|
| 0 | 18 | | private static readonly Lazy<Transliterator?> _transliterator = new(() => |
| 0 | 19 | | { |
| 0 | 20 | | try |
| 0 | 21 | | { |
| 0 | 22 | | return Transliterator.GetInstance(_transliteratorId.Value); |
| 0 | 23 | | } |
| 0 | 24 | | catch (ArgumentException) |
| 0 | 25 | | { |
| 0 | 26 | | return null; |
| 0 | 27 | | } |
| 0 | 28 | | }); |
| | 29 | |
|
| | 30 | | // Matches non-conforming unicode chars |
| | 31 | | // https://mnaoumov.wordpress.com/2014/06/14/stripping-invalid-characters-from-utf-16-strings/ |
| | 32 | |
|
| | 33 | | [GeneratedRegex("([\ud800-\udbff](?![\udc00-\udfff]))|((?<![\ud800-\udbff])[\udc00-\udfff])|(�)")] |
| | 34 | | private static partial Regex NonConformingUnicodeRegex(); |
| | 35 | |
|
| | 36 | | /// <summary> |
| | 37 | | /// Removes the diacritics character from the strings. |
| | 38 | | /// </summary> |
| | 39 | | /// <param name="text">The string to act on.</param> |
| | 40 | | /// <returns>The string without diacritics character.</returns> |
| | 41 | | public static string RemoveDiacritics(this string text) |
| 208 | 42 | | => Diacritics.Extensions.StringExtensions.RemoveDiacritics( |
| 208 | 43 | | NonConformingUnicodeRegex().Replace(text, string.Empty)); |
| | 44 | |
|
| | 45 | | /// <summary> |
| | 46 | | /// Checks whether or not the specified string has diacritics in it. |
| | 47 | | /// </summary> |
| | 48 | | /// <param name="text">The string to check.</param> |
| | 49 | | /// <returns>True if the string has diacritics, false otherwise.</returns> |
| | 50 | | public static bool HasDiacritics(this string text) |
| 12 | 51 | | => Diacritics.Extensions.StringExtensions.HasDiacritics(text) |
| 12 | 52 | | || NonConformingUnicodeRegex().IsMatch(text); |
| | 53 | |
|
| | 54 | | /// <summary> |
| | 55 | | /// Counts the number of occurrences of [needle] in the string. |
| | 56 | | /// </summary> |
| | 57 | | /// <param name="value">The haystack to search in.</param> |
| | 58 | | /// <param name="needle">The character to search for.</param> |
| | 59 | | /// <returns>The number of occurrences of the [needle] character.</returns> |
| | 60 | | public static int Count(this ReadOnlySpan<char> value, char needle) |
| | 61 | | { |
| 4 | 62 | | var count = 0; |
| 4 | 63 | | var length = value.Length; |
| 108 | 64 | | for (var i = 0; i < length; i++) |
| | 65 | | { |
| 50 | 66 | | if (value[i] == needle) |
| | 67 | | { |
| 6 | 68 | | count++; |
| | 69 | | } |
| | 70 | | } |
| | 71 | |
|
| 4 | 72 | | return count; |
| | 73 | | } |
| | 74 | |
|
| | 75 | | /// <summary> |
| | 76 | | /// Returns the part on the left of the <c>needle</c>. |
| | 77 | | /// </summary> |
| | 78 | | /// <param name="haystack">The string to seek.</param> |
| | 79 | | /// <param name="needle">The needle to find.</param> |
| | 80 | | /// <returns>The part left of the <paramref name="needle" />.</returns> |
| | 81 | | public static ReadOnlySpan<char> LeftPart(this ReadOnlySpan<char> haystack, char needle) |
| | 82 | | { |
| 377 | 83 | | if (haystack.IsEmpty) |
| | 84 | | { |
| 2 | 85 | | return ReadOnlySpan<char>.Empty; |
| | 86 | | } |
| | 87 | |
|
| 375 | 88 | | var pos = haystack.IndexOf(needle); |
| 375 | 89 | | return pos == -1 ? haystack : haystack[..pos]; |
| | 90 | | } |
| | 91 | |
|
| | 92 | | /// <summary> |
| | 93 | | /// Returns the part on the right of the <c>needle</c>. |
| | 94 | | /// </summary> |
| | 95 | | /// <param name="haystack">The string to seek.</param> |
| | 96 | | /// <param name="needle">The needle to find.</param> |
| | 97 | | /// <returns>The part right of the <paramref name="needle" />.</returns> |
| | 98 | | public static ReadOnlySpan<char> RightPart(this ReadOnlySpan<char> haystack, char needle) |
| | 99 | | { |
| 9 | 100 | | if (haystack.IsEmpty) |
| | 101 | | { |
| 1 | 102 | | return ReadOnlySpan<char>.Empty; |
| | 103 | | } |
| | 104 | |
|
| 8 | 105 | | var pos = haystack.LastIndexOf(needle); |
| 8 | 106 | | if (pos == -1) |
| | 107 | | { |
| 1 | 108 | | return haystack; |
| | 109 | | } |
| | 110 | |
|
| 7 | 111 | | if (pos == haystack.Length - 1) |
| | 112 | | { |
| 3 | 113 | | return ReadOnlySpan<char>.Empty; |
| | 114 | | } |
| | 115 | |
|
| 4 | 116 | | return haystack[(pos + 1)..]; |
| | 117 | | } |
| | 118 | |
|
| | 119 | | /// <summary> |
| | 120 | | /// Returns a transliterated string which only contain ascii characters. |
| | 121 | | /// </summary> |
| | 122 | | /// <param name="text">The string to act on.</param> |
| | 123 | | /// <returns>The transliterated string.</returns> |
| | 124 | | public static string Transliterated(this string text) |
| | 125 | | { |
| 0 | 126 | | return (_transliterator.Value is null) ? text : _transliterator.Value.Transliterate(text); |
| | 127 | | } |
| | 128 | |
|
| | 129 | | /// <summary> |
| | 130 | | /// Ensures all strings are non-null and trimmed of leading an trailing blanks. |
| | 131 | | /// </summary> |
| | 132 | | /// <param name="values">The enumerable of strings to trim.</param> |
| | 133 | | /// <returns>The enumeration of trimmed strings.</returns> |
| | 134 | | public static IEnumerable<string> Trimmed(this IEnumerable<string> values) |
| | 135 | | { |
| 5 | 136 | | return values.Select(i => (i ?? string.Empty).Trim()); |
| | 137 | | } |
| | 138 | | } |
| | 139 | | } |