c# - DataGrid 更改时添加 MessageBox 检查

标签 c# wpf mvvm

我已经发布了一个桌面应用程序(所以我很感激将更改和回归测试保持在最低限度的答案)并且我需要在网格更改时添加一致性检查 CanBeDeleted .

<DataGrid AutoGenerateColumns="False" 
    ItemsSource="{Binding CurrentPosIn.PosInLocationsList}" 
    CanUserAddRows="{Binding UpdateEnabled}" CanUserDeleteRows="{Binding UpdateEnabled}" >

我正在使用 UpdateEnabled 做一些不同的事情(配置文件权限),我不想让 DataGrid 只读:我更喜欢 (除非它太复杂) 以查看阻止更改的阻止警报(MessageBox)。

到目前为止我所做的是

  1. 反对 MVVM,因为我已将警报放入模型中(但我可以接受,如果它使更改变得快速和简单的话)
  2. 不起作用,因为我的更改的第二部分(见下文)产生了无效操作异常

ViewModel 包含以下列表

    [Association(ThisKey="Partita", OtherKey="Partita", CanBeNull=true, IsBackReference=true)]
    public ObservableCollection<Model.PosInLocation> posin_locations_list = new  ObservableCollection<Model.PosInLocation>(); 
    public ObservableCollection<PosInLocation> PosInLocationsList {
        get { return posin_locations_list; }
        set { 
            posin_locations_list = value;
            OnPropertyChanged( () => PosInLocationsList );
        }
    } 

我在这里添加一致性检查

    string _storage;
    [Column(Name = "storage"), PrimaryKey]
    public string Storage {
        get { return _storage; }
        set { 
            if (this.loadedEF) {
                string validate_msg;
                if (!PurchasePosIn.CanBeDeleted(out validate_msg)) {
                    // against MVVM
                    MessageBox.Show(validate_msg, "Alert", MessageBoxButton.OK); 
                    OnPropertyChanged( () => Storage  );
                    return;
                }
                Persistence.MyContext.deletePosInLocation(this);
            }
            _storage = value;
            OnPropertyChanged( () => Storage  );
            if (this.loadedEF) {
                Persistence.MyContext.insertPosInLocation(this);
            }               
        }
    }

和这里(第二部分)

    internal void posin_locations_list_CollectionChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs args)
    {                   
        string validate_msg;
        if (!CanBeDeleted(out validate_msg)) {
            // indirectly produces an invalid operation exception 
            MessageBox.Show(validate_msg, "Alert", MessageBoxButton.OK);
            return;
        }
        if (args.OldItems != null)
            foreach(var oldItem in args.OldItems) {
                if ( ((PosInLocation)oldItem).Partita != null)        
                    Persistence.MyContext.deletePosInLocation((PosInLocation)oldItem);              
            }       
        if (args.NewItems != null)
            foreach(var newItem in args.NewItems)
            {
                PosInLocation newPosInLocation = (PosInLocation)newItem;
                if ( newPosInLocation.Partita == null) {
                    newPosInLocation.Partita = this.Partita;
                    newPosInLocation.PurchasePosIn = this;
                    newPosInLocation.loadedEF = true;
                }
             }
    }

最佳答案

如果 ObservableCollection 实现了一个“previewCollectionChanged”,事情就会简单得多。

根据您的需要,我建议创建 ObservableCollection 的子类并重载 protected 方法 RemoveItem。
根据您对应用程序执行的操作,您可能希望覆盖除 RemoveItem 之外的其他方法(例如 ClearItems)。

当继承 ObservableCollection 时,有 5 个 protected 方法可以覆盖:ClearItems、RemoveItem、InsertItem、SetItem 和 MoveItem。
这些方法最终会被所有公共(public)方法调用,因此覆盖它们可以让您完全控制。

这是一个您可以运行的小应用程序,它演示了这一点:

ObservableCollection 子类

public class ObservableCollectionWithDeletionControl<T> : ObservableCollection<T>
{
    public delegate void DeletionDeniedEventHandler(object sender, int indexOfDeniedDeletion);
    public event DeletionDeniedEventHandler DeletionDenied;

