我在MVVM应用程序内的WPF表单上有一个可编辑的数据网格。
用户可以在此页面上执行两种可能的操作,这些操作会导致其中一行内的某些数据发生更改。其中一个可编辑字段 - 请求 - 可以导致另一个属性发生变化,并且样式触发器依赖于其值:
public bool OverRequested
{
get
{
if(this.Requested > this.Volume)
{
return true;
}
return false;
}
}
在XAML中:
<DataGridTextColumn Header="Requested" Binding="{Binding Requested}">
<DataGridTextColumn.CellStyle>
<Style TargetType="DataGridCell">
<Style.Triggers>
<DataTrigger Binding="{Binding OverRequested}" Value="true">
<Setter Property="Foreground" Value="Red"/>
<Setter Property="ToolTip" Value="The requested volume is greater than the available volume" />
</DataTrigger>
</Style.Triggers>
</Style>
</DataGridTextColumn.CellStyle>
</DataGridTextColumn>
另一个是更新行后面的数据项并更改项目值的按钮。
尽管datagrid本身对更改做出响应 - 如果您从基础ObservableCollection中添加或删除行,它会在UI中相应更新 - 对行下方项目的这些属性的更改不会更新。
我理解这是设计上的,因为observableCollection中的项目更改不会冒泡,必须特别注意,但我很难找到如何实现我想要的
我已经尝试过 - 正如其他地方所建议的那样 - 在ObservableCollection上覆盖CollectionChanged的事件处理程序,为内部的项添加通知警告:
private void ObservableCollection_CollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
{
if (e.NewItems != null)
{
foreach (var item in e.NewItems)
{
(item as INotifyPropertyChanged).PropertyChanged += new PropertyChangedEventHandler(item_PropertyChanged);
}
}
}
但是里面的项目没有实现INotifyPropertyChanged接口 - 它们是实体框架对象而不是ViewModels。演员阵容失败了。
有没有办法实现这个?只需要观看两个属性,我很乐意手动执行 - 但即使调用OnPropertyChanged(&#34;&#34;)来更新ViewModel中的所有属性也不会导致数据网格内的属性刷新。
答案 0 :(得分:1)
也许你可以用另一种方式尝试,使用转换器。我有一个类似的东西的应用程序。我通常需要一个测试应用才能让这些东西恰到好处,但试试这个:
<DataGridTextColumn Header="Requested" Binding="{Binding Requested}">
<DataGridTextColumn.CellStyle>
<Style TargetType="DataGridCell">
<Setter Property="Foreground">
<Setter.Value>
<MultiBinding Converter="{StaticResource OverRequestedForegroundMultiConverter}">
<Binding Path="Requested" />
<Binding Path="Volume" />
</MultiBinding>
</Setter.Value>
</Setter>
<Setter Property="ToolTip">
<Setter.Value>
<MultiBinding Converter="{StaticResource OverRequestedTooltipMultiConverter}">
<Binding Path="Requested" />
<Binding Path="Volume" />
</MultiBinding>
</Setter.Value>
</Setter>
</Style>
</DataGridTextColumn.CellStyle>
转换器看起来像这样:
public class OverRequestedForegroundMultiConverter : IMultiValueConverter
{
#region IValueConverter Members
public object Convert(object[] value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
if (value != null && value.Length == 2)
{
if (value[0] is int && value[1] is int)
{
int requested = (int)value[0];
int volume = (int)value[1];
if (requested > volume)
return Colors.Red;
}
}
return Colors.Gray; // Or whatever color you want
}
public object[] ConvertBack(object value, Type[] targetType, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotImplementedException();
}
#endregion
}
public class OverRequestedTooltipMultiConverter : IMultiValueConverter
{
#region IValueConverter Members
public object Convert(object[] value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
if (value != null && value.Length == 2)
{
if (value[0] is int && value[1] is int)
{
int requested = (int)value[0];
int volume = (int)value[1];
if (requested > volume)
return "The requested volume is greater than the available volume";
}
}
return null;
}
public object[] ConvertBack(object value, Type[] targetType, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotImplementedException();
}
#endregion
}
不要忘记将转换器添加到app.xaml:
<app:OverRequestedForegroundMultiConverter x:Key="OverRequestedForegroundMultiConverter" />
<app:OverRequestedTooltipMultiConverter x:Key="OverRequestedTooltipMultiConverter" />