From 125ad943e18bfbd85e11b6e50aff3757c45c0699 Mon Sep 17 00:00:00 2001 From: David Kallesen Date: Thu, 25 Jan 2024 21:23:15 +0100 Subject: [PATCH] feat: Add StringCaseFormatter --- docs/CodeDoc/Atc/Atc.md | 27 +++++ docs/CodeDoc/Atc/Index.md | 1 + docs/CodeDoc/Atc/IndexExtended.md | 6 + src/Atc/Formatters/StringCaseFormatter.cs | 92 +++++++++++++++ .../Formatters/StringCaseFormatterTests.cs | 110 ++++++++++++++++++ 5 files changed, 236 insertions(+) create mode 100644 src/Atc/Formatters/StringCaseFormatter.cs create mode 100644 test/Atc.Tests/Formatters/StringCaseFormatterTests.cs diff --git a/docs/CodeDoc/Atc/Atc.md b/docs/CodeDoc/Atc/Atc.md index 10f3fa63..77376c07 100644 --- a/docs/CodeDoc/Atc/Atc.md +++ b/docs/CodeDoc/Atc/Atc.md @@ -1144,6 +1144,33 @@ Enumeration: SortDirectionType. +
+ +## StringCaseFormatter +Provides custom string formatting based on specified case formatting options.

Supported formats: UConverts the entire string to uppercase.uCapitalizes the first character of the string.LConverts the entire string to lowercase.lConverts the first character of the string to lowercase.UlConverts the string to uppercase with the first character in lowercase.LuConverts the string to lowercase with the first character capitalized.[Format]:Applies the specified format and appends a colon ´:´ to the end.[Format].Applies the specified format and appends a period ´.´ to the end. Examples: string.Format(StringCaseFormatter.Default, "{0:U} {1:u} {2:L} {3:l}", .... 4 parameters );string.Format(StringCaseFormatter.Default, "{0:U.} {1:u.} {2:L:} {3:l:}", .... 4 parameters );string.Format(StringCaseFormatter.Default, "{0:Ul:} {1:Lu:}", ... 2 parameters );string.Format(new StringCaseFormatter(), "{0:U} {1:u} {2:L} {3:l}", ... 4 parameters ); + +>```csharp +>public class StringCaseFormatter : IFormatProvider, ICustomFormatter +>``` + +### Static Fields + +#### Default +>```csharp +>StringCaseFormatter Default +>``` +>Summary: Static `Atc.StringCaseFormatter` using `System.Globalization.CultureInfo.CurrentCulture`. +### Methods + +#### Format +>```csharp +>string Format(string format, object arg, IFormatProvider formatProvider) +>``` +#### GetFormat +>```csharp +>object GetFormat(Type formatType) +>``` +
## TemplatePatternType diff --git a/docs/CodeDoc/Atc/Index.md b/docs/CodeDoc/Atc/Index.md index f53eaf21..5fd05d04 100644 --- a/docs/CodeDoc/Atc/Index.md +++ b/docs/CodeDoc/Atc/Index.md @@ -42,6 +42,7 @@ - [Point2D](Atc.md#point2d) - [Point3D](Atc.md#point3d) - [SortDirectionType](Atc.md#sortdirectiontype) +- [StringCaseFormatter](Atc.md#stringcaseformatter) - [TemplatePatternType](Atc.md#templatepatterntype) - [TriggerActionType](Atc.md#triggeractiontype) - [TupleEqualityComparer<T1, T2>](Atc.md#tupleequalitycomparer<t1-t2>) diff --git a/docs/CodeDoc/Atc/IndexExtended.md b/docs/CodeDoc/Atc/IndexExtended.md index e95e3088..841a1910 100644 --- a/docs/CodeDoc/Atc/IndexExtended.md +++ b/docs/CodeDoc/Atc/IndexExtended.md @@ -118,6 +118,12 @@ - ToString() - ToStringShort() - [SortDirectionType](Atc.md#sortdirectiontype) +- [StringCaseFormatter](Atc.md#stringcaseformatter) + - Static Fields + - StringCaseFormatter Default + - Methods + - Format(string format, object arg, IFormatProvider formatProvider) + - GetFormat(Type formatType) - [TemplatePatternType](Atc.md#templatepatterntype) - [TriggerActionType](Atc.md#triggeractiontype) - [TupleEqualityComparer<T1, T2>](Atc.md#tupleequalitycomparer<t1-t2>) diff --git a/src/Atc/Formatters/StringCaseFormatter.cs b/src/Atc/Formatters/StringCaseFormatter.cs new file mode 100644 index 00000000..03a62861 --- /dev/null +++ b/src/Atc/Formatters/StringCaseFormatter.cs @@ -0,0 +1,92 @@ +// ReSharper disable once CheckNamespace +namespace Atc; + +/// +/// Provides custom string formatting based on specified case formatting options.

