Как мне конвертировать PascalCase в кебаб-кейс с C#?

Как мне преобразовать строковое значение в PascalCase (другое имя UpperCamelCase) в кебаб-кейс с C#?

например "VeryLongName" в "very-long-name"

8 ответов

Решение

Вот как это сделать с помощью регулярного выражения:

public static class StringExtensions
{
    public static string PascalToKebabCase(this string value)
    {
        if (string.IsNullOrEmpty(value))
            return value;

        return Regex.Replace(
            value,
            "(?<!^)([A-Z][a-z]|(?<=[a-z])[A-Z])",
            "-$1",
            RegexOptions.Compiled)
            .Trim()
            .ToLower();
    }
}

Вот мое решение, использующее соглашения о капитализации Microsoft. https://docs.microsoft.com/en-us/dotnet/standard/design-guidelines/capitalization-conventions

public static class StringExtensions
{
    public static string ToKebabCase(this string source)
    {
        if (source is null) return null;

        if(source.Length == 0) return string.Empty;

        StringBuilder builder = new StringBuilder();

        for(var i = 0; i < source.Length; i++)
        {
            if (char.IsLower(source[i])) // if current char is already lowercase
            {
                builder.Append(source[i]);
            }
            else if(i == 0) // if current char is the first char
            {
                builder.Append(char.ToLower(source[i]));
            }
            else if (char.IsLower(source[i - 1])) // if current char is upper and previous char is lower
            {
                builder.Append("-");
                builder.Append(char.ToLower(source[i]));
            }
            else if(i + 1 == source.Length || char.IsUpper(source[i + 1])) // if current char is upper and next char doesn't exist or is upper
            {
                builder.Append(char.ToLower(source[i]));
            }
            else // if current char is upper and next char is lower
            {
                builder.Append("-");
                builder.Append(char.ToLower(source[i]));
            }
        }
        return builder.ToString();
    }
}

Тестовое задание

string[] stringArray = new[]
{
    null,
    "",
    "I",
    "IO",
    "FileIO",
    "SignalR",
    "IOStream",
    "COMObject",
    "WebAPI"
};

foreach (var str in stringArray)
{
    Console.WriteLine($"{str} --> {str.ToKebabCase()}");
}

// Output:
//  -->
//  -->
// I --> i
// IO --> io
// FileIO --> file-io
// SignalR --> signal-r
// IOStream --> io-stream
// COMObject --> com-object
// WebAPI --> web-api

Вот способ сделать это без использования Regex:

public static string PascalToKebabCase(this string str)
{
    if (string.IsNullOrEmpty(str))
        return string.Empty;

    var builder = new StringBuilder();
    builder.Append(char.ToLower(str.First()));

    foreach (var c in str.Skip(1))
    {
        if (char.IsUpper(c))
        {
            builder.Append('-');
            builder.Append(char.ToLower(c));
        }
        else
        {
            builder.Append(c);
        }
    }

    return builder.ToString();
}

У вас будут проблемы с хорошо известными сокращениями, которые используют заглавные буквы. Например: COMObject, Это решение, очевидно, не будет работать.

Это то, что я придумал, взяв кусочки кода и регулярное выражение от других:

var pascalCase = "MyReally-CoolMFAString";
var dashCase = Regex.Replace(pascalCase, @"(?<!^)(?<!-)((?<=\p{Ll})\p{Lu}|\p{Lu}(?=\p{Ll}))", "-$1").ToLower();
Console.WriteLine(dashCase);

Результат:

моя-действительно-крутая-мфа-строка

Вдохновленный @InBetween:

      public static string PascalToKebabCase(this string str)
{
    IEnumerable<char> ConvertChar(char c, int index)
    {
        if (char.IsUpper(c))
        {
            if (index != 0) yield return char.ToLower('-');
            yield return char.ToLower(c);
        }
        else yield return c;
    }

    return string.Concat(str.SelectMany(ConvertChar));
}

Вдохновленный @johnathan-barclay:

      string.Concat(str.Select((c, i) => (char.IsUpper(c) && i > 0 ? "-" : "") + char.ToLower(c)));

Вот более короткое решение с Regex.Replace:

public static class KebabConverter
{
    public static string ToKebabCase(this string str)
    {
        // find and replace all parts that starts with one capital letter e.g. Net
        var str1 = Regex.Replace(str, "[A-Z][a-z]+", m => $"-{m.ToString().ToLower()}");
        
        // find and replace all parts that are all capital letter e.g. NET
        var str2 = Regex.Replace(str1, "[A-Z]+", m => $"-{m.ToString().ToLower()}");
        
        return str2.TrimStart('-');
    }
}

https://dotnetfiddle.net/WSE6sy

              string input = "VeryLongName";
    
    char[] arr = input.ToCharArray();

    string output = Convert.ToString( arr[0]);

    for(int i=1; i<arr.Length; i++){

        if(char.IsUpper(arr[i]))
            output = output + "-" + Convert.ToString( arr[i]);
        else
            output = output + Convert.ToString( arr[i]);
    }

    Console.WriteLine(output.ToLower());
Другие вопросы по тегам