【问题标题】:How to get DataTemplate.DataTrigger to check for greater than or less than?如何让 DataTemplate.DataTrigger 检查大于或小于?
【发布时间】:2010-10-22 02:10:30
【问题描述】:

如果年龄等于 30,则以下DataTemplate.DataTrigger 会使年龄显示为红色。

如果年龄大于 30,如何使年龄显示为红色?

<DataTemplate DataType="{x:Type local:Customer}">
    <Grid x:Name="MainGrid" Style="{StaticResource customerGridMainStyle}">
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="100"/>
            <ColumnDefinition Width="150"/>
        </Grid.ColumnDefinitions>
        <Grid.RowDefinitions>
            <RowDefinition/>
            <RowDefinition/>
            <RowDefinition/>
        </Grid.RowDefinitions>
        <TextBlock Grid.Column="0" Grid.Row="0" Text="First Name" Margin="5"/>
        <TextBlock Grid.Column="1" Grid.Row="0" Text="{Binding FirstName}" Margin="5"/>
        <TextBlock Grid.Column="0" Grid.Row="1" Text="Last Name" Margin="5"/>
        <TextBlock Grid.Column="1" Grid.Row="1" Text="{Binding LastName}" Margin="5"/>
        <TextBlock Grid.Column="0" Grid.Row="2" Text="Age" Margin="5"/>
        <TextBlock x:Name="Age" Grid.Column="1" Grid.Row="2" Text="{Binding Age}" Margin="5"/>
    </Grid>

    <DataTemplate.Triggers>
        <DataTrigger Binding="{Binding Path=Age}">
            <DataTrigger.Value>30</DataTrigger.Value>
            <Setter TargetName="Age" Property="Foreground" Value="Red"/> 
        </DataTrigger>
    </DataTemplate.Triggers>

</DataTemplate>

【问题讨论】:

    标签: wpf xaml datatemplate datatrigger


    【解决方案1】:

    我建议使用IValueConverter 绑定到年龄TextBlockForeground 元素并在那里隔离着色逻辑。

    <TextBlock x:Name="Age"  
               Text="{Binding Age}" 
               Foreground="{Binding Path=Age, Converter={StaticResource AgeToColorConverter}}" />
    

    然后在代码中:

    [ValueConversion(typeof(int), typeof(Brush))]
    public class AgeToColorConverter : IValueConverter
    {
       public object Convert(object value, Type target)
       {
          int age;
          Int32.TryParse(value.ToString(), age);
          return (age >= 30 ? Brushes.Red : Brushes.Black);
       }
    }
    

    【讨论】:

      【解决方案2】:

      我相信通过使用 MVVM 和INotifyPropertyChanged 的强大功能可以更简单地实现目标。


      使用Age 属性创建另一个属性,该属性将是一个名为IsAgeValid 的布尔值。 IsAgeValid 将只是按需检查,技术上不需要OnNotify 调用。如何?

      要将更改推送到 Xaml,请将要为 IsAgeValid 触发的 OnNotifyPropertyChanged 事件 Age setter 中代替。

      任何与IsAgeValid 的绑定都将在任何Age 更改订阅上发送通知消息;这才是真正被关注的……


      一旦设置完成,当然要根据IsAgeValid结果绑定样式触发器为false和true。

      public bool IsAgeValid{ get { return Age > 30; } }
      
      public int Age
      { 
        get { return _Age; }
      
        set
        {
         _Age=value;
         OnPropertyChanged("Age");   
         OnPropertyChanged("IsAgeValid"); // When age changes, so does the
                                          // question *is age valid* changes. So 
                                          // update the controls dependent on it.
         } 
       }
      

      【讨论】:

      • 这应该是这个具体例子的答案。在可能的情况下,代表真实业务逻辑的控制属性应该通过 ViewModel 提供给视图,而不是“视图端”。这允许通过配置/数据库等配置值。
      • @TomDeloford 在虚拟机上拥有业务逻辑,对我来说也很有意义。
      【解决方案3】:

      您可以创建一个IValueConverter,它根据CutOff 将整数转换为布尔值。然后使用DataTrigger.ValueTrue(或False,取决于您返回的内容)。

      如果我没记错的话,WPF DataTriggers 是严格的相等比较器。

      所以类似于:

      public class CutoffConverter : IValueConverter {
          public object Convert(object value, Type targetType, object parameter, CultureInfo culture) {
              return ((int)value) > Cutoff;
          }
      
          public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture) {
              throw new NotImplementedException();
          }
      
          public int Cutoff { get; set; }
      }
      

      然后使用以下 XAML。

      <Window.Resources>
          <myNamespace:CutoffConverter x:Key="AgeConverter" Cutoff="30" />
      </Window.Resources>
      
      <DataTemplate.Triggers>
          <DataTrigger Binding="{Binding Path=Age,
                                         Converter={StaticResource AgeConverter}}">
              <DataTrigger.Value>true</DataTrigger.Value>
              <Setter TargetName="Age" Property="Foreground" Value="Red"/> 
          </DataTrigger>
      </DataTemplate.Triggers>
      

      【讨论】:

      • 或者,如果您需要不同触发器的不同值,您可以使用 ConverterParameter 传入截止值:Binding="{Binding Path=Age, Converter={StaticResource AgeConverter}, ConverterParameter=30}"
      【解决方案4】:

      如果可能,您可以为模型添加属性,这是最简单的方法。 例如。

      public int AgeBoundry
      {
          get
          {
              if (Age < 30)
                  return 0;
              else if (Age == 30)
                  return 1;
              else
                  return 2;
          }
      }
      

      然后你可以检查整数的值。

      <DataTemplate.Triggers>
          <DataTrigger Binding="{Binding Path=Age}">
              <DataTrigger.Value>0</DataTrigger.Value>
              <Setter TargetName="Age" Property="Foreground" Value="Green"/> 
          </DataTrigger>
          <DataTrigger Binding="{Binding Path=Age}">
              <DataTrigger.Value>1</DataTrigger.Value>
              <Setter TargetName="Age" Property="Foreground" Value="Orange"/> 
          </DataTrigger>
          <DataTrigger Binding="{Binding Path=Age}">
              <DataTrigger.Value>2</DataTrigger.Value>
              <Setter TargetName="Age" Property="Foreground" Value="Red"/> 
          </DataTrigger>
      </DataTemplate.Triggers>
      

      【讨论】:

      • Ageboundry 对触发器的影响在哪里?而不是绑定到Age,不应该是AgeBoundary吗?
      猜你喜欢
      • 1970-01-01
      • 2017-10-23
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-03-24
      • 1970-01-01
      • 2015-05-22
      相关资源
      最近更新 更多