Search code examples
asp.netasp.net-mvcvalidationasp.net-mvc-2data-annotations

Data Annotations for validation, at least one required field?


If I have a search object with a list of fields, can I, using the System.ComponentModel.DataAnnotations namespace, set it up to validate that at least one of the fields in the search is not null or empty? i.e All the fields are optional but at least one should always be entered.


Solution

  • I'd create a custom validator for this - it won't give you client side validation, just server side.

    Note that for this to work, you'll need to be using nullable types, as value types will default to 0 or false:

    First create a new validator:

    using System.ComponentModel.DataAnnotations;
    using System.Reflection;
    
    // This is a class-level attribute, doesn't make sense at the property level
    [AttributeUsage(AttributeTargets.Class)]
    public class AtLeastOnePropertyAttribute : ValidationAttribute
    {
      // Have to override IsValid
      public override bool IsValid(object value)
      {
        //  Need to use reflection to get properties of "value"...
        var typeInfo = value.GetType();
    
        var propertyInfo = typeInfo.GetProperties();
    
        foreach (var property in propertyInfo)
        {
          if (null != property.GetValue(value, null))
          {
            // We've found a property with a value
            return true;
          }
        }
    
        // All properties were null.
        return false;
      }
    }
    

    You can then decorate your models with this:

    [AtLeastOneProperty(ErrorMessage="You must supply at least one value")]
    public class SimpleTest
    {
        public string StringProp { get; set; }
        public int? Id { get; set; }
        public bool? BoolProp { get; set; }
    }
    

    Then when you call ModelState.IsValid your validator will be called, and your message will be added to the ValidationSummary on your view.

    Note that you could extend this to check for the type of property coming back, or look for attributes on them to include/exclude from validation if you want to - this is assuming a generic validator that doesn't know anything about the type it's validating.