添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

POCO ( Plain Old CLR Objects ) View Models simplify and speed up the development process.

POCO View Models allow you to:

  • Define bindable properties as simple auto-implemented properties.
  • Create methods that function as commands at runtime.
  • Make properties and methods implement MVVM-specific interfaces.
  • This allows you to create clean, simple, maintainable, and testable MVVM code.

    The POCO View Models are fully compatible with any WPF control.

    You can use View Models Generated at Compile Time to generate boilerplate code for your ViewModels at compile time.

    Basics of Generating POCO View Models

    A POCO class does not implement an interface, and does not need to be inherited from a base class, such as ViewModelBase or BindableBase . To transform a POCO class into a fully functional ViewModel, create a class instance with the DevExpress.Mvvm.POCO.ViewModelSource.Create method. See the example below.

    public class LoginViewModel {
        //This property will be converted to a bindable one
        public virtual string UserName { get; set; }
        //SaveAccountSettingsCommand will be created for the SaveAccountSettings and CanSaveAccountSettings methods:
        //SaveAccountSettingsCommand = new DelegateCommand<string>(SaveAccountSettings, CanSaveAccountSettings);
        public void SaveAccountSettings(string fileName) {
            //...
        public bool CanSaveAccountSettings(string fileName) {
            return !string.IsNullOrEmpty(fileName);
        //We recommend that you not use public constructors to prevent creating the View Model without the ViewModelSource
        protected LoginViewModel() { }
        //This is a helper method that uses the ViewModelSource class for creating a LoginViewModel instance
        public static LoginViewModel Create() {
            return ViewModelSource.Create(() => new LoginViewModel());
    

    You can use the ViewModelSource class to create a View Model instance in XAML.

    <UserControl x:Class="DXPOCO.Views.LoginView"
        xmlns:dxmvvm="http://schemas.devexpress.com/winfx/2008/xaml/mvvm"
        xmlns:ViewModels="clr-namespace:DXPOCO.ViewModels"
        DataContext="{dxmvvm:ViewModelSource Type=ViewModels:LoginViewModel}"
            <!--...-->
        </Grid>
    </UserControl>
    

    The ViewModelSource.Create method uses Reflection Emit to create a descendant of the specified ViewModel class and returns the descendant class instance at runtime. The code below is similar to the one that the ViewModelSource generates based on the LoginViewModel class.

    public class LoginViewModel_EXTENSION : LoginViewModel, INotifyPropertyChanged {
        public override string UserName {
            get { return base.UserName; }
            set {
                if(base.UserName == value) return;
                base.UserName = value;
                RaisePropertyChanged("UserName");
        DelegateCommand<string> saveAccountSettingsCommand;
        public DelegateCommand<string> SaveAccountSettingsCommand {
            get {
                return saveAccountSettingsCommand ?? 
                    (saveAccountSettingsCommand = 
                    new DelegateCommand<string>(SaveAccountSettings, CanSaveAccountSettings));
        //INotifyPropertyChanged Implementation
    

    Pass Parameters to the ViewModel Constructor

    You can pass parameters to the ViewModel’s constructor using any of the following approaches.

  • Use lambda expressions. Lambda expressions work slower, because they are not cached and are newly compiled with each method call.

    ViewModelSource.Create(() => new LoginViewModel(caption: "Login") {
        UserName = "John Smith"
    
  • Use delegates. This technique works faster than lambda expressions, because the compiled delegate instances can be cached. This is the quickest technique to pass parameters to the ViewModel constructor.

    var factory = ViewModelSource.Factory((string caption) => new LoginViewModel(caption));
    factory("Login");
    
    <UserControl x:Class="Example.View.LoginView"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:ViewModel="clr-namespace:Example.ViewModel"
        xmlns:dxmvvm="http://schemas.devexpress.com/winfx/2008/xaml/mvvm"
        xmlns:dx="http://schemas.devexpress.com/winfx/2008/xaml/core"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        mc:Ignorable="d" d:DesignHeight="500" d:DesignWidth="600"
        DataContext="{dxmvvm:ViewModelSource Type=ViewModel:LoginViewModel}">
        <dxmvvm:Interaction.Behaviors>
            <dx:DXMessageBoxService/>
        </dxmvvm:Interaction.Behaviors>
        <Grid x:Name="LayoutRoot" Background="White">
            <StackPanel Orientation="Vertical">
                <StackPanel Orientation="Horizontal" Margin="10">
                    <TextBlock Text="UserName: " Margin="3" VerticalAlignment="Center"/>
                    <TextBox Text="{Binding UserName, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" Margin="3" Width="80"/>
                </StackPanel>
                <Button Content="Login" Command="{Binding LoginCommand}" Margin="5"
                        HorizontalAlignment="Left"/>
            </StackPanel>
        </Grid>
    </UserControl>
        public class LoginViewModel {
            public static LoginViewModel Create() {
                return ViewModelSource.Create(() => new LoginViewModel());
            protected LoginViewModel() { }
            public virtual string UserName { get; set; }
            public void Login() {
                this.GetService<IMessageBoxService>().Show("Login succeeded", "Login", MessageButton.OK, MessageIcon.Information, MessageResult.OK);
            public bool CanLogin() {
                return !string.IsNullOrEmpty(UserName);
    
    Imports DevExpress.Mvvm
    Imports DevExpress.Mvvm.DataAnnotations
    Imports DevExpress.Mvvm.POCO
    Namespace Example.ViewModel
        <POCOViewModel> _
        Public Class LoginViewModel
            Public Shared Function Create() As LoginViewModel
                Return ViewModelSource.Create(Function() New LoginViewModel())
            End Function
            Protected Sub New()
            End Sub
            Public Overridable Property UserName() As String
            Public Sub Login()
                Me.GetService(Of IMessageBoxService)().Show("Login succeeded", "Login", MessageButton.OK, MessageIcon.Information, MessageResult.OK)
            End Sub
            Public Function CanLogin() As Boolean
                Return Not String.IsNullOrEmpty(UserName)
            End Function
        End Class
    End Namespace
    

    Bindable Properties

    The POCO mechanism generates bindable properties for properties that meet all the following requirements:

  • The property is public and auto-implemented.

  • The property has the virtual (C#) or Overridable (VB) modifier.

  • The property has a public getter, and a protected or public setter.

    If the property has no setter, you can use the RaisePropertyChanged extension method to explicitly raise the PropertyChanged event for this property. Refer to the example below for more information on how to use the RaisePropertyChanged extension method to explicitly raise the PropertyChanged event:

    View Example

    You can define methods that are invoked when properties are changed. These method names should use the following formats: On[PropertyName]Changed and On[PropertyName]Changing.

    public class LoginViewModel {
            public virtual string UserName { get; set; }
            protected void OnUserNameChanged() {
                //...
        public class LoginViewModel {
            public virtual string UserName { get; set; }
            protected void OnUserNameChanged(string oldValue) {
                //...
            protected void OnUserNameChanging(string newValue) {
                //...
    

    You can use the BindableProperty attribute to:

  • prevent the POCO mechanism from generating a bindable property for a specified property;
  • specify which method should be invoked when a property value is changing or has been changed. This is useful when the method’s name does not match the On[PropertyName]Changed and On[PropertyName]Changing convention.
  • public class LoginViewModel {
            [BindableProperty(isBindable: false)]
            public virtual bool IsEnabled { get; set; }
            [BindableProperty(OnPropertyChangedMethodName = "Update")]
            public virtual string UserName { get; set; }
            protected void Update() {
                //...
    

    You can use the Fluent API to control POCO ViewModel generation.

    [MetadataType(typeof(Metadata))]
    public class LoginViewModel {
        public class Metadata : IMetadataProvider<LoginViewModel> {
            void IMetadataProvider<LoginViewModel>.BuildMetadata
                (MetadataBuilder<LoginViewModel> builder) {
                builder.Property(x => x.UserName).
                    OnPropertyChangedCall(x => x.Update());
                builder.Property(x => x.IsEnabled).
                    DoNotMakeBindable();
        public virtual bool IsEnabled { get; set; }
        public virtual string UserName { get; set; }
        protected void Update() {
            //...
    

    Commands

    The POCO mechanism generates commands for all public methods that have no parameters or a single parameter. A generated command’s name follows the [MethodName]Command pattern. You can use the Command attribute or the Fluent API to control the command generation mechanism.

    public class LoginViewModel {
        [Command(isCommand: false)]
        public void SaveCore() {
            //...
        [Command(CanExecuteMethodName = "CanSaveAccountSettings",
            Name = "SaveCommand",
            UseCommandManager = true)]
        public void SaveAccountSettings(string fileName) {
            //...
        public bool CanSaveAccountSettings(string fileName) {
            return !string.IsNullOrEmpty(fileName);
    [MetadataType(typeof(Metadata))]
    public class LoginViewModel {
        public class Metadata : IMetadataProvider<LoginViewModel> {
            void IMetadataProvider<LoginViewModel>.BuildMetadata(MetadataBuilder<LoginViewModel> builder) {
                builder.CommandFromMethod(x => x.SaveCore()).
                    DoNotCreateCommand();
                builder.CommandFromMethod(x => x.SaveAccountSettings(default(string))).
                    CanExecuteMethod(x => x.CanSaveAccountSettings(default(string))).
                    CommandName("SaveCommand");
        public void SaveCore() {
            //...
        public void SaveAccountSettings(string fileName) {
            //...
        public bool CanSaveAccountSettings(string fileName) {
            return !string.IsNullOrEmpty(fileName);
    

    To update an automatically generated command in a POCO View Model, use the RaiseCanExecuteChanged extension method available from the DevExpress.Mvvm.POCO.POCOViewModelExtensions class.

    [POCOViewModel]
    public class ViewModel {
        public void GoBack(){
            //...
        public bool CanGoBack(){
            //...
        public void UpdateSaveCommand(){
            this.RaiseCanExecuteChanged(c => c.GoBack());
    

    Refer to the following topic for more information: Commands.

    Services

    The DevExpress MVVM Framework includes the Services mechanism. The code sample below demonstrates how to access the Message Box service.

    using DevExpress.Mvvm.POCO;
    public class LoginViewModel {
        public IMessageBoxService MessageBoxService { get { return this.GetService<IMessageBoxService>(); } }
    

    Review the following topic for more information about how to access services: Services in POCO objects.

    Dependency Injection

    To bind a view to a view model, create a MarkupExtension that resolves the correct ViewModel type:

    public class DISource : MarkupExtension {
        public static Func<Type, object, string, object> Resolver { get; set; }
        public Type Type { get; set; }
        public object Key { get; set; }
        public string Name { get; set; }
        public override object ProvideValue(IServiceProvider serviceProvider) => Resolver?.Invoke(Type, Key, Name);
    

    Register the resolver at the application startup:

    protected override void OnStartup(StartupEventArgs e) {
        base.OnStartup(e);
        DISource.Resolver = Resolve;
    object Resolve(Type type, object key, string name) {
        if(type == null)
            return null;
        if(key != null)
            return Container.ResolveKeyed(key, type);
        if(name != null)
            return Container.ResolveNamed(name, type);
        return Container.Resolve(type);
    

    Specify the DataContext in XAML in the following manner:

    DataContext="{common:DISource Type=common:MainViewModel}"
    

    To use a POCO View Model in a Dependency Injection container, utilize the ViewModelSource.GetPOCOType method to register the POCO type generated at runtime:

    container.RegisterType(typeof(IMainViewModel),
                        ViewModelSource.GetPOCOType(typeof(MainViewModel)));
    

    View Example

    View Model Parent-Child Relationships

    POCO View Models can relate to each other with the parent-child relationship. This is achieved with the ISupportParentViewModel interface that is automatically implemented when you create a POCO object with the ViewModelSource class. With this interface, child View Models may access Services registered in the main View Model. The following topic contains more information on how to set the parent-child relationship and its advantages: ViewModel relationships (ISupportParentViewModel).

    Automatic IDataErrorInfo Implementation

    The IDataErrorInfo interface is the standard mechanism for data validation in WPF. You can use this interface to define validation rules for each individual property or for the entire object. The POCO mechanism allows you to automatically implement the IDataErrorInfo interface based on defined attributes or Fluent API.

    To enable this feature, apply the POCOViewModel attribute for your View Model and set the POCOViewModel.ImplementIDataErrorInfo parameter to True.

    //Attribute-based approach
    [POCOViewModel(ImplementIDataErrorInfo = true)] 
    public class LoginViewModel { 
        [Required(ErrorMessage = "Please enter the user name.")] 
        public virtual string UserName { get; set; }
    //Fluent API
    [POCOViewModel(ImplementIDataErrorInfo = true)]
    [MetadataType(typeof(LoginViewModel.Metadata))]
    public class LoginViewModel {
       public class Metadata : IMetadataProvider<LoginViewModel> {
           void IMetadataProvider<LoginViewModel>.BuildMetadata(MetadataBuilder<LoginViewModel> builder) {
               builder.Property(x => x.UserName).
                   Required(() => "Please enter the user name.");
        public virtual string UserName { get; set; }
    

    When the ViewModelSource generates a descendant of a View Model, it implements the IDataErrorInfo interface as follows:

    public class LoginViewModel : IDataErrorInfo { 
        string IDataErrorInfo.Error { 
            get { return string.Empty; } 
        string IDataErrorInfo.this[string columnName] { 
            get { return IDataErrorInfoHelper.GetErrorText(this, columnName); } 
    

    The IDataErrorInfoHelper class allows you to get an error based on specified DataAnnotation attributes or Fluent API.

    The code example below demonstrates how to use the POCO mechanism to implement the IDataErrorInfo interface.

    View Example

    MainView.xaml
    <UserControl x:Class="Example.View.MainView"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:dx="http://schemas.devexpress.com/winfx/2008/xaml/core"
        xmlns:dxe="http://schemas.devexpress.com/winfx/2008/xaml/editors"
        xmlns:dxmvvm="http://schemas.devexpress.com/winfx/2008/xaml/mvvm"
        xmlns:ViewModel="clr-namespace:Example.ViewModel"
        mc:Ignorable="d" d:DesignHeight="400" d:DesignWidth="400"
        DataContext="{dxmvvm:ViewModelSource Type=ViewModel:MainViewModel}">
        <UserControl.Resources>
            <dxmvvm:BooleanNegationConverter x:Key="BooleanNegationConverter"/>
        </UserControl.Resources>
            <StackPanel Orientation="Vertical" Margin="10" dxe:ValidationService.IsValidationContainer="True" x:Name="validationContainer">
                    <Grid.ColumnDefinitions>
                        <ColumnDefinition Width="*"/>
                        <ColumnDefinition Width="*"/>
                    </Grid.ColumnDefinitions>
                    <StackPanel Orientation="Vertical" Margin="0,0,4,6">
                        <TextBlock Text="Name" Margin="6,2,0,2"/>
                        <dxe:TextEdit NullText="First" EditValue="{Binding FirstName, ValidatesOnDataErrors=True, UpdateSourceTrigger=PropertyChanged}"/>
                    </StackPanel>
                    <StackPanel Orientation="Vertical" Margin="4,0,0,6" Grid.Column="1">
                        <TextBlock Text=" " Margin="6,2,0,2"/>
                        <dxe:TextEdit NullText="Last" EditValue="{Binding LastName, ValidatesOnDataErrors=True, UpdateSourceTrigger=PropertyChanged}"/>
                    </StackPanel>
                </Grid>
                <StackPanel Orientation="Vertical" Margin="0,0,0,6">
                    <TextBlock Text="Email" Margin="6,2,0,2"/>
                    <dxe:TextEdit EditValue="{Binding Email, ValidatesOnDataErrors=True, UpdateSourceTrigger=PropertyChanged}"/>
                </StackPanel>
                <StackPanel Orientation="Vertical" Margin="0,0,0,6">
                    <TextBlock Text="Password" Margin="6,2,0,2"/>
                    <dxe:PasswordBoxEdit EditValue="{Binding Password, ValidatesOnDataErrors=True, UpdateSourceTrigger=PropertyChanged}"/>
                </StackPanel>
                <StackPanel Orientation="Vertical" Margin="0,0,0,6">
                    <TextBlock Text="Confirm Password" Margin="6,2,0,2"/>
                    <dxe:PasswordBoxEdit EditValue="{Binding ConfirmPassword, ValidatesOnDataErrors=True, UpdateSourceTrigger=PropertyChanged}"/>
                </StackPanel>
                <Button VerticalAlignment="Top" Content="Sign Up" Width="150" HorizontalAlignment="Right" Margin="0,10"
                    IsEnabled="{Binding Path=(dxe:ValidationService.HasValidationError), ElementName=validationContainer, Converter={StaticResource BooleanNegationConverter}}"/>
            </StackPanel>
        </Grid>
    </UserControl>
    namespace Example.ViewModel {
        [POCOViewModel(ImplementIDataErrorInfo = true)]
        public class MainViewModel : ViewModelBase {
            static PropertyMetadataBuilder<MainViewModel, string> AddPasswordCheck(PropertyMetadataBuilder<MainViewModel, string> builder) {
                return builder.MatchesInstanceRule((name, vm) => vm.Password == vm.ConfirmPassword, () => "The passwords don't match.")
                    .MinLength(8, () => "The password must be at least 8 characters long.")
                    .MaxLength(20, () => "The password must not exceed the length of 20.");
            public static void BuildMetadata(MetadataBuilder<MainViewModel> builder) {
                builder.Property(x => x.FirstName)
                    .Required(() => "Please enter the first name.");
                builder.Property(x => x.LastName)
                    .Required(() => "Please enter the last name.");
                builder.Property(x => x.Email)
                    .EmailAddressDataType(() => "Please enter a correct email address.");
                AddPasswordCheck(builder.Property(x => x.Password))
                    .Required(() => "Please enter the password.");
                AddPasswordCheck(builder.Property(x => x.ConfirmPassword))
                    .Required(() => "Please confirm the password.");
            public virtual string FirstName { get; set; }
            public virtual string LastName { get; set; }
            public virtual string Email { get; set; }
            public virtual string Password { get; set; }
            public virtual string ConfirmPassword { get; set; }
            public void OnPasswordChanged() {
                this.RaisePropertyChanged(() => ConfirmPassword);
            public void OnConfirmPasswordChanged() {
                this.RaisePropertyChanged(() => Password);
    
    Imports DevExpress.Mvvm
    Imports DevExpress.Mvvm.DataAnnotations
    Imports System.Windows.Media
    Namespace Example.ViewModel
        <POCOViewModel(ImplementIDataErrorInfo := True)> _
        Public Class MainViewModel
            Inherits ViewModelBase
            Private Shared Function AddPasswordCheck(ByVal builder As PropertyMetadataBuilder(Of MainViewModel, String)) As PropertyMetadataBuilder(Of MainViewModel, String)
                Return builder.MatchesInstanceRule(Function(name, vm) vm.Password = vm.ConfirmPassword, Function() "The passwords don't match.").MinLength(8, Function() "The password must be at least 8 characters long.").MaxLength(20, Function() "The password must not exceed the length of 20.")
            End Function
            Public Shared Sub BuildMetadata(ByVal builder As MetadataBuilder(Of MainViewModel))
                builder.Property(Function(x) x.FirstName).Required(Function() "Please enter the first name.")
                builder.Property(Function(x) x.LastName).Required(Function() "Please enter the last name.")
                builder.Property(Function(x) x.Email).EmailAddressDataType(Function() "Please enter a correct email address.")
                AddPasswordCheck(builder.Property(Function(x) x.Password)).Required(Function() "Please enter the password.")
                AddPasswordCheck(builder.Property(Function(x) x.ConfirmPassword)).Required(Function() "Please confirm the password.")
            End Sub
            Public Overridable Property FirstName() As String
            Public Overridable Property LastName() As String
            Public Overridable Property Email() As String
            Public Overridable Property Password() As String
            Public Overridable Property ConfirmPassword() As String
            Public Sub OnPasswordChanged()
                Me.RaisePropertyChanged(Function() ConfirmPassword)
            End Sub
            Public Sub OnConfirmPasswordChanged()
                Me.RaisePropertyChanged(Function() Password)
            End Sub
        End Class
    End Namespace
    

    If you need to extend the default IDataErrorInfo implementation, you can manually implement the IDataErrorInfo interface and use the IDataErrorInfoHelper class.

    Use of this site constitutes acceptance of our Website Terms of Use and Privacy Policy (Updated). Cookies Settings
  •