ToolBX.Reflection4Humans.Extensions 3.0.0

dotnet add package ToolBX.Reflection4Humans.Extensions --version 3.0.0
                    
NuGet\Install-Package ToolBX.Reflection4Humans.Extensions -Version 3.0.0
                    
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="ToolBX.Reflection4Humans.Extensions" Version="3.0.0" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="ToolBX.Reflection4Humans.Extensions" Version="3.0.0" />
                    
Directory.Packages.props
<PackageReference Include="ToolBX.Reflection4Humans.Extensions" />
                    
Project file
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add ToolBX.Reflection4Humans.Extensions --version 3.0.0
                    
#r "nuget: ToolBX.Reflection4Humans.Extensions, 3.0.0"
                    
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#:package ToolBX.Reflection4Humans.Extensions@3.0.0
                    
#:package directive can be used in C# file-based apps starting in .NET 10 preview 4. Copy this into a .cs file before any lines of code to reference the package.
#addin nuget:?package=ToolBX.Reflection4Humans.Extensions&version=3.0.0
                    
Install as a Cake Addin
#tool nuget:?package=ToolBX.Reflection4Humans.Extensions&version=3.0.0
                    
Install as a Cake Tool

Reflection4Humans

Reflection4Humans.Extensions

Reflection extension methods meant to be used by humans.

Member search extensions

GetAll[MEMBER]

  • Type.GetAllMembers
  • Type.GetAllProperties
  • Type.GetAllFields
  • Type.GetAllPropertiesOrFields
  • Type.GetAllMethods
  • Type.GetAllConstructors
  • Type GetAllEvents

The above extension methods will return all members of said kind on any .NET type. There is no confusing flag for which you need to specify whether or not you want private, protected or static members. It's all done via predicates.

//Returns all members including those on inherited types
var allMembers = typeof(Dummy).GetAllMembers();

//Returns all private members that start with "Johnny"
var members = typeof(Dummy).GetAllMembers(x => x.Name.StartsWith("Johnny") && x.IsPrivate());

//Returns all get-only static properties
var properties = typeof(Dummy).GetAllProperties(x => x.IsGet() && !x.IsSet() && x.IsStatic());

//Returns public fields only
var fields = typeof(Dummy).GetAllFields(x => x.IsPublic());

//Returns methods that have parameters of string, char and int (in that order)
var methods = typeof(Dummy).GetAllMethods(x => x.HasParameters<string, char, int>());

//You can use some of the same approach for constructors as you would regular methods
var constructors = typeof(Dummy).GetAllConstructors(x => x.HasParameters<int>() && !x.IsStatic());

GetSingle[MEMBER] & GetSingle[MEMBER]OrDefault

  • Type.GetSingleMember & Type.GetSingleMemberOrDefault
  • Type.GetSingleProperty & Type.GetSinglePropertyOrDefault
  • Type.GetSingleField & Type.GetSingleFieldOrDefault
  • Type.GetSinglePropertyOrField & Type.GetSinglePropertyOrFieldOrDefault
  • Type.GetSingleMethod & Type.GetSingleMethodOrDefault
  • Type.GetSingleConstructor & Type.GetSingleConstructorOrDefault
  • Type.GetSingleEvent & Type.GetSingleEventOrDefault

As you probably expect, the overload without "Default" in it will throw an exception if there is no member found. That is the only difference between the two. They are used the exact same way as the above "GetAll" extensions. There are additional overloads that do a simple search by name but they will throw an exception if there is more than one result to your query so make sure that there is indeed only one member by that name.

//Will return a single get-only property from the Dummy type or will throw if there is more than one public get-only property
var property = typeof(Dummy).GetSingleProperty(x => x.IsPublic() && !x.IsSet())

//It may be better to go by name in the case of properties if you can
var propertyByName = typeof(Dummy).GetSingleProperty("Level");

//But in the case of methods, multiple ones can share the same name and even some parameters. It can be difficult to find the right one in some cases
var method = typeof(Dummy).GetSingleMethodOrDefault(x => x.Name == "LevelUp" && x.HasParameters<int, string>() && x.IsGeneric());

Has[MEMBER] methods

  • Type.HasMember
  • Type.HasField
  • Type.HasProperty
  • Type.HasMethod
  • Type.HasConstructor
  • Type.HasEvent

Type.GetHumanReadableName

Have you ever wanted to output a type name in an exception message before only for it to read like this?

"Reflection4Humans.Extensions.Tests.Dummy`1[System.String] did something weird!"

Wouldn't it have been easier on the eyes if it had said this instead?