    public bool CanDelete { get; set; }

    protected virtual void OnDeletionDenied(int indexOfDeniedDeletion)
    {
        if (DeletionDenied != null) { DeletionDenied(this, indexOfDeniedDeletion); }
    }

    protected override void RemoveItem(int index)
    {
        if (CanDelete)
        {
            base.RemoveItem(index);
        }
        else
        {
            OnDeletionDenied(index);
        }
    }
}

我使用 DeletionDenied 事件,这样这个类就不用负责显示错误窗口了,而且它更容易重用。

View 模型

public class MainWindowViewModel
{
    public MainWindow window { get; set; }

    public ObservableCollectionWithDeletionControl<Person> People { get; set; } = new ObservableCollectionWithDeletionControl<Person>();

    public MainWindowViewModel()
    {
        People.DeletionDenied += People_DeletionDenied;
    }

    private void People_DeletionDenied(object sender, int indexOfDeniedDeletion)
    {
        Person personSavedFromDeletion = People[indexOfDeniedDeletion];
        window.displayDeniedDeletion(personSavedFromDeletion.Name);
    }
}

主窗口的 ViewModel。
它知道它的窗口的唯一目的是显示错误消息。
(我确信有比这更好的解决方案,但我还没有找到一个好的和简短的方法来显示弹出窗口mvvm.)
当 DeletionDenied 事件触发时,调用错误窗口。

型号

public class Person : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    public string Name
    {
        get { return _name; }
        set
        {
            if(_name == value) { return; }
            _name = value;
            if( PropertyChanged != null ) { PropertyChanged(this, new PropertyChangedEventArgs("Name")); }
        }
    }

    private string _name = "";
}


XAML

<Window x:Class="WpfApplication1.MainWindow"
    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:local="clr-namespace:WpfApplication1"
    mc:Ignorable="d"
    Title="MainWindow" Height="350" Width="525">
    <DockPanel>
        <CheckBox DockPanel.Dock="Top" Content="Can delete" IsChecked="{Binding People.CanDelete}" Margin="5" HorizontalAlignment="Left"/>
        <DataGrid ItemsSource="{Binding People}" Margin="5,0"/>
    </DockPanel>
</Window>


XAML.CS

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
    }

    public void displayDeniedDeletion(string name)
    {
        TextBox errorMessage = new TextBox();
        errorMessage.Text = string.Format("Cannot delete {0} : access denied !", name);

        Window popupErrorMessage = new Window();
        popupErrorMessage.Content = errorMessage;

        popupErrorMessage.ShowDialog();
    }
}

主应用

public partial class App : Application
{
    private void Application_Startup(object sender, StartupEventArgs e)
    {
        MainWindow window = new MainWindow();
        MainWindowViewModel viewModel = new MainWindowViewModel();

        viewModel.window = window;
        window.DataContext = viewModel;
        window.Show();

        App.Current.MainWindow = window;
    }
}

我已经在启动时设置了 ViewModel 的窗口,但您可能应该在创建 ViewModel 的任何地方进行设置


关于c# - DataGrid 更改时添加 MessageBox 检查,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42528460/

相关文章:

c# - Silverlight/C# 图像未找到异常处理

wpf - 带有 MVVM 的 WPF 中的 COMBOBOX 过滤

c# - 如何在 WPF 中使用异步初始化对 ViewModel 进行单元测试

c# - 条件属性是否也消除子表达式?

c# - 在 Entity Framework 中,如何在没有枚举所有可能 DbSet 的 switch 语句的情况下将通用实体添加到其相应的 DbSet?

wpf - MVVM 中的 ObservableCollection 和 Repository 模式 - 如何关联它们,以及 "Update"方法的案例

wpf - 如何点击页面的一个按钮来改变Windows的框架来源?

c# - WinUi MVVM,在 UI 线程上处理来自模型的事件

c# - 如何在 Azure Functions 中配置 session cookie?

c# - 如何阻止 WPF KeyDown 事件从某些包含的控件(例如 TextBox)中冒出?