FusionReactor.SourceGenerators.EnumExtensions
2.0.0
dotnet add package FusionReactor.SourceGenerators.EnumExtensions --version 2.0.0
NuGet\Install-Package FusionReactor.SourceGenerators.EnumExtensions -Version 2.0.0
<PackageReference Include="FusionReactor.SourceGenerators.EnumExtensions" Version="2.0.0" />
<PackageVersion Include="FusionReactor.SourceGenerators.EnumExtensions" Version="2.0.0" />
<PackageReference Include="FusionReactor.SourceGenerators.EnumExtensions" />
paket add FusionReactor.SourceGenerators.EnumExtensions --version 2.0.0
#r "nuget: FusionReactor.SourceGenerators.EnumExtensions, 2.0.0"
#addin nuget:?package=FusionReactor.SourceGenerators.EnumExtensions&version=2.0.0
#tool nuget:?package=FusionReactor.SourceGenerators.EnumExtensions&version=2.0.0
FusionReactor.SourceGenerators.EnumExtensions
A C# source generator to create extensions for an enum type.
- Optimized for speed and low resource consumption
- Support for non-standard enum declarations
public enum EPublicFoo : byte
- .NET 8+ support by using FrozenDictionary & FrozenSet
- .NET 5+ support by using IReadOnlyDictionary & IReadOnlySet
- .NET Framework 4.5+ support by using IReadOnlyDictionary & HashSet
Package - FusionReactor.SourceGenerators.EnumExtensions
Add the package to your application using
dotnet add package FusionReactor.SourceGenerators.EnumExtensions
Adding the package will automatically add a marker attribute, [GenerateEnumExtensions]
, to your project.
To use the generator, add the [GenerateEnumExtensions]
attribute to an enum. For example:
[GenerateEnumExtensions]
[Description("Test 123")]
[IntArray(new[] { 123 })]
[StringArray(new[] { "Foo", "Bar", "Baz" })]
public enum EPublicFoo
{
[Display(
ShortName = "Fo",
Name = "Foo - 0",
Description = "Zero",
Prompt = "ooF",
GroupName = "Foos",
Order = 0)]
Foo = 0,
[Display(
ShortName = "Ba",
Name = "Bar - 1",
Description = "One",
Prompt = "raB",
GroupName = "Bars",
Order = 1)]
Bar = 1,
Batz = 2,
}
This will generate a class called EPublicFooExtensions
(EPublicFoo
+ Extensions
), which contains a number of helper methods.
EPublicFoo.Extensions.g.cs
:
/// <summary>
/// Extension methods for the <see cref = "EPublicFoo"/> enum.
/// </summary>
[GeneratedCode("FusionReactor.SourceGenerators.EnumExtensions", null)]
public static class EPublicFooExtensions
{
#if NET8_0_OR_GREATER
private static readonly FrozenDictionary<EPublicFoo, Int32> content
= new Dictionary<EPublicFoo, Int32>
{
{ EPublicFoo.Foo, 0 },
{ EPublicFoo.Bar, 1 },
{ EPublicFoo.Batz, 2 },
}
.ToFrozenDictionary();
#else
private static readonly Dictionary<EPublicFoo, Int32> contentDictionary
= new Dictionary<EPublicFoo, Int32>
{
{ EPublicFoo.Foo, 0 },
{ EPublicFoo.Bar, 1 },
{ EPublicFoo.Batz, 2 },
};
private static readonly IReadOnlyDictionary<EPublicFoo, Int32> content
= new ReadOnlyDictionary<EPublicFoo, Int32>(contentDictionary);
#endif
#if NET8_0_OR_GREATER
private static readonly FrozenSet<string> names
= new HashSet<string>()
{
"Foo",
"Bar",
"Batz",
}
.ToFrozenSet();
#elif NET5_0_OR_GREATER
private static readonly IReadOnlySet<string> names
= new HashSet<string>()
{
"Foo",
"Bar",
"Batz",
};
#else
private static readonly HashSet<string> names
= new HashSet<string>()
{
"Foo",
"Bar",
"Batz",
};
#endif
#if NET8_0_OR_GREATER
private static readonly FrozenSet<EPublicFoo> values
= new HashSet<EPublicFoo>()
{
EPublicFoo.Foo,
EPublicFoo.Bar,
EPublicFoo.Batz,
}
.ToFrozenSet();
#elif NET5_0_OR_GREATER
private static readonly IReadOnlySet<EPublicFoo> names
= new HashSet<EPublicFoo>()
{
EPublicFoo.Foo,
EPublicFoo.Bar,
EPublicFoo.Batz,
};
#else
private static readonly HashSet<EPublicFoo> names
= new HashSet<EPublicFoo>()
{
EPublicFoo.Foo,
EPublicFoo.Bar,
EPublicFoo.Batz,
};
#endif
#if NET8_0_OR_GREATER
private static readonly FrozenSet<Attribute> rootAttributes
= new HashSet<Attribute>()
{
new global::FusionReactor.SourceGenerators.EnumExtensions.GenerateEnumExtensionsAttribute(),
new global::System.ComponentModel.DescriptionAttribute(
"Test 123"),
new global::FusionReactor.SourceGenerators.EnumExtensions.Tests.Attributes.IntArrayAttribute(
new int[]{ 123 }),
new global::FusionReactor.SourceGenerators.EnumExtensions.Tests.Attributes.StringArrayAttribute(
new string[]{ "Foo", "Bar", "Baz" }),
}
.ToFrozenSet();
#elif NET5_0_OR_GREATER
private static readonly IReadOnlySet<Attribute> rootAttributes
= new HashSet<Attribute>()
{
new global::FusionReactor.SourceGenerators.EnumExtensions.GenerateEnumExtensionsAttribute(),
new global::System.ComponentModel.DescriptionAttribute(
"Test 123"),
new global::FusionReactor.SourceGenerators.EnumExtensions.Tests.Attributes.IntArrayAttribute(
new int[]{ 123 }),
new global::FusionReactor.SourceGenerators.EnumExtensions.Tests.Attributes.StringArrayAttribute(
new string[]{ "Foo", "Bar", "Baz" }),
};
#else
private static readonly HashSet<Attribute> rootAttributes
= new HashSet<Attribute>()
{
new global::FusionReactor.SourceGenerators.EnumExtensions.GenerateEnumExtensionsAttribute(),
new global::System.ComponentModel.DescriptionAttribute(
"Test 123"),
new global::FusionReactor.SourceGenerators.EnumExtensions.Tests.Attributes.IntArrayAttribute(
new int[]{ 123 }),
new global::FusionReactor.SourceGenerators.EnumExtensions.Tests.Attributes.StringArrayAttribute(
new string[]{ "Foo", "Bar", "Baz" }),
};
#endif
#if NET8_0_OR_GREATER
private static readonly FrozenDictionary<EPublicFoo, Attribute[]> memberAttributes
= new Dictionary<EPublicFoo, Attribute[]>()
{
{
EPublicFoo.Foo,
new Attribute[1]
{
new global::System.ComponentModel.DataAnnotations.DisplayAttribute()
{
ShortName = "Fo",
Name = "Foo - 0",
Description = "Zero",
Prompt = "ooF",
GroupName = "Foos",
Order = 0,
},
}
},
{
EPublicFoo.Bar,
new Attribute[1]
{
new global::System.ComponentModel.DataAnnotations.DisplayAttribute()
{
ShortName = "Ba",
Name = "Bar - 1",
Description = "One",
Prompt = "raB",
GroupName = "Bars",
Order = 1,
},
}
},
{
EPublicFoo.Batz,
new Attribute[0]
},
}
.ToFrozenDictionary();
#else
private static readonly Dictionary<EPublicFoo, Attribute[]> memberAttributesDictionary
= new Dictionary<EPublicFoo, Attribute[]>()
{
{
EPublicFoo.Foo,
new Attribute[1]
{
new global::System.ComponentModel.DataAnnotations.DisplayAttribute()
{
ShortName = "Fo",
Name = "Foo - 0",
Description = "Zero",
Prompt = "ooF",
GroupName = "Foos",
Order = 0,
},
}
},
{
EPublicFoo.Bar,
new Attribute[1]
{
new global::System.ComponentModel.DataAnnotations.DisplayAttribute()
{
ShortName = "Ba",
Name = "Bar - 1",
Description = "One",
Prompt = "raB",
GroupName = "Bars",
Order = 1,
},
}
},
{
EPublicFoo.Batz,
new Attribute[0]
},
};
private static readonly IReadOnlyDictionary<EPublicFoo, Attribute[]> memberAttributes
= new ReadOnlyDictionary<EPublicFoo, Attribute[]>(memberAttributesDictionary);
#endif
/// <summary>
/// Gets the content dictionary containing mappings of <see cref="EPublicFoo"/> enum values to values.
/// </summary>
/// <returns>The read-only content dictionary.</returns>
#if NET8_0_OR_GREATER
public static FrozenDictionary<EPublicFoo, Int32> GetContent()
#else
public static IReadOnlyDictionary<EPublicFoo, Int32> GetContent()
#endif
=> content;
/// <summary>
/// Gets the content dictionary containing mappings of <see cref="EPublicFoo"/> enum values to values.
/// </summary>
/// <param name="enumValue">The enum value for which to get the content dictionary.</param>
/// <returns>The read-only content dictionary.</returns>
#if NET8_0_OR_GREATER
public static FrozenDictionary<EPublicFoo, Int32> GetContent(this EPublicFoo enumValue)
#else
public static IReadOnlyDictionary<EPublicFoo, Int32> GetContent(this EPublicFoo enumValue)
#endif
=> content;
/// <summary>
/// Retrieves the name of the constant in the <see cref="EPublicFoo"/>.
/// </summary>
/// <param name="enumValue">The enum value to convert.</param>
/// <returns>
/// A string containing the name of the <see cref="EPublicFoo"/>;
/// or <see langword="null"/> if no such constant is found.
/// </returns>
public static string? GetName(this EPublicFoo enumValue)
=> enumValue switch
{
EPublicFoo.Foo => nameof(EPublicFoo.Foo),
EPublicFoo.Bar => nameof(EPublicFoo.Bar),
EPublicFoo.Batz => nameof(EPublicFoo.Batz),
_ => null,
};
/// <summary>
/// Retrieves all available names of the <see cref="EPublicFoo"/>.
/// </summary>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> names.</returns>
#if NET8_0_OR_GREATER
public static FrozenSet<string> GetNames()
#elif NET5_0_OR_GREATER
public static IReadOnlySet<string> GetNames()
#else
public static HashSet<string> GetNames()
#endif
=> names;
/// <summary>
/// Retrieves all available names of the <see cref="EPublicFoo"/>.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> names.</returns>
#if NET8_0_OR_GREATER
public static FrozenSet<string> GetNames(this EPublicFoo enumValue)
#elif NET5_0_OR_GREATER
public static IReadOnlySet<string> GetNames(this EPublicFoo enumValue)
#else
public static HashSet<string> GetNames(this EPublicFoo enumValue)
#endif
=> names;
/// <summary>
/// Retrieves all available values of the <see cref="EPublicFoo"/>.
/// </summary>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> values.</returns>
#if NET8_0_OR_GREATER
public static FrozenSet<EPublicFoo> GetValues()
#elif NET5_0_OR_GREATER
public static IReadOnlySet<EPublicFoo> GetValues()
#else
public static HashSet<EPublicFoo> GetValues()
#endif
=> values;
/// <summary>
/// Retrieves all available values of the <see cref="EPublicFoo"/>.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> values.</returns>
#if NET8_0_OR_GREATER
public static FrozenSet<EPublicFoo> GetValues(this EPublicFoo enumValue)
#elif NET5_0_OR_GREATER
public static IReadOnlySet<EPublicFoo> GetValues(this EPublicFoo enumValue)
#else
public static HashSet<EPublicFoo> GetValues(this EPublicFoo enumValue)
#endif
=> values;
/// <summary>
/// Parses the specified string representation of the enumeration value to its corresponding
/// <see cref="EPublicFoo"/> value.
/// </summary>
/// <param name="value">A string containing the name or value to convert.</param>
/// <param name="ignoreCase">
/// A boolean indicating whether to ignore case during the parsing. Default is <c>false</c>.
/// </param>
/// <returns>
/// The <see cref="EPublicFoo"/> value equivalent to the specified string representation.
/// </returns>
public static EPublicFoo ParseFast(string value, bool ignoreCase = false)
{
if (ignoreCase)
{
return value.ToLowerInvariant() switch
{
"foo" => EPublicFoo.Foo,
"bar" => EPublicFoo.Bar,
"batz" => EPublicFoo.Batz,
_ => throw new ArgumentException(),
};
}
else
{
return value switch
{
"Foo" => EPublicFoo.Foo,
"Bar" => EPublicFoo.Bar,
"Batz" => EPublicFoo.Batz,
_ => throw new ArgumentException(),
};
}
}
/// <summary>
/// Parses the specified string representation of the enumeration value to its corresponding
/// <see cref="EPublicFoo"/> value.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <param name="value">A string containing the name or value to convert.</param>
/// <param name="ignoreCase">
/// A boolean indicating whether to ignore case during the parsing. Default is <c>false</c>.
/// </param>
/// <returns>
/// The <see cref="EPublicFoo"/> value equivalent to the specified string representation.
/// </returns>
public static EPublicFoo ParseFast(this EPublicFoo enumValue, string value, bool ignoreCase = false)
{
if (ignoreCase)
{
return value.ToLowerInvariant() switch
{
"foo" => EPublicFoo.Foo,
"bar" => EPublicFoo.Bar,
"batz" => EPublicFoo.Batz,
_ => throw new ArgumentException(),
};
}
else
{
return value switch
{
"Foo" => EPublicFoo.Foo,
"Bar" => EPublicFoo.Bar,
"Batz" => EPublicFoo.Batz,
_ => throw new ArgumentException(),
};
}
}
/// <summary>
/// Tries to parse the specified string representation of an enumeration value to its corresponding
/// <see cref="EPublicFoo"/> enumeration value.
/// </summary>
/// <param name="value">The string representation of the enumeration value.</param>
/// <param name="result">
/// When this method returns, contains the <see cref="EPublicFoo"/> value equivalent
/// to the string representation, if the parse succeeded, or default(EPublicFoo) if the parse failed.</param>
/// <returns><c>true</c> if the parsing was successful; otherwise, <c>false</c>.</returns>
public static bool TryParseFast(string value, out EPublicFoo? result)
=> TryParseFast(value, false, out result);
/// <summary>
/// Tries to parse the specified string representation of an enumeration value to its corresponding
/// <see cref="EPublicFoo"/> enumeration value.
/// </summary>
/// <param name="value">The value to parse.</param>
/// <param name="ignoreCase">
/// A boolean indicating whether to ignore case during the parsing. Default is <c>false</c>.
/// </param>
/// <param name="result">
/// When this method returns, contains the <see cref="EPublicFoo"/> value equivalent
/// to the string representation, if the parse succeeded, or <c>null</c> if the parse failed.
/// </param>
/// <returns><c>true</c> if the parsing was successful; otherwise, <c>false</c>.</returns>
public static bool TryParseFast(string value, bool ignoreCase, out EPublicFoo? result)
{
if (ignoreCase)
{
result = value.ToLowerInvariant() switch
{
"foo" => EPublicFoo.Foo,
"bar" => EPublicFoo.Bar,
"batz" => EPublicFoo.Batz,
_ => null,
};
}
else
{
result = value switch
{
"Foo" => EPublicFoo.Foo,
"Bar" => EPublicFoo.Bar,
"Batz" => EPublicFoo.Batz,
_ => null,
};
}
return result is not null;
}
/// <summary>
/// Tries to parse the specified string representation of an enumeration value to its corresponding
/// <see cref="EPublicFoo"/> enumeration value.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <param name="value">The value to parse.</param>
/// <param name="result">
/// When this method returns, contains the <see cref="EPublicFoo"/> value equivalent
/// to the string representation, if the parse succeeded, or <c>null</c> if the parse failed.
/// </param>
/// <returns><c>true</c> if the parsing was successful; otherwise, <c>false</c>.</returns>
public static bool TryParseFast(this EPublicFoo enumValue, string value, out EPublicFoo? result)
=> enumValue.TryParseFast(value, false, out result);
/// <summary>
/// Tries to parse the specified string representation of an enumeration value to its corresponding
/// <see cref="EPublicFoo"/> enumeration value.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <param name="value">The value to parse.</param>
/// <param name="ignoreCase">
/// A boolean indicating whether to ignore case during the parsing. Default is <c>false</c>.
/// </param>
/// <param name="result">
/// When this method returns, contains the <see cref="EPublicFoo"/> value equivalent
/// to the string representation, if the parse succeeded, or <c>null</c> if the parse failed.
/// </param>
/// <returns><c>true</c> if the parsing was successful; otherwise, <c>false</c>.</returns>
public static bool TryParseFast(this EPublicFoo enumValue, string value, bool ignoreCase, out EPublicFoo? result)
{
if (ignoreCase)
{
result = value.ToLowerInvariant() switch
{
"foo" => EPublicFoo.Foo,
"bar" => EPublicFoo.Bar,
"batz" => EPublicFoo.Batz,
_ => null,
};
}
else
{
result = value switch
{
"Foo" => EPublicFoo.Foo,
"Bar" => EPublicFoo.Bar,
"Batz" => EPublicFoo.Batz,
_ => null,
};
}
return result is not null;
}
/// <summary>
/// Retrieves all available attributes of the <see cref="EPublicFoo"/>.
/// </summary>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> attributes.</returns>
#if NET8_0_OR_GREATER
public static FrozenSet<Attribute> GetRootAttributes()
#elif NET5_0_OR_GREATER
public static IReadOnlySet<Attribute> GetRootAttributes()
#else
public static HashSet<Attribute> GetRootAttributes()
#endif
=> rootAttributes;
/// <summary>
/// Retrieves all available attributes of the <see cref="EPublicFoo"/>.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> attributes.</returns>
#if NET8_0_OR_GREATER
public static FrozenSet<Attribute> GetRootAttributes(this EPublicFoo enumValue)
#elif NET5_0_OR_GREATER
public static IReadOnlySet<Attribute> GetRootAttributes(this EPublicFoo enumValue)
#else
public static HashSet<Attribute> GetRootAttributes(this EPublicFoo enumValue)
#endif
=> rootAttributes;
/// <summary>
/// Retrieves all available attributes of the <see cref="EPublicFoo"/>.
/// </summary>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> attributes.</returns>
#if NET8_0_OR_GREATER
public static FrozenDictionary<EPublicFoo, Attribute[]> GetMemberAttributes()
#else
public static Dictionary<EPublicFoo, Attribute[]> GetMemberAttributes()
#endif
=> memberAttributes;
/// <summary>
/// Retrieves all available attributes of the <see cref="EPublicFoo"/>.
/// </summary>
/// <param name="enumValue">The enumeration value.</param>
/// <returns>An enumerable collection of <see cref="EPublicFoo"/> attributes.</returns>
#if NET8_0_OR_GREATER
public static FrozenDictionary<EPublicFoo, Attribute[]> GetMemberAttributes(this EPublicFoo enumValue)
#else
public static Dictionary<EPublicFoo, Attribute[]> GetMemberAttributes(this EPublicFoo enumValue)
#endif
=> memberAttributes;
}
The generated extension files are available in your IDE under the Source Generators files.
Contributing
Create an issue if you find a BUG or have a Suggestion or Question. If you want to develop this project :
- Fork it!
- Create your feature branch:
git checkout -b my-new-feature
- Commit your changes:
git commit -am 'Add some feature'
- Push to the branch:
git push origin my-new-feature
- Submit a pull request
License
FusionReactor.SourceGenerators.EnumExtensions is Copyright © 2024 OhFlowi under the MIT License.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.