"Dummy<String> did something weird!"

This is what this method helps you achieve.

//Instead of this
typeof(Dummy<string>).Name

//Use this
typeof(Dummy<string>).GetHumanReadableName();

Type.IsAttribute

//Isn't it weird that you have the following in cs
var isInterface = type.IsInterface;
var isAbstract = type.IsAbstract;
var isClass = type.IsClass;
var isGenericType = type.IsGenericType;

//But not this?
var isAttribute = type.IsAttribute();

Well now you have it.

Type.GetPropertyPath

//Say you have a multi-level object like this
public record Person
{
	public string Name { get; init; }
	public Job Job { get; init; }
}

//Maybe you would like to have code looking at the Salary from a Person object
public record Job
{
	public string Title { get; init; }
	public decimal Salary { get; init; }
}

public void DoStuff()
{
	//The propertyPath variable now holds the PropertyInfo objects for Person's Job property as well as Job's Salary property, 
	//including the types for Person and Job. Essentially looking something like this : 
	//{ { Property = [Job PropertyInfo], Owner = [Person type object] }, { Property = [Salary PropertyInfo], Owner = [Job type object] } }
	var propertyPath = typeof(Person).GetPropertyPath("Job.Salary");

	...
}

Type.IsAttribute

True if the the type is an attribute.

if (type.IsAttribute())
{
	...
}

Type.GetDirectInterfaces

Returns all interfaces that the type implements. In other words, it excludes those that are implemented "indirectly" via a base class.

var directInterfaces = type.GetDirectInterfaces();

Type.Implements

Provides an easy way to check whether or not a type implements an interface (directly OR indirectly).

if (type.Implements<IDisposable>)
{
	...
}

