我知道我可以创建一个setter来检查一个值是否为NULL并执行某些操作。例如:
<TextBlock>
<TextBlock.Style>
<Style>
<Style.Triggers>
<DataTrigger Binding="{Binding SomeField}" Value="{x:Null}">
<Setter Property="TextBlock.Text" Value="It's NULL Baby!" />
</DataTrigger>
</Style.Triggers>
</Style>
</TextBlock.Style>
</TextBlock>
但是如何检查“not”值...如“NOT NULL”或“NOT = 3”?这可能在XAML中吗?
结果:谢谢你的答案......我知道我可以做一个价值转换器(这意味着我必须进入代码,这不是我希望的纯XAML) 。但是,这确实回答了在纯XAML中有效“不”你无法做到的问题。但是,选择的答案可能是创建这种功能的最佳方式。很好找。
答案 0 :(得分:138)
您可以使用IValueConverter:
<TextBlock>
<TextBlock.Resources>
<conv:IsNullConverter x:Key="isNullConverter"/>
</TextBlock.Resources>
<TextBlock.Style>
<Style>
<Style.Triggers>
<DataTrigger Binding="{Binding SomeField, Converter={StaticResource isNullConverter}}" Value="False">
<Setter Property="TextBlock.Text" Value="It's NOT NULL Baby!"/>
</DataTrigger>
</Style.Triggers>
</Style>
</TextBlock.Style>
</TextBlock>
IsNullConverter在别处定义(并且conv设置为引用其命名空间):
public class IsNullConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
return (value == null);
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new InvalidOperationException("IsNullConverter can only be used OneWay.");
}
}
更通用的解决方案是实现一个IValueConverter来检查与ConverterParameter的相等性,这样你就可以检查任何东西,而不仅仅是null。
答案 1 :(得分:131)
这有点作弊,但我只是设置了默认样式,然后使用DataTrigger覆盖它,如果值为null ...
<Style>
<!-- Highlight for Reviewed (Default) -->
<Setter Property="Control.Background" Value="PaleGreen" />
<Style.Triggers>
<!-- Highlight for Not Reviewed -->
<DataTrigger Binding="{Binding Path=REVIEWEDBY}" Value="{x:Null}">
<Setter Property="Control.Background" Value="LightIndianRed" />
</DataTrigger>
</Style.Triggers>
</Style>
答案 2 :(得分:36)
我遇到了与DataTriggers类似的限制,看起来你只能检查是否相等。我见过的最接近你可能会帮助你的是一种除了平等之外进行其他类型比较的技巧。
This blog post描述了如何在DataTrigger中进行比较,如LT,GT等。
这可以在某种程度上解决DataTrigger的限制,使用转换器将数据按到一个特殊值,然后可以比较,如Robert Macnee的答案所示。
答案 3 :(得分:22)
与null比较(正如Michael Noonan所说):
<Style>
<Style.Triggers>
<DataTrigger Binding="{Binding SomeProperty}" Value="{x:Null}">
<Setter Property="Visibility" Value="Collapsed" />
</DataTrigger>
</Style.Triggers>
</Style>
比较非null(没有转换器):
<Style>
<Setter Property="Visibility" Value="Collapsed" />
<Style.Triggers>
<DataTrigger Binding="{Binding SomeProperty}" Value="{x:Null}">
<Setter Property="Visibility" Value="Visible" />
</DataTrigger>
</Style.Triggers>
</Style>
答案 4 :(得分:14)
如果选择了listview项目(即非空),我使用它只启用按钮:
<Style TargetType="{x:Type Button}">
<Setter Property="IsEnabled" Value="True"/>
<Style.Triggers>
<DataTrigger Binding="{Binding ElementName=lvMyList, Path=SelectedItem}" Value="{x:Null}">
<Setter Property="IsEnabled" Value="False"/>
</DataTrigger>
</Style.Triggers>
</Style>
答案 5 :(得分:14)
您可以在 Expression Blend 附带的 Microsoft.Expression.Interactions.dll 中使用DataTrigger
课程。
代码示例:
<i:Interaction.Triggers>
<i:DataTrigger Binding="{Binding YourProperty}" Value="{x:Null}" Comparison="NotEqual">
<ie:ChangePropertyAction PropertyName="YourTargetPropertyName" Value="{Binding YourValue}"/>
</i:DataTrigger
</i:Interaction.Triggers>
使用此方法,您也可以触发GreaterThan
和LessThan
。
要使用此代码,您应该引用两个dll:
System.Windows.Interactivity.dll
Microsoft.Expression.Interactions.dll
答案 6 :(得分:6)
<StackPanel.Style>
<Style>
<Setter Property="StackPanel.Visibility" Value="Visible"></Setter>
<Style.Triggers>
<DataTrigger Binding="{Binding ElementName=ProfileSelectorComboBox, Path=SelectedItem.Tag}" Value="{x:Null}">
<Setter Property="StackPanel.Visibility" Value="Collapsed"></Setter>
</DataTrigger>
</Style.Triggers>
</Style>
</StackPanel.Style>
我刚刚在这里使用了逆逻辑...当我的comboitem没有填充时,将我的stackpanel设置为不可见,它的效果非常好!
答案 7 :(得分:5)
停止!没有转换器!我不想“卖”这个人的库,但我讨厌每次我想在XAML中比较东西时做转换器的事实。
使用此库:Decoraptor
你可以做到[还有更多]:
首先,在windows / userControl的声明中:
<Windows....
xmlns:conv="clr-namespace:CalcBinding;assembly=CalcBinding"
>
然后,在文本块中
<TextBlock>
<TextBlock.Style>
<Style.Triggers>
<DataTrigger Binding="{conv:Binding 'MyValue==null'}" Value="false">
<Setter Property="Background" Value="#FF80C983"></Setter>
</DataTrigger>
</Style.Triggers>
</TextBlock.Style>
</TextBlock>
神奇的部分是 conv:Binding'MYValue == null'。事实上,您可以设置您想要的任何条件[查看文档]。
请注意,我不是第三方的粉丝。但这个库是免费的,影响不大(只需在项目中加上2.dll)。答案 8 :(得分:3)
我的解决方案在DataContext实例中(如果使用MVVM,则为ViewModel)。如果满足我想要的Not Null条件,我添加一个返回true的属性。
Public ReadOnly Property IsSomeFieldNull() As Boolean
Get
Return If(SomeField is Null, True, False)
End Get
End Property
并将DataTrigger绑定到上面的属性。 注意:在VB.NET中一定要使用运算符If和NOT IIf函数,它不能与Null对象一起使用。 然后XAML是:
<DataTrigger Binding="{Binding IsSomeFieldNull}" Value="False">
<Setter Property="TextBlock.Text" Value="It's NOT NULL Baby!" />
</DataTrigger>
答案 9 :(得分:2)
转换器:
public class NullableToVisibilityConverter: IValueConverter
{
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
return value == null ? Visibility.Collapsed : Visibility.Visible;
}
}
结合:
Visibility="{Binding PropertyToBind, Converter={StaticResource nullableToVisibilityConverter}}"
答案 10 :(得分:2)
您可以在ViewModel中使用转换器或创建新属性,如下所示:
public bool CanDoIt
{
get
{
return !string.IsNullOrEmpty(SomeField);
}
}
并使用它:
<DataTrigger Binding="{Binding SomeField}" Value="{Binding CanDoIt}">
答案 11 :(得分:2)
如果您正在寻找不使用IValueConverter的解决方案,您可以随时使用以下机制
<StackPanel>
<TextBlock Text="Border = Red when null value" />
<Border x:Name="border_objectForNullValueTrigger" HorizontalAlignment="Stretch" Height="20">
<Border.Style>
<Style TargetType="Border">
<Setter Property="Background" Value="Black" />
<Style.Triggers>
<DataTrigger Binding="{Binding ObjectForNullValueTrigger}" Value="{x:Null}">
<Setter Property="Background" Value="Red" />
</DataTrigger>
</Style.Triggers>
</Style>
</Border.Style>
</Border>
<TextBlock Text="Border = Green when not null value" />
<Border HorizontalAlignment="Stretch" Height="20">
<Border.Style>
<Style TargetType="Border">
<Setter Property="Background" Value="Green" />
<Style.Triggers>
<DataTrigger Binding="{Binding Background, ElementName=border_objectForNullValueTrigger}" Value="Red">
<Setter Property="Background" Value="Black" />
</DataTrigger>
</Style.Triggers>
</Style>
</Border.Style>
</Border>
<Button Content="Invert Object state" Click="Button_Click_1"/>
</StackPanel>