ComboBox 样式问题:DisplayMemberPath 的应用困难

4
我有一个ComboBox,我已经将combo.ItemsSource属性设置为List对象。Book类包含两个属性:“Abbreviation”和“Name”。
我已将ComboBox的DisplayMemberPath设置为“Abbreviation”,但是在ComboBox上设置的以下样式不显示Abbreviation属性,而是显示类的名称“Words.Book”。ComboBox下拉列表正确地显示了列表(使用指定的Abbreviation属性)。问题出现在选定的ComboBox项目中,即当ComboBox的下拉列表关闭时显示的项目。
我猜问题出现在DataTemplate中的ContentPresenter,但是我找不到成功的解决方案。目前,ContentPresenter的Content属性设置为Content =“{TemplateBinding Content}”,但我不知道是否应该这样设置。
有什么想法可以在所选项目上显示DisplayMemberPath指定的属性吗?
谢谢
代码如下:
<ControlTemplate x:Key="ComboBoxToggleButton" TargetType="ToggleButton">
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition />
            <ColumnDefinition Width="20" />
        </Grid.ColumnDefinitions>
        <Border x:Name="Border" Grid.ColumnSpan="2" CornerRadius="2" BorderThickness="1" Background="{DynamicResource ribbonTitleFade}" />
        <Border Grid.Column="0" CornerRadius="2,0,0,2" Margin="1,6,1,6" BorderBrush="{DynamicResource boSecE}" BorderThickness="0,0,1,0" />
        <Path x:Name="Arrow" Grid.Column="1" Fill="White" 
              HorizontalAlignment="Center" VerticalAlignment="Center" Data="M 0 0 L 4 4 L 8 0 Z"
              />
    </Grid>
    <ControlTemplate.Triggers>
        <Trigger Property="ToggleButton.IsMouseOver" Value="true">
            <Setter TargetName="Border" Property="Background" Value="{DynamicResource ribbonTitleFade}" />
        </Trigger>
        <Trigger Property="ToggleButton.IsChecked" Value="true">
            <Setter TargetName="Border" Property="Background" Value="{DynamicResource ribbonTitleFade}" />
        </Trigger>
        <Trigger Property="IsEnabled" Value="False">
            <Setter TargetName="Border" Property="Background" Value="Black" />
            <Setter TargetName="Border" Property="BorderBrush" Value="Black" />
            <Setter Property="Foreground" Value="Gray"/>
            <Setter TargetName="Arrow" Property="Fill" Value="Gray" />
        </Trigger>
    </ControlTemplate.Triggers>
</ControlTemplate>

<Style x:Key="comboBoxBorderTransparent" TargetType="Control">
    <Setter Property="BorderBrush" Value="{DynamicResource boPrimC}" />
</Style>


<Style x:Key="comboItemStyle" TargetType="ComboBoxItem">
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type ComboBoxItem}">
                <Border x:Name="backBorder" >
                    <StackPanel Orientation="Horizontal">
                        <Rectangle x:Name="rectA" Stroke="White" Width="4" Height="4" Fill="#80FFFFFF" Margin="5,0,0,0" HorizontalAlignment="Left" />
                        <TextBlock x:Name="text" Foreground="White" FontSize="10px">
                            <ContentPresenter Margin="8,1,0,1"
                                              SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" 
                                              HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}" 
                                              VerticalAlignment="{TemplateBinding VerticalContentAlignment}"
                                              />
                        </TextBlock>
                    </StackPanel>
                </Border>
                <ControlTemplate.Triggers>
                    <Trigger Property="ComboBoxItem.IsMouseOver" Value="true">
                        <Setter TargetName="rectA" Property="Stroke" Value="Black" />
                        <Setter TargetName="rectA" Property="Fill" Value="#80000000" />
                        <Setter TargetName="backBorder" Property="Background" Value="White"/>
                        <Setter TargetName="text" Property="Foreground" Value="{DynamicResource boPrimC}"/>
                    </Trigger>
                </ControlTemplate.Triggers>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>

<DataTemplate x:Key="comboSelectedItemTemplate">
    <TextBlock Foreground="White" FontSize="10px">
        <ContentPresenter Content="{TemplateBinding Content}" />
    </TextBlock>
</DataTemplate>