//Also comes with a non-generic overload
if (type.Implements(typeof(IDisposable))
{
	...
}

Type.DirectlyImplements

Returns true when a type directly implements an interface (as opposed to the "regular" Implements method). It is also used the same as the Implements method.

if (type.DirectlyImplements<IDisposable>)
{
	...
}

//Also comes with a non-generic overload
if (type.DirectlyImplements(typeof(IDisposable))
{
	...
}

Type.HasInterface

True if the type implements at least one interface.

if (type.HasInterface())
{
	...
}

Type.HasAttribute

  • HasAttribute()
//True if the type has at least one attribute of any type
if (type.HasAttribute())
{
	...
}
  • HasAttribute<TAttribute>()
//Is true if the type has an attribute of type AutoInjectAttribute regardless of the attribute's properties
if (type.HasAttribute<AutoInjectAttribute>())
{
	...
}

//Can also be used to check the attribute's properties as well
if (type.HasAttribute<AutoInjectAttribute>(x => x.Interface == typeof(IDummy)))
{
	...
}

//There is also a non-generic overload but it can't be used to check the attributes' properties
if (type.HasAttribute(typeof(AutoInjectAttribute))
{
	...
}

MemberInfo extensions

  • IsStatic
  • IsInstance
  • IsPrivate
  • IsProptected
  • IsInternal
  • IsPublic
  • IsConstructor
  • IsMethod
  • IsField
  • IsProperty

These are meant to make it easier to search MemberInfo but could be used in other scenarios as well.

MethodInfo & MethodBase extensions

  • HasParameters
//Returns true if the method has a signature of string, char and long (in this precise order)
if (method.HasParameters<string, char, long>()) 
{
	...
}

//It also comes with a non-generic overload
if (method.HasParameters(typeof(string), typeof(char), typeof(long))
{
	...
}
  • HasParameters(count)
//True if the constructor has exactly two parameters
if (constructor.HasParameters(2))
{
	...
}
  • HasNoParameter
//True if the method has no parameter
if (method.HasNoParameter())
{
	...
}

PropertyInfo

  • IsStatic
//Normally, you would need to access the get or set method to check whether or not the property is static. Now you can check out the property directly.
if (property.IsStatic())
{
	...
}
  • IsIndexer
//Returns true if the property is an indexer (aka this[int index])
if (property.IsIndexer())
{
	...
}

PropertyOrField

Encompasses both fields and properties as a single object. It is used for those times where you just don't care about whether or not you want fields or properties. You can limit your search to public or internal members to avoid getting properties and their own backing fields.

PropertOrField member = instance.GetType().GetSinglePropertyOrField(x => ...);
//This does what you expect it to do... except that it will throw if your member is a write-only property (aka has no get; accessor)
var result = member.GetValue(instance);

//This one doesn't throw and instead encapsulates your value in a Result<T> (provided your member is a field or a property with a get accessor)
var result = member.TryGetValue(instance);

//The same rules apply to SetValue. This will throw if member is a property without a set or init accessor
member.SetValue(instance, value);

//And this will avoid throwing an exception if it's a read-only property
member.TrySetValue(instance, value);

It's also possible to convert a collection of members into a collection of IPropertyOrField.

//Will throw an exception if there are other types of members in your collection, however
var result = members.AsPropertyOrField();

//Will filter out anything that is not a property or a field to avoid throwing exceptions
var result = members.TryAsPropertyOrField();

It's important to note that by default, the GetAllPropertiesOrFields method (without a predicate) will return all properties and fields on a given type regardless of its accessibility or wheter it's a backing field (automatically-generated or otherwise).

//Ensures that you're not getting anything in double with a property and its backing field for instance
var result = typeof(MyType).GetAllPropertiesOrFields(x => !x.IsBackingField())

IsBackingField & IsAutomaticBackingField

These two extension methods to FieldInfo come in handy when you want to include or exclude backing fields. If you didn't know, the following property will generate a backing field by default :

public string SomeProperty { get; set; }

Its name will look something like <SomeProperty>k__BackingField. These are referred to by Reflection4Humans as Automatic backing fields. If you made your own backing field following the commonly-accepted "official" C# standards, it would probably be something like _someProperty. That's not exclusive to backing fields however as that's just how we normally write private field names in C#.

Now, the IsAutomaticBackingField is a lot more accurate in acertaining that the field is indeed a backing field because you can't (normally) use "<>" in variable or member names in C#.

Your own backing fields are a lot more difficult to identify since they don't even have as much as an attribute to them. They're likely named just like any other field. By default, the IsBackingField method will look for fields that have properties with similar names prefixed with a _. It works for most cases. There's also a way to add your own BackingFieldConvention by passing it to the method or by adding it to a global config for Reflection4Humans.

BackingFieldConvention

By default, the ReflectionConfig object only contains BackingFieldConvention.Csharp as its naming convention.

//This is only meant to be an example on how to add a new naming convention
ReflectionConfig.Add(new BackingFieldConvention { Prefix = "m_", Suffix = "_backingField" })

Only use the above method if you have global naming conventions that you want to apply to all your project.

You can just use this if you want to do it per-call :

var result = typeof(MyType).GetAllPropertiesOrFields(x => !x.IsBackingField(new BackingFieldConvention { Prefix = "m_", Suffix = "_backingField" }))
Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  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.  net10.0 was computed.  net10.0-android was computed.  net10.0-browser was computed.  net10.0-ios was computed.  net10.0-maccatalyst was computed.  net10.0-macos was computed.  net10.0-tvos was computed.  net10.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (10)

Showing the top 5 NuGet packages that depend on ToolBX.Reflection4Humans.Extensions:

Package Downloads
ToolBX.Reflection4Humans.TypeFetcher

Provides a more straightforward way for a human being to get types from all loaded assemblies using the TypeFetcher class.

ToolBX.Reflection4Humans.ValueEquality

Adds extensions to automatically compare objects by value.

ToolBX.EasyTypeParsing

Extension methods to make it easier to parse strings into other types.

ToolBX.Collections.ReadOnly

Actual read-only collections with helper extension methods to make them less painful to use.

ToolBX.Collections.ObservableList

A modern re-implementation of a dynamic one-dimensional array.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last Updated
3.0.0 8,381 9/26/2024
3.0.0-beta5 684 9/22/2024
3.0.0-beta4 480 9/22/2024
3.0.0-beta3 188 9/22/2024
3.0.0-beta2 285 9/7/2024
3.0.0-beta1 343 5/31/2024
2.2.0 11,643 1/11/2024
2.2.0-beta9 1,146 1/7/2024
2.2.0-beta8 410 12/29/2023
2.2.0-beta7 484 12/14/2023
2.2.0-beta6 443 12/14/2023
2.2.0-beta4 661 11/16/2023
2.2.0-beta3 176 11/15/2023
2.2.0-beta2 228 8/16/2023
2.2.0-beta1 631 7/24/2023
2.1.0-beta1 217 6/28/2023
2.0.3 13,750 6/19/2023
2.0.2 2,820 4/23/2023
2.0.1 320 2/17/2023
2.0.1-beta2 220 2/16/2023
2.0.1-beta1 236 2/14/2023
2.0.0 1,868 11/9/2022
2.0.0-beta1 749 9/20/2022
1.0.0 13,305 12/24/2021