如何阻止 INotify 更新两次



我正在更新一个Datagrid,当用户输入一个已经存在的数字时,我想通知用户他们已经存在,然后从datagrid中清除值。

我知道为什么会发生这种情况,但我不知道如何阻止这种情况或如何变通。

这是非常简化的代码:首先将 EF 代码与 MVVM 模型一起使用。

public partial class StaffMasterData
{
    public System.Guid Id { get; set; } // ID (Primary key)        
    public int? StaffNo { get; set; } // StaffNo        
    public StaffMasterData()
    {
        InitializePartial();
    }
    partial void InitializePartial();        
}

StaffMasterData的实体扩展类:

public partial class StaffMasterData : INotifyPropertyChanged
{
    partial void InitializePartial()
    {
        Id = Guid.NewGuid();            
    }
    public event PropertyChangedEventHandler PropertyChanged;
    protected virtual void OnPropertyChanged([CallerMemberName]string propertyName = null)
    {
        PropertyChangedEventHandler handler = PropertyChanged;
        handler?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }

以及保存数据的方法:

public void SaveMasterData(StaffMasterData nwRowData)
{
    using (var db = CreateDbContext())
    {
        //MasterDataBinding is the observableCollection
        //the datagrid is being bound to.
        var staffNoExists = MasterDataBinding.Any(p => p.StaffNo == nwRowData.StaffNo);
        if (!staffNoExists)
        {
            db.StaffMasterDatas.AddOrUpdate(nwRowData);
            db.SaveChanges();
        }
        else
        {
            Alerts.Error("Staff Number exists");
            nwRowData.StaffNo = null;
        }
    }
}

并且集合的断言更改了事件:

public class ShiftManagerViewModel : INotifyPropertyChanged
{
    private ObservableCollection<StaffMasterData> _mMasterDataBinding = new ObservableCollection<StaffMasterData>();
    public ObservableCollection<StaffMasterData> MasterDataBinding
    {
        get { return _mMasterDataBinding; }
        set
        {
            if (value != _mMasterDataBinding)
            {
                _mMasterDataBinding = value;
                OnPropertyChanged();                    
            }
        }
    }

    public ShiftManagerViewModel()
    {          
        _mMasterDataBinding.CollectionChanged += collectionChanged_Event;
    }

    private void collectionChanged_Event(object sender, NotifyCollectionChangedEventArgs e)
    {
        if (e.NewItems != null && e.NewItems.Count > 0)
        {
            foreach (INotifyPropertyChanged item in e.NewItems.OfType<INotifyPropertyChanged>())
            {
                item.PropertyChanged += propertyChanged_Event;
            }
        }
        if (e.OldItems != null && e.OldItems.Count > 0)
        {
            foreach (INotifyPropertyChanged item in e.OldItems.OfType<INotifyPropertyChanged>())
            {
                item.PropertyChanged -= propertyChanged_Event;
            }
        }
    }
    public void propertyChanged_Event(object sender, PropertyChangedEventArgs e)
    {
        if (sender is StaffMasterData)
        {
            SaveMasterData((StaffMasterData)sender);
        }
    }
    public event PropertyChangedEventHandler PropertyChanged;
    protected virtual void OnPropertyChanged([CallerMemberName]string propertyName = null)
    {
        PropertyChangedEventHandler handler = PropertyChanged;
        handler?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }
}

可能很清楚,当运行这行代码时 nwRowData.StaffNo = null; ,它会在集合被修改时再次触发事件,然后依次运行messageBox代码并弹出两次。

老实说,我已经用这个碰到了一堵砖墙,任何正确方向的点都将不胜感激。

可以使用一个标志来确定是否实际调用 SaveMasterData 方法。在将 StaffNo 属性设置为 null 之前将此标志设置为 false,然后立即将其设置回true

private bool _handle = true;
public void SaveMasterData(StaffMasterData nwRowData)
{
    using (var db = CreateDbContext())
    {
        //MasterDataBinding is the observableCollection
        //the datagrid is being bound to.
        var staffNoExists = MasterDataBinding.Any(p => p.StaffNo == nwRowData.StaffNo);
        if (!staffNoExists)
        {
            db.StaffMasterDatas.AddOrUpdate(nwRowData);
            db.SaveChanges();
        }
        else
        {
            Alerts.Error("Staff Number exists");
            _handle = false;
            nwRowData.StaffNo = null;
            _handle = true;
        }
    }
}
public void propertyChanged_Event(object sender, PropertyChangedEventArgs e)
{
    if (!_handle && sender is StaffMasterData)
    {
        SaveMasterData((StaffMasterData)sender);
    }
}

最新更新