diff --git a/src/BizHawk.Common/Extensions/StringExtensions.cs b/src/BizHawk.Common/Extensions/StringExtensions.cs index 52f0d47c95..3859caf2fa 100644 --- a/src/BizHawk.Common/Extensions/StringExtensions.cs +++ b/src/BizHawk.Common/Extensions/StringExtensions.cs @@ -26,6 +26,18 @@ namespace BizHawk.Common.StringExtensions /// public static string RemovePrefix(this string str, char prefix, string notFoundValue) => str.Length != 0 && str[0] == prefix ? str.Substring(1, str.Length - 1) : notFoundValue; + /// + /// with the leading substring removed, or + /// the original if does not start with + /// + public static string RemovePrefix(this string str, string prefix) => str.RemovePrefix(prefix, notFoundValue: str); + + /// + /// with the leading substring removed, or + /// if does not start with + /// + public static string RemovePrefix(this string str, string prefix, string notFoundValue) => str.StartsWith(prefix) ? str.Substring(prefix.Length, str.Length - prefix.Length) : notFoundValue; + /// /// with the last char removed, or /// the original if the last char of is not @@ -35,6 +47,34 @@ namespace BizHawk.Common.StringExtensions ? str.Substring(0, str.Length - 1) : str; + /// + /// with the trailing substring removed, or + /// the original if does not end with + /// + public static string RemoveSuffix(this string str, string suffix) => str.RemoveSuffix(suffix, notFoundValue: str); + + /// + /// with the trailing substring removed, or + /// if does not end with + /// + public static string RemoveSuffix(this string str, string suffix, string notFoundValue) => str.EndsWith(suffix) ? str.Substring(0, str.Length - suffix.Length) : notFoundValue; + + /// + /// the substring of after the first occurrence of , or + /// the original if not found + /// + public static string SubstringAfter(this string str, string delimiter) => str.SubstringAfter(delimiter, notFoundValue: str); + + /// + /// the substring of after the first occurrence of , or + /// if not found + /// + public static string SubstringAfter(this string str, string delimiter, string notFoundValue) + { + var index = str.IndexOf(delimiter); + return index < 0 ? notFoundValue : str.Substring(index + delimiter.Length, str.Length - index - delimiter.Length); + } + /// /// the substring of before the first occurrence of , or /// the original if not found @@ -51,6 +91,22 @@ namespace BizHawk.Common.StringExtensions return index < 0 ? notFoundValue : str.Substring(0, index); } + /// + /// the substring of before the last occurrence of , or + /// the original if not found + /// + public static string SubstringBeforeLast(this string str, char delimiter) => str.SubstringBeforeLast(delimiter, notFoundValue: str); + + /// + /// the substring of before the last occurrence of , or + /// if not found + /// + public static string SubstringBeforeLast(this string str, char delimiter, string notFoundValue) + { + var index = str.LastIndexOf(delimiter); + return index < 0 ? notFoundValue : str.Substring(0, index); + } + /// /// the substring of before the first occurrence of , or /// if not found diff --git a/src/BizHawk.Tests/Common/StringExtensions/StringExtensionTests.cs b/src/BizHawk.Tests/Common/StringExtensions/StringExtensionTests.cs index faf9ea2d00..1428d7f9a4 100644 --- a/src/BizHawk.Tests/Common/StringExtensions/StringExtensionTests.cs +++ b/src/BizHawk.Tests/Common/StringExtensions/StringExtensionTests.cs @@ -6,6 +6,10 @@ namespace BizHawk.Tests.Common.StringExtensions [TestClass] public class StringExtensionTests { + private const string abcdef = "abcdef"; + + private const string qrs = "qrs"; + [TestMethod] public void In_CaseInsensitive() { @@ -21,14 +25,62 @@ namespace BizHawk.Tests.Common.StringExtensions [TestMethod] public void TestRemovePrefix() { - const string abcdef = "abcdef"; - const string qrs = "qrs"; - Assert.AreEqual("bcdef", abcdef.RemovePrefix('a', qrs)); Assert.AreEqual(string.Empty, "a".RemovePrefix('a', qrs)); Assert.AreEqual(qrs, abcdef.RemovePrefix('c', qrs)); Assert.AreEqual(qrs, abcdef.RemovePrefix('x', qrs)); Assert.AreEqual(qrs, string.Empty.RemovePrefix('a', qrs)); + + Assert.AreEqual("def", abcdef.RemovePrefix("abc", qrs)); + Assert.AreEqual("bcdef", abcdef.RemovePrefix("a", qrs)); + Assert.AreEqual(abcdef, abcdef.RemovePrefix(string.Empty, qrs)); + Assert.AreEqual(string.Empty, abcdef.RemovePrefix(abcdef, qrs)); + Assert.AreEqual(string.Empty, "a".RemovePrefix("a", qrs)); + Assert.AreEqual(qrs, abcdef.RemovePrefix("c", qrs)); + Assert.AreEqual(qrs, abcdef.RemovePrefix("x", qrs)); + Assert.AreEqual(qrs, string.Empty.RemovePrefix("abc", qrs)); + } + + [TestMethod] + public void TestRemoveSuffix() + { + Assert.AreEqual("abc", abcdef.RemoveSuffix("def", qrs)); + Assert.AreEqual("abcde", abcdef.RemoveSuffix("f", qrs)); + Assert.AreEqual(abcdef, abcdef.RemoveSuffix(string.Empty, qrs)); + Assert.AreEqual(string.Empty, abcdef.RemoveSuffix(abcdef, qrs)); + Assert.AreEqual(string.Empty, "f".RemoveSuffix("f", qrs)); + Assert.AreEqual(qrs, abcdef.RemoveSuffix("d", qrs)); + Assert.AreEqual(qrs, abcdef.RemoveSuffix("x", qrs)); + Assert.AreEqual(qrs, string.Empty.RemoveSuffix("def", qrs)); + } + + [TestMethod] + public void TestSubstringAfter() + { + Assert.AreEqual("def", abcdef.SubstringAfter("bc", qrs)); + Assert.AreEqual(abcdef, abcdef.SubstringAfter(string.Empty, qrs)); + Assert.AreEqual(string.Empty, abcdef.SubstringAfter(abcdef, qrs)); + Assert.AreEqual(string.Empty, abcdef.SubstringAfter("f", qrs)); + Assert.AreEqual(string.Empty, "f".SubstringAfter("f", qrs)); + Assert.AreEqual("abcdab", "abcdabcdab".SubstringAfter("cd", qrs)); + Assert.AreEqual(qrs, abcdef.SubstringAfter("x", qrs)); + Assert.AreEqual(qrs, string.Empty.SubstringAfter("abc", qrs)); + } + + [TestMethod] + public void TestSubstringBefore() + { + Assert.AreEqual("abc", abcdef.SubstringBefore('d', qrs)); + Assert.AreEqual(string.Empty, abcdef.SubstringBefore('a', qrs)); + Assert.AreEqual(string.Empty, "a".SubstringBefore('a', qrs)); + Assert.AreEqual("abc", "abcdabcdab".SubstringBefore('d', qrs)); + Assert.AreEqual(qrs, abcdef.SubstringBefore('x', qrs)); + Assert.AreEqual(qrs, string.Empty.SubstringBefore('d', qrs)); + + // fewer tests for SubstringBeforeLast as its implementation should match SubstringBefore, save for using LastIndexOf + + Assert.AreEqual("abcdabc", "abcdabcdab".SubstringBeforeLast('d', qrs)); + Assert.AreEqual(qrs, "abcdabcdab".SubstringBeforeLast('x', qrs)); } } }