使用数据库优先方法的 WPF 中的 DataAnnotation - 如何将数据注释移动到伙伴 class,包括。 IsValid 函数
DataAnnotation in WPF with Database first approach - how to move data annotations to buddy class incl. IsValid function
正如主题中的简短描述:如何将所有 DataAnnotations 从模型移动到 MetaData 模型,以免在更新 edmx 时将其清除?
换句话说,我希望数据注释是安全的,而不是随着 edmx 的每次更新而被删除,我会在 dataannotation 中有一个选项来检查是否满足所有数据注释要求(IsValid 方法)以在 RelayCommand 的 CanExecute 方法中使用它.
我有一个class如下:
public partial class Customer : IDataErrorInfo
{
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2214:DoNotCallOverridableMethodsInConstructors")]
public int ID{ get; set; }
[Required(ErrorMessage = "Field required")]
public string Name{ get; set; }
[Required(ErrorMessage = "Field required")]
public string LastName{ get; set; }
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2227:CollectionPropertiesShouldBeReadOnly")]
public virtual ICollection<tblKontrahent> tblKontrahent { get; set; }
#region Validation
public bool IsValid { get; set; }
public string Error { get { return null; } }
public string this[string columnName]
{
get
{
Validation();
return InputValidation<Customer >.Validate(this, columnName);
}
}
public ICollection<string> AllErrors()
{
return InputValidation<Customer >.Validate(this);
}
private void Validation()
{
ICollection<string> allErrors = AllErrors();
if (allErrors.Count == 0)
IsValid = true;
else
IsValid = false;
}
#endregion
#region Shallow copy
public Customer ShallowCopy()
{
return (Customer )this.MemberwiseClone();
}
#endregion
}
如何使用注释和 IsValid 函数将其从模型移动到元数据模型。如果能把ShallowCopy方法也搬过来就好了
非常感谢您的任何建议!
对于大多数重要的应用程序,我将 EF classes 完全分开。我将属性从 entity framework 复制到一个自跟踪视图模型。
对于较小的应用程序,我过去常常避免这样做。
您可以看到其中使用的一种方法:
https://gallery.technet.microsoft.com/scriptcenter/WPF-Entity-Framework-MVVM-78cdc204
它使用 INotifyDataErrorInfo,您会在 BaseEntity 中找到 IsValid。这是一个相当复杂的 class 但是 re-usable.
您或许可以将 shallowcopy 重构为 BaseEntity。如果您可以在任何地方投射,那么很容易。
注释在单独的伙伴 classes 中。您可以在 Customer.metadata.cs 和 Product.metadata.cs 中查看示例。这些是部分 classes,它们将 BaseEntity 的继承添加到实体 classes。因此 EF class Customer 继承了 BaseEntity。
一个例子:
using DataAnnotationsExtensions;
namespace wpf_EntityFramework.EntityData
{
[MetadataTypeAttribute(typeof(Product.ProductMetadata))]
public partial class Product : BaseEntity, IEntityWithId
{
public void MetaSetUp()
{
// In wpf you need to explicitly state the metadata file.
// Maybe this will be improved in future versions of EF.
TypeDescriptor.AddProviderTransparent(
new AssociatedMetadataTypeTypeDescriptionProvider(typeof(Product),
typeof(ProductMetadata)),
typeof(Product));
}
internal sealed class ProductMetadata
{
// Some of these datannotations rely on dataAnnotationsExtensions ( Nuget package )
[Required(ErrorMessage="Product Short Name is required")]
public string ProductShortName { get; set; }
[Required(ErrorMessage = "Product Weight is required")]
[Min(0.01, ErrorMessage = "Minimum weight is 0.01")]
[Max(70.00, ErrorMessage = "We don't sell anything weighing more than 70Kg")]
public Nullable<decimal> Weight { get; set; }
[Required(ErrorMessage = "Bar Code is required")]
[RegularExpression(@"[0-9]{11}$", ErrorMessage="Bar codes must be 11 digits")]
public string BarCode { get; set; }
[Required(ErrorMessage = "Price per product is required")]
[Range(0,200, ErrorMessage="Price must be 0 - £200") ]
public Nullable<decimal> PricePer { get; set; }
private ProductMetadata()
{ }
}
}
}
正如评论中所说。
您需要在每个实例上调用该 Metasetup。除非最近几年发生了一些变化。伙伴 class 并不像使用 MVC 那样被拾起。
该示例还反馈来自 UI 的转换失败。
参见 Dictionary1 中的模板。
<ControlTemplate x:Key="EditPopUp" TargetType="ContentControl">
<ControlTemplate.Resources>
<Style TargetType="{x:Type TextBox}" BasedOn="{StaticResource ErrorToolTip}">
<Setter Property="HorizontalAlignment" Value="Stretch"/>
</Style>
</ControlTemplate.Resources>
<Grid Visibility="{Binding IsInEditMode, Converter={StaticResource BooleanToVisibilityConverter}}"
Width="{Binding ElementName=dg, Path=ActualWidth}"
Height="{Binding ElementName=dg, Path=ActualHeight}"
>
<i:Interaction.Triggers>
<local:RoutedEventTrigger RoutedEvent="{x:Static Validation.ErrorEvent}">
<e2c:EventToCommand
Command="{Binding EditVM.TheEntity.ConversionErrorCommand, Mode=OneWay}"
EventArgsConverter="{StaticResource BindingErrorEventArgsConverter}"
PassEventArgsToCommand="True" />
</local:RoutedEventTrigger>
<local:RoutedEventTrigger RoutedEvent="{x:Static Binding.SourceUpdatedEvent}">
<e2c:EventToCommand
Command="{Binding EditVM.TheEntity.SourceUpdatedCommand, Mode=OneWay}"
EventArgsConverter="{StaticResource BindingSourcePropertyConverter}"
PassEventArgsToCommand="True" />
</local:RoutedEventTrigger>
</i:Interaction.Triggers>
正如主题中的简短描述:如何将所有 DataAnnotations 从模型移动到 MetaData 模型,以免在更新 edmx 时将其清除?
换句话说,我希望数据注释是安全的,而不是随着 edmx 的每次更新而被删除,我会在 dataannotation 中有一个选项来检查是否满足所有数据注释要求(IsValid 方法)以在 RelayCommand 的 CanExecute 方法中使用它.
我有一个class如下:
public partial class Customer : IDataErrorInfo
{
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2214:DoNotCallOverridableMethodsInConstructors")]
public int ID{ get; set; }
[Required(ErrorMessage = "Field required")]
public string Name{ get; set; }
[Required(ErrorMessage = "Field required")]
public string LastName{ get; set; }
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2227:CollectionPropertiesShouldBeReadOnly")]
public virtual ICollection<tblKontrahent> tblKontrahent { get; set; }
#region Validation
public bool IsValid { get; set; }
public string Error { get { return null; } }
public string this[string columnName]
{
get
{
Validation();
return InputValidation<Customer >.Validate(this, columnName);
}
}
public ICollection<string> AllErrors()
{
return InputValidation<Customer >.Validate(this);
}
private void Validation()
{
ICollection<string> allErrors = AllErrors();
if (allErrors.Count == 0)
IsValid = true;
else
IsValid = false;
}
#endregion
#region Shallow copy
public Customer ShallowCopy()
{
return (Customer )this.MemberwiseClone();
}
#endregion
}
如何使用注释和 IsValid 函数将其从模型移动到元数据模型。如果能把ShallowCopy方法也搬过来就好了
非常感谢您的任何建议!
对于大多数重要的应用程序,我将 EF classes 完全分开。我将属性从 entity framework 复制到一个自跟踪视图模型。
对于较小的应用程序,我过去常常避免这样做。
您可以看到其中使用的一种方法:
https://gallery.technet.microsoft.com/scriptcenter/WPF-Entity-Framework-MVVM-78cdc204
它使用 INotifyDataErrorInfo,您会在 BaseEntity 中找到 IsValid。这是一个相当复杂的 class 但是 re-usable.
您或许可以将 shallowcopy 重构为 BaseEntity。如果您可以在任何地方投射,那么很容易。
注释在单独的伙伴 classes 中。您可以在 Customer.metadata.cs 和 Product.metadata.cs 中查看示例。这些是部分 classes,它们将 BaseEntity 的继承添加到实体 classes。因此 EF class Customer 继承了 BaseEntity。
一个例子:
using DataAnnotationsExtensions;
namespace wpf_EntityFramework.EntityData
{
[MetadataTypeAttribute(typeof(Product.ProductMetadata))]
public partial class Product : BaseEntity, IEntityWithId
{
public void MetaSetUp()
{
// In wpf you need to explicitly state the metadata file.
// Maybe this will be improved in future versions of EF.
TypeDescriptor.AddProviderTransparent(
new AssociatedMetadataTypeTypeDescriptionProvider(typeof(Product),
typeof(ProductMetadata)),
typeof(Product));
}
internal sealed class ProductMetadata
{
// Some of these datannotations rely on dataAnnotationsExtensions ( Nuget package )
[Required(ErrorMessage="Product Short Name is required")]
public string ProductShortName { get; set; }
[Required(ErrorMessage = "Product Weight is required")]
[Min(0.01, ErrorMessage = "Minimum weight is 0.01")]
[Max(70.00, ErrorMessage = "We don't sell anything weighing more than 70Kg")]
public Nullable<decimal> Weight { get; set; }
[Required(ErrorMessage = "Bar Code is required")]
[RegularExpression(@"[0-9]{11}$", ErrorMessage="Bar codes must be 11 digits")]
public string BarCode { get; set; }
[Required(ErrorMessage = "Price per product is required")]
[Range(0,200, ErrorMessage="Price must be 0 - £200") ]
public Nullable<decimal> PricePer { get; set; }
private ProductMetadata()
{ }
}
}
}
正如评论中所说。
您需要在每个实例上调用该 Metasetup。除非最近几年发生了一些变化。伙伴 class 并不像使用 MVC 那样被拾起。
该示例还反馈来自 UI 的转换失败。
参见 Dictionary1 中的模板。
<ControlTemplate x:Key="EditPopUp" TargetType="ContentControl">
<ControlTemplate.Resources>
<Style TargetType="{x:Type TextBox}" BasedOn="{StaticResource ErrorToolTip}">
<Setter Property="HorizontalAlignment" Value="Stretch"/>
</Style>
</ControlTemplate.Resources>
<Grid Visibility="{Binding IsInEditMode, Converter={StaticResource BooleanToVisibilityConverter}}"
Width="{Binding ElementName=dg, Path=ActualWidth}"
Height="{Binding ElementName=dg, Path=ActualHeight}"
>
<i:Interaction.Triggers>
<local:RoutedEventTrigger RoutedEvent="{x:Static Validation.ErrorEvent}">
<e2c:EventToCommand
Command="{Binding EditVM.TheEntity.ConversionErrorCommand, Mode=OneWay}"
EventArgsConverter="{StaticResource BindingErrorEventArgsConverter}"
PassEventArgsToCommand="True" />
</local:RoutedEventTrigger>
<local:RoutedEventTrigger RoutedEvent="{x:Static Binding.SourceUpdatedEvent}">
<e2c:EventToCommand
Command="{Binding EditVM.TheEntity.SourceUpdatedCommand, Mode=OneWay}"
EventArgsConverter="{StaticResource BindingSourcePropertyConverter}"
PassEventArgsToCommand="True" />
</local:RoutedEventTrigger>
</i:Interaction.Triggers>