我的WPF应用程序中有一个包含对象的DataGrid
控件。该对象有一个布尔属性,可以通过用户操作进行更改。当该属性的值更改时,我需要更改行的样式。
我写了一个源自StyleSelector
:的类
public class LiveModeSelector : StyleSelector {
public Style LiveModeStyle { get; set; }
public Style NormalStyle { get; set; }
public override Style SelectStyle( object item, DependencyObject container ) {
DataGridRow gridRow = container as DataGridRow;
LPRCamera camera = item as LPRCamera;
if ( camera != null && camera.IsInLiveMode ) {
return LiveModeStyle;
}
return NormalStyle;
}
}
有问题的视图模型类实现INotifyPropertyChanged
,当有问题的属性更改时,它将引发PropertyChanged
事件。
// Note: The ModuleMonitor class implements INotifyPropertyChanged and raises the PropertyChanged
// event in the SetAndNotify generic method.
public class LPRCamera : ModuleMonitor, ICloneable {
. . .
public bool IsInLiveMode {
get { return iIsInLiveMode; }
private set { SetAndNotify( "IsInLiveMode", ref iIsInLiveMode, value ); }
}
private bool iIsInLiveMode;
. . .
/// </summary>
public void StartLiveMode() {
IsInLiveMode = true;
. . .
}
public void StopLiveMode() {
IsInLiveMode = false;
. . .
}
}
当用户执行所需操作时,属性的值会更改,但样式不会更改。
我在SelectStyle方法中放置了一个断点,当第一次加载控件时,我看到断点会被命中,但当属性的值更改时,它不会被命中。
我错过了什么?
我从@Rachel对我问题的回答中找到了一种方法。然而,代码的细节有些不同,我想确切地展示什么是有效的。
第一步是将两个不同的Styles
组合为一个DataGridRow
类:
<Style TargetType="DataGridRow" x:Key="CameraStyle">
<Setter Property="Foreground" Value="{DynamicResource TextForeground}" />
<Setter Property="Background" Value="{DynamicResource DataBackground}" />
<Style.Triggers>
<DataTrigger Binding="{Binding Path=IsInLiveMode}" Value="True">
<Setter Property="Foreground" Value="Red" />
<Setter Property="Background" Value="Yellow" />
</DataTrigger>
</Style.Triggers>
</Style>
第二步是将DataGrid
控件的RowStyle
属性设置为这种新样式:
<DataGrid . . .
RowStyle={StaticResource CameraStyle}">
. . .
</DataGrid>
这是有效的。当用户将与该行关联的LPRCamera
置于实时模式时,该行的前景和背景会发生变化,而当用户将其从实时模式中取出时,前景和背景又会恢复正常,这正是我想要的。
谢谢@Rachel!
我认为StyleSelector
不会侦听PropertyChange
通知,所以当IsInLiveMode
属性更改时,它不会重新运行。
将你的风格放在基于IsInLiveMode
的DataTrigger
中,当属性更改通知发出时,它将被重新评估。
<DataGrid.Resources>
<Style TargetType="{x:Type DataGridRow}" x:Key="Style1">
<Setter Property="Background" Value="Red" />
</Style>
<Style TargetType="{x:Type DataGridRow}" x:Key="Style2">
<Setter Property="Background" Value="Blue" />
</Style>
</DataGrid.Resources>
<DataGrid.Style>
<Style TargetType="{x:Type DataGrid}">
<Setter Property="RowStyle" Value="{StaticResource Style1}" />
<Style.Triggers>
<DataTrigger Binding="{Binding ElementName=MyDataGrid, Path=DataContext.IsInLiveMode}" Value="True">
<Setter Property="RowStyle" Value="{StaticResource Style2}" />
</DataTrigger>
</Style.Triggers>
</Style>
</DataGrid.Style>