+/// Supported formats: +/// +/// +/// U +/// Converts the entire string to uppercase. +/// +/// +/// u +/// Capitalizes the first character of the string. +/// +/// +/// L +/// Converts the entire string to lowercase. +/// +/// +/// l +/// Converts the first character of the string to lowercase. +/// +/// +/// Ul +/// Converts the string to uppercase with the first character in lowercase. +/// +/// +/// Lu +/// Converts the string to lowercase with the first character capitalized. +/// +/// +/// [Format]: +/// Applies the specified format and appends a colon ´:´ to the end. +/// +/// +/// [Format]. +/// Applies the specified format and appends a period ´.´ to the end. +/// +/// +/// Examples: +/// string.Format(StringCaseFormatter.Default, "{0:U} {1:u} {2:L} {3:l}", .... 4 parameters ); +/// string.Format(StringCaseFormatter.Default, "{0:U.} {1:u.} {2:L:} {3:l:}", .... 4 parameters ); +/// string.Format(StringCaseFormatter.Default, "{0:Ul:} {1:Lu:}", ... 2 parameters ); +/// string.Format(new StringCaseFormatter(), "{0:U} {1:u} {2:L} {3:l}", ... 4 parameters ); +///
+public sealed class StringCaseFormatter : IFormatProvider, ICustomFormatter +{ + /// + /// Static using . + /// + public static readonly StringCaseFormatter Default = new(); + + public object? GetFormat(Type? formatType) + => formatType == typeof(ICustomFormatter) + ? this + : null; + + public string Format( + string? format, + object? arg, + IFormatProvider? formatProvider) + { + var str = arg?.ToString() ?? string.Empty; + + return format switch + { + "U" => str.ToUpper(CultureInfo.CurrentCulture), + "Ul" => str.ToUpper(CultureInfo.CurrentCulture).EnsureFirstCharacterToLower(), + "u" => str.EnsureFirstCharacterToUpper(), + "L" => str.ToLower(CultureInfo.CurrentCulture), + "Lu" => str.ToLower(CultureInfo.CurrentCulture).EnsureFirstCharacterToUpper(), + "l" => str.EnsureFirstCharacterToLower(), + + "U." => str.ToUpper(CultureInfo.CurrentCulture).EnsureEndsWithDot(), + "Ul." => str.ToUpper(CultureInfo.CurrentCulture).EnsureFirstCharacterToLower().EnsureEndsWithDot(), + "u." => str.EnsureFirstCharacterToUpper().EnsureEndsWithDot(), + "L." => str.ToLower(CultureInfo.CurrentCulture).EnsureEndsWithDot(), + "Lu." => str.ToLower(CultureInfo.CurrentCulture).EnsureFirstCharacterToUpper().EnsureEndsWithDot(), + "l." => str.EnsureFirstCharacterToLower().EnsureEndsWithDot(), + + "U:" => str.ToUpper(CultureInfo.CurrentCulture).EnsureEndsWithColon(), + "Ul:" => str.ToUpper(CultureInfo.CurrentCulture).EnsureFirstCharacterToLower().EnsureEndsWithColon(), + "u:" => str.EnsureFirstCharacterToUpper().EnsureEndsWithColon(), + "L:" => str.ToLower(CultureInfo.CurrentCulture).EnsureEndsWithColon(), + "Lu:" => str.ToLower(CultureInfo.CurrentCulture).EnsureFirstCharacterToUpper().EnsureEndsWithColon(), + "l:" => str.EnsureFirstCharacterToLower().EnsureEndsWithColon(), + + _ => str, + }; + } +} \ No newline at end of file diff --git a/test/Atc.Tests/Formatters/StringCaseFormatterTests.cs b/test/Atc.Tests/Formatters/StringCaseFormatterTests.cs new file mode 100644 index 00000000..d717e4d4 --- /dev/null +++ b/test/Atc.Tests/Formatters/StringCaseFormatterTests.cs @@ -0,0 +1,110 @@ +// ReSharper disable StringLiteralTypo +#pragma warning disable xUnit1012 +namespace Atc.Tests.Formatters; + +public class StringCaseFormatterTests +{ + private readonly StringCaseFormatter formatter = StringCaseFormatter.Default; + + [Theory] + [InlineData("test", "TesT", "L")] + [InlineData("tesT", "TesT", "l")] + [InlineData("TEST", "TesT", "U")] + [InlineData("TesT", "TesT", "u")] + [InlineData("test", "tESt", "L")] + [InlineData("tESt", "tESt", "l")] + [InlineData("TEST", "tESt", "U")] + [InlineData("TESt", "tESt", "u")] + [InlineData("test.", "TesT", "L.")] + [InlineData("tesT.", "TesT", "l.")] + [InlineData("TEST.", "TesT", "U.")] + [InlineData("TesT.", "TesT", "u.")] + [InlineData("test.", "tESt", "L.")] + [InlineData("tESt.", "tESt", "l.")] + [InlineData("TEST.", "tESt", "U.")] + [InlineData("TESt.", "tESt", "u.")] + [InlineData("test:", "TesT", "L:")] + [InlineData("tesT:", "TesT", "l:")] + [InlineData("TEST:", "TesT", "U:")] + [InlineData("TesT:", "TesT", "u:")] + [InlineData("test:", "tESt", "L:")] + [InlineData("tESt:", "tESt", "l:")] + [InlineData("TEST:", "tESt", "U:")] + [InlineData("TESt:", "tESt", "u:")] + public void FormatStringForDefault( + string expected, + string input, + string formatSpecifier) + => Assert.Equal( + expected, + string.Format(StringCaseFormatter.Default, $"{{0:{formatSpecifier}}}", input)); + + [Theory] + [InlineData("test", "test", "L")] + [InlineData("test", "TEST", "L")] + [InlineData("test", "test", "l")] + [InlineData("tEST", "TEST", "l")] + [InlineData("TEST", "test", "U")] + [InlineData("TEST", "TEST", "U")] + [InlineData("Test", "test", "u")] + [InlineData("TEST", "TEST", "u")] + [InlineData("test", "test", "x")] + [InlineData("test", "test", "")] + [InlineData("", null, "U")] + [InlineData("john doe", "john doe", "L")] + [InlineData("john doe", "john doe", "l")] + [InlineData("JOHN DOE", "john doe", "U")] + [InlineData("John doe", "john doe", "u")] + [InlineData("john doe", "JOHN DOE", "L")] + [InlineData("jOHN DOE", "JOHN DOE", "l")] + [InlineData("JOHN DOE", "JOHN DOE", "U")] + [InlineData("JOHN DOE", "JOHN DOE", "u")] + [InlineData("John doe", "JOHN DOE", "Lu")] + [InlineData("jOHN DOE", "JOHN DOE", "Ul")] + [InlineData("John doe.", "john doe", "Lu.")] + [InlineData("jOHN DOE.", "JOHN DOE", "Ul.")] + [InlineData("John doe:", "john doe", "Lu:")] + [InlineData("jOHN DOE:", "john doe", "Ul:")] + public void FormatStringForOneParameter( + string expected, + string input, + string formatSpecifier) + => Assert.Equal( + expected, + string.Format(formatter, $"{{0:{formatSpecifier}}}", input)); + + [Theory] + [InlineData("hallo world", "Hallo", "World", "L", "L")] + [InlineData("hallo world", "Hallo", "World", "L", "l")] + [InlineData("hallo world", "Hallo", "World", "l", "l")] + [InlineData("hallo world", "Hallo", "World", "l", "L")] + [InlineData("hallo world", "HALLO", "WORLD", "L", "L")] + [InlineData("hallo wORLD", "HALLO", "WORLD", "L", "l")] + [InlineData("hALLO wORLD", "HALLO", "WORLD", "l", "l")] + [InlineData("hALLO world", "HALLO", "WORLD", "l", "L")] + [InlineData("hallo world", "hallo", "world", "L", "L")] + [InlineData("hallo world", "hallo", "world", "L", "l")] + [InlineData("hallo world", "hallo", "world", "l", "l")] + [InlineData("hallo world", "hallo", "world", "l", "L")] + [InlineData("HALLO WORLD", "Hallo", "World", "U", "U")] + [InlineData("HALLO World", "Hallo", "World", "U", "u")] + [InlineData("Hallo World", "Hallo", "World", "u", "u")] + [InlineData("Hallo WORLD", "Hallo", "World", "u", "U")] + [InlineData("HALLO WORLD", "HALLO", "WORLD", "U", "U")] + [InlineData("HALLO WORLD", "HALLO", "WORLD", "U", "u")] + [InlineData("HALLO WORLD", "HALLO", "WORLD", "u", "u")] + [InlineData("HALLO WORLD", "HALLO", "WORLD", "u", "U")] + [InlineData("HALLO WORLD", "hallo", "world", "U", "U")] + [InlineData("HALLO World", "hallo", "world", "U", "u")] + [InlineData("Hallo World", "hallo", "world", "u", "u")] + [InlineData("Hallo WORLD", "hallo", "world", "u", "U")] + public void FormatStringForTwoParameters( + string expected, + string input1, + string input2, + string formatSpecifier1, + string formatSpecifier2) + => Assert.Equal( + expected, + string.Format(formatter, $"{{0:{formatSpecifier1}}} {{1:{formatSpecifier2}}}", input1, input2)); +} \ No newline at end of file