从模型更新 ViewModel 中的属性的正确方法

gan*_*eii 0 wpf notifications model mvvm viewmodel

我对 WPF 相当新手。我的理解是模型中的数据发生变化,它应该通知视图模型,并且视图将绑定到视图模型中的属性和类似的东西。它是否正确?如果是这样,我一直在读该模型应该实现INotifyPropertyChanged,并且看起来像这样

 public class LoginModel : INotifyPropertyChanged
{

    public event PropertyChangedEventHandler PropertyChanged;

    public void NotifyPropertyChanged(string propName)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propName));
    }

    public bool Authenticated { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

在我的 ViewModel 中,我有一个属性“AuthResult”,它应该从模型属性“Authenticated”获取更新

public partial class view1 : UserControl, INotifyPropertyChanged{
 public bool AuthResult
    {
        get
        {
            return _authVal;
        }
        set
        {
            _authVal = value;
            NotifyPropertyChanged("AuthResult");
        }
    }

public event PropertyChangedEventHandler PropertyChanged;

    public void NotifyPropertyChanged(string propName)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propName));
    }
}
Run Code Online (Sandbox Code Playgroud)

我知道当前的实施是不正确的。我发现我应该从我的模型订阅 PropertyChanged 通知,如下所示:

    LoginModel.PropertyChanged += new System.ComponentModel.PropertyChangedEventHandler(LoginModel_PropertyChanged);

void LoginModel_PropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e)
{
    if(e.PropertyName == "Authenticated")
    {
         //do something
    }
}
Run Code Online (Sandbox Code Playgroud)

我不知道应该在哪里更新“AuthResult”属性。我会在 If 语句中做一些类似的事情吗AuthResult = _model.Authenticated;

编辑:

在我的构造函数中?

LoginModel _model;

        public view1(LoginModel model)
        {
            _model = model;
            InitializeComponent();           
        }
Run Code Online (Sandbox Code Playgroud)

mm8*_*mm8 5

如果模型实现了 INotifyPropertyChanged 接口,您可以从视图直接绑定到它:

<Button Content="Button" IsEnabled="{Binding Authenticated}" />
Run Code Online (Sandbox Code Playgroud)

请注意,只要将 Authenticated 属性设置为新值,LoginModel 类就必须引发 PropertyChanged 事件。

您还可以通过视图模型类公开整个模型实体:

public class ViewModel
{
    public ViewModel(LoginModel model)
    {
        Model = model;
    }

    public LoginModel Model { get; }
}
Run Code Online (Sandbox Code Playgroud)

...并像这样绑定它:

<Button Content="Button" IsEnabled="{Binding Model.Authenticated}" />
Run Code Online (Sandbox Code Playgroud)

它仍然是模型类必须实现 INotifyPropertyChanged 接口并引发更改通知。

另一种选择是让视图模型包装您希望能够从视图绑定到的模型类的任何属性。然后,您绑定到视图模型类的属性,该属性又包装模型类的属性,如下所示:

public class ViewModel
{
    private readonly LoginModel _model;
    public ViewModel(LoginModel model)
    {
        _model = model;
    }

    public bool AuthResult
    {
        get
        {
            return _model.Authenticated;
        }
        set
        {
            _model.Authenticated = value;
            NotifyPropertyChanged("AuthResult");
        }
    }
}

<Button Content="Button" IsEnabled="{Binding AuthResult}" />
Run Code Online (Sandbox Code Playgroud)

使用后一种方法的好处是视图不依赖于模型类。它仅绑定到视图模型类,这就是 MVVM 设计模式通常的实现方式。

但是,如果您确实绑定到视图模型的(包装器)属性,并且希望每当设置模型类的属性时更新视图,则模型必须通知视图模型它已以某种方式更改,即它必须引发某种事件或类似事件。这通常意味着实现 INotifyPropertyChanged 接口。然后,视图模型可以订阅模型的 PropertyChanged 事件,并在模型更新时为数据绑定属性引发自己的 PropertyChanged 事件,例如:

public class ViewModel
{
    private readonly LoginModel _model;

    public ViewModel(LoginModel model)
    {
        if (model == null)
            throw new ArgumentNullException("model");

        _model = model;
        _model.PropertyChanged += OnModelChanged;
    }

    private void OnModelChanged(object sender, PropertyChangedEventArgs e)
    {
        if (e.PropertyName == "Authenticated")
            NotifyPropertyChanged("AuthResult");
    }

    public bool AuthResult
    {
        get
        {
            return _model.Authenticated;
        }
        set
        {
            _model.Authenticated = value;
            NotifyPropertyChanged("AuthResult");
        }
    }
}
Run Code Online (Sandbox Code Playgroud)