<Style TargetType="ComboBox">
    <Setter Property="SnapsToDevicePixels" Value="true"/>
    <Setter Property="OverridesDefaultStyle" Value="true"/>
    <Setter Property="ScrollViewer.HorizontalScrollBarVisibility" Value="Auto"/>
    <Setter Property="ScrollViewer.VerticalScrollBarVisibility" Value="Auto"/>
    <Setter Property="ScrollViewer.CanContentScroll" Value="true"/>
    <Setter Property="MinWidth" Value="60"/>
    <Setter Property="MinHeight" Value="20"/>
    <Setter Property="ItemContainerStyle" Value="{DynamicResource comboItemStyle}"/>
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="ComboBox">
                <Grid>
                    <ToggleButton 
                        Name="ToggleButton" Grid.Column="2" 
                        Template="{StaticResource ComboBoxToggleButton}" 
                        Focusable="false"
                        IsChecked="{Binding Path=IsDropDownOpen,Mode=TwoWay,RelativeSource={RelativeSource TemplatedParent}}"
                        ClickMode="Press">
                    </ToggleButton>

                    <ContentPresenter HorizontalAlignment="Center"
                                      VerticalAlignment="Center"
                                      Name="ContentSite"
                                      IsHitTestVisible="False" 
                                      Content="{TemplateBinding SelectionBoxItem}"
                                      ContentTemplate="{DynamicResource comboSelectedItemTemplate}"
                                      ContentTemplateSelector="{TemplateBinding ItemTemplateSelector}"
                                      Margin="3,3,23,3"
                                      />

                    <Popup Name="Popup" Placement="Bottom" IsOpen="{TemplateBinding IsDropDownOpen}"
                           AllowsTransparency="False" Focusable="False" PopupAnimation="Slide">
                        <Grid Name="DropDown" SnapsToDevicePixels="True" 
                              MinWidth="{TemplateBinding ActualWidth}" MaxHeight="{TemplateBinding MaxDropDownHeight}">
                            <Border x:Name="DropDownBorder" Background="{DynamicResource ribbonTitleFade}"
                                    BorderThickness="1" BorderBrush="{DynamicResource boPrimC}" />
                            <ScrollViewer Margin="4,6,4,6" SnapsToDevicePixels="True">
                                <StackPanel IsItemsHost="True" KeyboardNavigation.DirectionalNavigation="Contained" />
                            </ScrollViewer>
                        </Grid>
                    </Popup>
                </Grid>
                <ControlTemplate.Triggers>
                    <Trigger Property="HasItems" Value="false">
                        <Setter TargetName="DropDownBorder" Property="MinHeight" Value="95"/>
                    </Trigger>
                    <Trigger Property="IsEnabled" Value="false">
                        <Setter Property="Foreground" Value="Black"/>
                    </Trigger>
                    <Trigger Property="IsGrouping" Value="true">
                        <Setter Property="ScrollViewer.CanContentScroll" Value="false"/>
                    </Trigger>
                    <Trigger SourceName="Popup" Property="Popup.AllowsTransparency" Value="true">
                        <Setter TargetName="DropDownBorder" Property="CornerRadius" Value="2"/>
                        <Setter TargetName="DropDownBorder" Property="Margin" Value="0"/>
                    </Trigger>
                    <Trigger Property="IsEditable" Value="true">
                        <Setter Property="IsTabStop" Value="false"/>
                        <!--<Setter TargetName="ContentSite" Property="Visibility" Value="Hidden"/>-->
                    </Trigger>
                </ControlTemplate.Triggers>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
    <Style.Triggers>
    </Style.Triggers>
</Style>
3个回答

3
我认为您希望在ComboBox模板中使用类似以下内容的内容:

我认为您希望在ComboBox模板中使用以下内容:

<ContentPresenter Content="{TemplateBinding ComboBox.SelectionBoxItem}"
 ContentTemplate="{TemplateBinding ComboBox.SelectionBoxItemTemplate}"
 ContentStringFormat="{TemplateBinding ComboBox.SelectionBoxItemStringFormat}"
 Margin="{TemplateBinding Control.Padding}"
 HorizontalAlignment="{TemplateBinding Control.HorizontalContentAlignment}"
 VerticalAlignment="{TemplateBinding Control.VerticalContentAlignment}"
 SnapsToDevicePixels="{TemplateBinding UIElement.SnapsToDevicePixels}" />

我已经按照你的建议更改了Content、ContentTemplate和ContentStringFormat,现在它开始按照预期显示。现在唯一的问题是我不知道在哪里设置文本的前景画刷应该是白色。我之前是在DataTemplate中设置的,但由于这个DataTemplate不再使用,那么在哪里放置Foreground="White"是正确的呢?谢谢。 - Kornelije Petak
1
我认为你可以在ComboBoxItem.ItemContainerStyle上放置一个触发器,当IsSelected = true时将Foreground更改为白色。 - Kent Boogaart
ComboBoxItem.ItemContainerStyle 反映了下拉列表中的项目。我需要的是选定项目的前景色(当下拉列表关闭时)。目前,它被设置为:ContentTemplate="{TemplateBinding ComboBox.SelectionBoxItemTemplate}",但我不知道如何更改 ComboBox.SelectionBoxItemTemplate。 - Kornelije Petak
尝试: ContentTemplateSelector={TemplateBinding ItemTemplateSelector} 如果您使用与Popup相同的Controltemplate - dba

0
如果您的模板没有明确处理DisplayMemberPath的值,那么该模板将不会使用它来更改ComboBoxItem。您必须直接在ControlTemplate中绑定到成员,或者使用单独的DataTemplate

0

我发现存在问题,它无法应用DisplayMemberPath,只显示对象名称。我有一个窗口有很多(15个以上)ComboBoxes,除了2个之外,所有的都使用DisplayMemberPath正常工作。

我从ContentPresenters和GridViewRowPresenters中删除了以下内容:

Content="{TemplateBinding Content}" ContentTemplate="{TemplateBinding ContentTemplate}"

ListBoxItem、ComboBoxItem、ListViewItem 中。可能还有其他需要修复的,但这是我们发现正在使用的全部。

使用 Microsoft.Net 4.0 Framework。我没有尝试过 4.5 或 3.5。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接