【问题标题】:How to set SelectedItem of a ComboBox when item is not part of the ItemsSource collection?当项目不属于 ItemsSource 集合时,如何设置 ComboBox 的 SelectedItem?
【发布时间】:2012-04-22 12:03:45
【问题描述】:

以下场景:

<ComboBox ItemsSource="{Binding Path=Names}"
          SelectedItem="{Binding Path=Name, UpdateSourceTrigger=PropertyChanged}">
</ComboBox>

Names 是一个字符串集合,它可能会通过 ViewModel 中的代码进行更改。例如,姓名可以设置为男性姓名(John、Paul、George)或女性姓名(Mary、Jane、Karen)。

假设男性集合处于活动状态,并且用户选择“John”使其成为 SelectedItem。现在该系列更改为女性名称。默认情况下,SelectedItem 将设置为 Null,并且不再显示 John。而是显示一个空字符串。

我想要实现的是“John”仍然作为 SelectedItem 可见,尽管它不再在集合中。但是,由于它不是集合的一部分,我想将 SelectedItem 的颜色更改为红色。

我玩过 IsEditable 属性,但不喜欢 ComboBox 不断变化的外观。无论如何,用户应该无法手动输入文本。

我尝试使用不同的模板和样式(例如 Template、ItemTemplate、ItemContainerStyle),但没有找到适合我的方法。

是否可以按照我需要的方式设置提供的 ComboBox 的样式,还是我必须创建自己的用户控件?

编辑: 我找到了可以接受的解决方案。

<ComboBox IsEditable="True" IsReadOnly="True"
          ItemsSource="{Binding Path=Names}"
          Text="{Binding Path=Name}">
  <ComboBox.Style>
    <Style TargetType="{x:Type ComboBox}">
      <Style.Triggers>
        <DataTrigger Binding="{Binding Path=IsNameInCollection}" Value="False">
          <Setter Property="Foreground" Value="Red"/>
        </DataTrigger>
      </Style.Triggers>
    </Style>
  </ComboBox.Style>
  <ComboBox.ItemContainerStyle>
    <Style TargetType="{x:Type ComboBoxItem}">
      <Setter Property="Foreground" Value="Black"/>
    </Style>
  </ComboBox.ItemContainerStyle>
</ComboBox>

在 ViewModel 中,我确保 Name 属性在 Names 集合更改时不会更改。布尔属性 IsNameInCollection 将在 Name 属性或集合更改时更新。

【问题讨论】:

  • 所以您基本上希望一个组合框充当使用 2 个组合框?
  • 实际上在上面给出的例子中只有两个。在实际应用中,我有许多不同的集合。

标签: wpf colors combobox selecteditem


【解决方案1】:

注意:这不是一个真正可行的答案/解决方案,但只是一些您可能认为值得考虑的想法:

首先,如果集合发生变化,SelectedItem 将始终更改为 null(通过将其替换为新集合或清除并重新填充它)。因此,当您绑定到某个对象的 Name 属性时,您能否防止该属性在内部被 null 覆盖?这样,ComboBox 中的值可能不会改变(也许您还需要为 Name 提供 TwoWayBinding 以使值保持不变;甚至触发更改事件,但这取决于一些测试)。

其次,渲染需要改变,所以最简单的方法是使用ItemTemplate和自定义的IValueConverter实现来转换不同颜色/样式的特殊条目的信息。但这样做的缺点是,ItemTemplate 通常会用于每个条目,包括当前选定的条目。

解决此问题的一种方法是使用ItemTemplateSelector,如下所述:http://www.wpftutorial.net/datatemplates.html(“如何使用 DataTemplateSelector 根据数据切换模板”一节)

另一种方法是使用容器而不是纯字符串。该容器还将保存一个布尔值来指示不同的渲染。该布尔值可用于ItemTemplate。但在这种情况下,您需要包装所有字符串值。

第三,如果选择的值仍然是不在当前列表中的旧值,然后用户选择另一个值,ComboBox应该如何反应。一种逻辑是旧值消失了,但随后用户无法将其重新选择。如果它在列表中仍然可用,那么您必须以某种方式将其添加到当前值列表中,但将其标记为特殊值。在这两种情况下,您还需要一个逻辑,以便稍后检测是否选择了根据当前可用列表无效的值。

我希望这有助于找到可行的解决方案。

【讨论】:

  • 1.我确实阻止了该属性将被 null 覆盖,但即使集合时该属性没有更改,ComboBox 也不再显示它。 2. 到目前为止,我没有成功使用模板,但我会继续尝试。 3. 在这种情况下,旧值可能会消失,所以这不是问题。
  • 虽然您没有提供解决方案,但我已将您的答案标记为已接受,因为您给了我一些想法,并将我推向了正确的方向:-)
  • 我很高兴至少能帮上一点忙。
猜你喜欢
  • 2014-12-10
  • 2021-09-26
  • 1970-01-01
  • 1970-01-01
  • 2015-11-16
  • 2014-05-30
  • 2013-10-02
  • 2011-06-24
  • 2016-03-12
相关资源
最近更新 更多