首页 文章

WPF MVVM - Datagrid不更新子项的已更改属性

提问于
浏览
0

我在MVVM应用程序中的WPF表单上有一个可编辑的数据网格 .

用户可以在此页面上执行两种可能的操作,这些操作会导致其中一行内的某些数据发生更改 . 其中一个可编辑字段 - Requested - 可以导致另一个属性发生更改,并且有一个样式触发器依赖于其值:

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(“”)来更新ViewModel中的所有属性也不会导致数据网格内的那些属性刷新 .

1 回答

  • 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" />
    

相关问题