gpt4 book ai didi

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

转载 作者:太空宇宙 更新时间:2023-11-03 15:07:59 24 4
gpt4 key购买 nike

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

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com