I've built a method that returns me an instance of attribute if it is found on the property:
public static U GetPropertyAttribute<T, TProperty, U>(this T instance, Expression<Func<T, TProperty>> propertySelector, U attribute) where U : Attribute
{
   return Attribute.GetCustomAttribute(instance.GetType().GetProperty((propertySelector.Body as MemberExpression).Member.Name), typeof(U), true) as U;
}
In order to get instance, I have to invoke:
var cc = new CustomClass();
cc.GetPropertyAttribute(x => x.Name, new NullableAttribute())
And it works fine, I get exact instance of the attribute class.
However, I don't like that I have to use new NullableAttribute() as parameter, I'd like to have invoke look like:
cc.GetPropertyAttribute<NullableAttribute>(x => x.Name)
This however does not work due to the fact that as soon as I remove second parameter and add 1 generic type in method name, it starts to require other two as well. Is there a way to force the method to infer 2 out of 3 generic parameters? I.e. I want to specify attribute class, but don't want to specify class name and property name.
UPDATE:
Here's the implemented code, thanks to Jon, as well as the code for string solution. I've decided to nest the class so that I don't polute the namespace if I introduce same approach for some other extension classes.
public static class AttributeExtensions
{
    public static ObjectProperty<T, TProperty> From<T, TProperty>(this T instance, Expression<Func<T, TProperty>> propertySelector)
    {
        return new ObjectProperty<T, TProperty>(instance, propertySelector);
    }
    public class ObjectProperty<T, TProperty>
    {
        private readonly T instance;
        private readonly Expression<Func<T, TProperty>> propertySelector;
        public ObjectProperty(T instance, Expression<Func<T, TProperty>> propertySelector)
        {
            this.instance = instance;
            this.propertySelector = propertySelector;
        }
        public U GetAttribute<U>() where U : Attribute
        {
            return Attribute.GetCustomAttribute(instance.GetType().GetProperty((propertySelector.Body as MemberExpression).Member.Name), typeof(U), true) as U;
        }
    }
    public static T GetPropertyAttribute<T>(this object instance, string propertyName) where T : Attribute
    {
        return Attribute.GetCustomAttribute(instance.GetType().GetProperty(propertyName), typeof(T), true) as T;
    }
}
So invoke now goes like this:
var cc = new CustomClass();
var attr = cc.From(x => x.Name).GetAttribute<NullableAttribute>();
Is there a way to force the method to infer 2 out of 3 generic parameters?
One common approach is to have an intermediate type with those two type parameters, and then have a generic method within that type to allow you to supply the last one:
public static AttributeFetcher<T, TProperty> FetchFrom<T, TProperty>
    (this T instance, Expression<Func<T, TProperty>> propertySelector)
{
    return new AttributeFetcher<T, TProperty>(instance, propertySelector);     
}
public class AttributeFetcher<T, TProperty>
{
    private readonly T instance;
    private readonly Expression<Func<T, TProperty>> propertySelector;
    // Add obvious constructor
    public U Attribute<U>() where U : Attribute
    {
        // Code as before
    }
}
Then you can write:
cc.FetchFrom(x => x.Name).Attribute<NullableAttribte>();
It's possibly that you can actually make AttributeFetcher non-generic, given that you really only need the PropertyInfo as far as I can tell. The code above is for the more general case.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With