3


2

MetadataTypeの問題

VS2008 SP1、WCF Ria Service、2009年7月CTPを使用しています。 MetadataTypeは部分クラスモードでは機能しないことがわかりました。実際に私が見逃したことはわかりません。

作業:-

public partial class Person
{
    private string _Name;

    [Required(AllowEmptyStrings=false, ErrorMessage="Name required entry")]
    [StringLength(3)]
    public string Name
    {
        set{_Name = value;}
        get{return _Name;}
    }
}

class Program
{
    static void Main(string[] args)
    {
        Person p = new Person { Name="123432" };
        List res = new List();
        Validator.TryValidateObject(p,new ValidationContext(p,null,null),
            res,true);
        if (res.Count > 0)
        {
            Console.WriteLine(res[0].ErrorMessage);
            Console.ReadLine();
        }
    }
}

うまくいかない

public partial class Person
{
    private string _Name;

    public string Name
    {
        set{_Name = value;}
        get{return _Name;}
    }
}

[MetadataType(typeof(PersonMetadata))]
public partial class Person
{
}


public partial class PersonMetadata
{
    [Required(AllowEmptyStrings=false, ErrorMessage="Name required entry")]
    [StringLength(3)]
    public string Name;
}

class Program
{
    static void Main(string[] args)
    {
        Person p = new Person { Name="123432" };
        List res = new List();
        Validator.TryValidateObject(p,new ValidationContext(p,null,null),
            res,true);
        if (res.Count > 0)
        {
            Console.WriteLine(res[0].ErrorMessage);
            Console.ReadLine();
        }
    }
}

3 Answer


9


*編集:*ここで答えが見つかりました:http://forums.silverlight.net/forums/p/149264/377212.aspx

検証する前に、メタデータクラスを手動で登録する必要があります。

TypeDescriptor.AddProviderTransparent(
            new AssociatedMetadataTypeTypeDescriptionProvider(typeof(Person), typeof(PersonMetadata)), typeof(Person));

        List res = new List();
        bool valid = Validator.TryValidateObject(p, new ValidationContext(p, null, null), res, true);

(元の回答が続きます)

問題は、特に部分クラスにあるのではなく、Validator.TryValidateObjectがMetaDataType属性を認識していないようです。 私も同じ問題を抱えています-MVC 2の組み込み検証はメタデータクラスを認識しますが、TryValidateObjectは認識しません。

これらを参照してください:https://stackoverflow.com/questions/2050161/validating-dataannotations-with-validator-class[Validating DataAnnotations with Validator class] https://stackoverflow.com/questions/2422031/validation-does-not-work -when-i-use-validator-tryvalidateobject [Validator.TryValidateObjectを使用すると検証が機能しません]

補足として、必要かどうかはわかりませんが、メタデータクラスで見たすべての例では、各プロパティでデフォルトのget / setを使用しています。

[Required(AllowEmptyStrings=false, ErrorMessage="Name required entry")]
[StringLength(3)]
public string Name { get; set; }


2


上記の答えをくれたJeremy Gruenwaldに感謝します…​ 私はこれに完全にこだわった。

このソリューションに基づいて標準の検証クラスを作成したかったのですが、メタデータクラスタイプを渡す必要はありませんでした。

これを実現するために、カスタム属性をルックアップしてメタデータクラスタイプを取得し、検証結果を返す前にそのクラスを登録する静的クラスを作成しました。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;

namespace MyApp.Validation
{
    public static class EntityValidator
    {
        public static List Validate(object instance, bool validateAllProperties = true)
        {
            RegisterMetadataClass(instance);

            var validationContext = new ValidationContext(instance, null, null);
            var validationResults = new List();

            Validator.TryValidateObject(instance, validationContext, validationResults, validateAllProperties);

            return validationResults;
        }

        private static void RegisterMetadataClass(object instance)
        {
            var modelType = instance.GetType();
            var metadataType = GetMetadataType(modelType);

            if (metadataType != null)
            {
                TypeDescriptor.AddProviderTransparent(new AssociatedMetadataTypeTypeDescriptionProvider(modelType, metadataType), modelType);
            }
        }

        private static Type GetMetadataType(Type type)
        {
            var attribute = (MetadataTypeAttribute)type.GetCustomAttributes(typeof (MetadataTypeAttribute), true).FirstOrDefault();
            return attribute == null ? null : attribute.MetadataClassType;
        }
    }
}

使い方は次のように簡単です:

var errors = EntityValidator.Validate(myEntity);


0


あなたがWPFとEFで作業している場合、これは常に私のために働いています…​

[MetadataType(typeof(Department.Metadata))]
public partial class Department : BaseModel
{
    static Department()
    {
        TypeDescriptor.AddProvider(new AssociatedMetadataTypeTypeDescriptionProvider(typeof(Department),typeof(Metadata)), typeof(Department));
    }
    private sealed class Metadata
    {
        [Required(AllowEmptyStrings = false, ErrorMessage = "Department Name is required.")]
        [StringLength(50, ErrorMessage = "Name must be between 3 and 50 characters.", MinimumLength = 3)]
        public string Name;

        [StringLength(250, ErrorMessage = "Name must be between 10 and 250 characters.", MinimumLength = 10)]
        public string Description;
    }
}

そして、それを実現する基本クラス…​

public abstract class BaseModel : IDataErrorInfo
{
    #region Validation
    string IDataErrorInfo.Error
    {
        get { return null; }
    }
    string IDataErrorInfo.this[string propertyName]
    {
        get
        {
            var propertyInfo = GetType().GetProperty(propertyName);
            var results = new List();
            var result = Validator.TryValidateProperty(propertyInfo.GetValue(this, null), new ValidationContext(this, null, null)
            {
                MemberName = propertyName
            }, results);

            if (result) return string.Empty;
            var validationResult = results.First();
            return validationResult.ErrorMessage;
        }
    }
    #endregion
}