首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >WPF:无法将ItemsPresenter设置为使用VirtualizingStackPanel

WPF:无法将ItemsPresenter设置为使用VirtualizingStackPanel
EN

Stack Overflow用户
提问于 2017-09-17 20:50:34
回答 2查看 448关注 0票数 0

我见过一些关于这个问题的帖子,包括在StackOverflow上,但是我找不到答案。

在WPF中,我为一个ControlTemplate创建了一个ComboBox,只需单击visual中的“编辑模板->复制”即可。

现在我的问题是下拉列表中的项是在ItemsPresenter中显示的,而我只是不知道如何在该ItemsPresenter中使用一个VirtualizingStackPanel。我尝试在控件的样式中设置控件的ItemsPanel,但这没有帮助。

以下是ControlTemplate的XAML。很标准的。ItemsPresenter被称为PART_ItemsPresenter。

代码语言:javascript
复制
<ControlTemplate x:Key="ComboBoxControlTemplate" TargetType="{x:Type ComboBox}">
    <Grid x:Name="templateRoot" SnapsToDevicePixels="True">
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="*" />
            <ColumnDefinition MinWidth="{DynamicResource {x:Static SystemParameters.VerticalScrollBarWidthKey}}"
                              Width="0" />
        </Grid.ColumnDefinitions>
        <Popup
            x:Name="PART_Popup"
            AllowsTransparency="True"
            Grid.ColumnSpan="2"
            IsOpen="{Binding IsDropDownOpen, RelativeSource={RelativeSource TemplatedParent}}"
            PopupAnimation="Fade"
            Placement="Bottom">
            <!--<themes:SystemDropShadowChrome x:Name="shadow" Color="Transparent" MaxHeight="{TemplateBinding MaxDropDownHeight}" MinWidth="{Binding ActualWidth, ElementName=templateRoot}">-->
            <Border
                x:Name="DropDownBorder"
                BorderBrush="{TemplateBinding BorderBrush}"
                BorderThickness="{TemplateBinding BorderThickness}"
                Background="{DynamicResource {x:Static SystemColors.WindowBrushKey}}"
                MinWidth="{Binding ActualWidth, ElementName=templateRoot}">
                <ScrollViewer
                    x:Name="DropDownScrollViewer">
                    <Grid x:Name="grid" RenderOptions.ClearTypeHint="Enabled">
                        <Canvas x:Name="canvas" HorizontalAlignment="Left" Height="0" VerticalAlignment="Top"
                                Width="0">
                            <Rectangle x:Name="OpaqueRect" Fill="{Binding Background, ElementName=DropDownBorder}"
                                       Height="{Binding ActualHeight, ElementName=DropDownBorder}"
                                       Width="{Binding ActualWidth, ElementName=DropDownBorder}" />
                        </Canvas>
                        <ItemsPresenter
                            x:Name="PART_ItemsPresenter"
                            KeyboardNavigation.DirectionalNavigation="Contained"
                            SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}">
                        </ItemsPresenter>
                    </Grid>
                </ScrollViewer>
            </Border>
            <!--</themes:SystemDropShadowChrome>-->
        </Popup>
        <ToggleButton
            x:Name="toggleButton"
            BorderBrush="{TemplateBinding BorderBrush}"
            BorderThickness="{TemplateBinding BorderThickness}"
            Background="{TemplateBinding Background}"
            Grid.ColumnSpan="2"
            IsChecked="{Binding IsDropDownOpen, Mode=TwoWay, RelativeSource={RelativeSource TemplatedParent}}">
            <ToggleButton.Style>
                <Style TargetType="{x:Type ToggleButton}">
                    <Setter Property="OverridesDefaultStyle" Value="True" />
                    <Setter Property="IsTabStop" Value="False" />
                    <Setter Property="Focusable" Value="False" />
                    <Setter Property="ClickMode" Value="Press" />
                    <Setter Property="Template">
                        <Setter.Value>
                            <ControlTemplate TargetType="{x:Type ToggleButton}">
                                <Border x:Name="templateRoot" BorderBrush="{TemplateBinding BorderBrush}"
                                        BorderThickness="{TemplateBinding BorderThickness}"
                                        Background="{StaticResource ControlBackgroundBrush}"
                                        SnapsToDevicePixels="True">
                                    <Border x:Name="splitBorder" BorderBrush="{TemplateBinding BorderBrush}"
                                            BorderThickness="1" HorizontalAlignment="Right" Margin="0,-1,-1,-1"
                                            Width="{DynamicResource {x:Static SystemParameters.VerticalScrollBarWidthKey}}">
                                        <Path
                                            x:Name="Arrow"
                                            Data="M0,0C0,0 3.5,4 3.5,4 3.5,4 7,0 7,0 7,0 0,0 0,0z"
                                            Fill="{TemplateBinding BorderBrush}"
                                            HorizontalAlignment="Center"
                                            VerticalAlignment="Center" />
                                    </Border>
                                </Border>
                                <ControlTemplate.Triggers>
                                    <Trigger Property="IsMouseOver" Value="True">
                                        <Setter Property="Background" TargetName="templateRoot"
                                                Value="{StaticResource ControlBackgroundMouseOverBrush}" />
                                        <Setter Property="BorderBrush" TargetName="templateRoot"
                                                Value="{StaticResource ControlBorderMouseOverBrush}" />
                                        <Setter Property="BorderBrush" TargetName="splitBorder"
                                                Value="{StaticResource ControlBorderMouseOverBrush}" />
                                        <Setter Property="Fill" TargetName="Arrow"
                                                Value="{StaticResource ControlBorderMouseOverBrush}" />
                                    </Trigger>
                                    <Trigger Property="IsPressed" Value="True">
                                        <Setter Property="Background" TargetName="templateRoot" Value="#FFC4E5F6" />
                                        <Setter Property="BorderBrush" TargetName="templateRoot" Value="#FF2C628B" />
                                    </Trigger>
                                    <Trigger Property="IsEnabled" Value="False">
                                        <Setter Property="Background" TargetName="templateRoot" Value="#FFF4F4F4" />
                                        <Setter Property="BorderBrush" TargetName="templateRoot" Value="#FFADB2B5" />
                                        <Setter Property="Background" TargetName="splitBorder" Value="#FFF4F4F4" />
                                        <Setter Property="BorderBrush" TargetName="splitBorder" Value="#FFADB2B5" />
                                        <Setter Property="Fill" TargetName="Arrow"
                                                Value="{DynamicResource {x:Static SystemColors.GrayTextBrushKey}}" />
                                    </Trigger>
                                </ControlTemplate.Triggers>
                            </ControlTemplate>
                        </Setter.Value>
                    </Setter>
                </Style>
            </ToggleButton.Style>
        </ToggleButton>
        <Border x:Name="Border" Background="{StaticResource ControlBackgroundBrush}"
                Margin="{TemplateBinding BorderThickness}">
            <TextBox x:Name="PART_EditableTextBox"
                     HorizontalContentAlignment="{TemplateBinding HorizontalContentAlignment}"
                     IsReadOnly="{Binding IsReadOnly, RelativeSource={RelativeSource TemplatedParent}}"
                     Margin="{TemplateBinding Padding}"
                     VerticalContentAlignment="{TemplateBinding VerticalContentAlignment}">
                <TextBox.Style>
                    <Style TargetType="{x:Type TextBox}">
                        <Setter Property="OverridesDefaultStyle" Value="True" />
                        <Setter Property="AllowDrop" Value="True" />
                        <Setter Property="MinWidth" Value="0" />
                        <Setter Property="MinHeight" Value="0" />
                        <Setter Property="FocusVisualStyle" Value="{x:Null}" />
                        <Setter Property="ScrollViewer.PanningMode" Value="VerticalFirst" />
                        <Setter Property="Stylus.IsFlicksEnabled" Value="False" />
                        <Setter Property="Template">
                            <Setter.Value>
                                <ControlTemplate TargetType="{x:Type TextBox}">
                                    <ScrollViewer
                                        x:Name="PART_ContentHost"
                                        Template="{Binding PART_ItemsPresenter.Template}"
                                        Background="Transparent"
                                        Focusable="False"
                                        HorizontalScrollBarVisibility="Hidden"
                                        VerticalScrollBarVisibility="Hidden" />
                                </ControlTemplate>
                            </Setter.Value>
                        </Setter>
                    </Style>
                </TextBox.Style>
            </TextBox>
        </Border>
    </Grid>
    <ControlTemplate.Triggers>
        <Trigger Property="IsEnabled" Value="False">
            <Setter Property="Opacity" TargetName="Border" Value="0.56" />
        </Trigger>
        <Trigger Property="IsKeyboardFocusWithin" Value="True">
            <Setter Property="Foreground" Value="Black" />
        </Trigger>
        <!--<Trigger Property="HasDropShadow" SourceName="PART_Popup" Value="True">
            <Setter Property="Margin" TargetName="shadow" Value="0,0,5,5"/>
            <Setter Property="Color" TargetName="shadow" Value="#71000000"/>
        </Trigger>-->
        <Trigger Property="HasItems" Value="False">
            <Setter Property="Height" TargetName="DropDownBorder" Value="95" />
        </Trigger>
        <MultiTrigger>
            <MultiTrigger.Conditions>
                <Condition Property="IsGrouping" Value="True" />
                <Condition Property="VirtualizingPanel.IsVirtualizingWhenGrouping" Value="False" />
            </MultiTrigger.Conditions>
            <Setter Property="ScrollViewer.CanContentScroll" Value="False" />
        </MultiTrigger>
        <Trigger Property="CanContentScroll" SourceName="DropDownScrollViewer" Value="False">
            <Setter Property="Canvas.Top" TargetName="OpaqueRect"
                    Value="{Binding VerticalOffset, ElementName=DropDownScrollViewer}" />
            <Setter Property="Canvas.Left" TargetName="OpaqueRect"
                    Value="{Binding HorizontalOffset, ElementName=DropDownScrollViewer}" />
        </Trigger>
    </ControlTemplate.Triggers>
</ControlTemplate>
EN

回答 2

Stack Overflow用户

发布于 2017-09-17 21:51:22

您需要做多件事来支持虚拟化。设置ItemsPanelTemplate是解决方案的一部分:

代码语言:javascript
复制
<Setter Property="VirtualizingPanel.IsVirtualizing" Value="True" />
<Setter Property="ScrollViewer.CanContentScroll" Value="True" />
<Setter Property="ItemsPanel">
    <Setter.Value>
        <ItemsPanelTemplate>
            <VirtualizingStackPanel />
        </ItemsPanelTemplate>
    </Setter.Value>
</Setter>
票数 0
EN

Stack Overflow用户

发布于 2017-09-18 10:19:43

ItemsPanel设置为VirtualizingStackPanel确实有效:

代码语言:javascript
复制
<Setter Property="ItemsPanel">
    <Setter.Value>
        <ItemsPanelTemplate>
            <VirtualizingStackPanel />
        </ItemsPanelTemplate>
    </Setter.Value>
</Setter>

但是,为了使实际的虚拟化工作,还应该在模板中设置DropDownBorder元素的DropDownBorder属性(或者不要注释掉SystemDropShadowChrome元素):

代码语言:javascript
复制
<Border x:Name="DropDownBorder" MaxHeight="{TemplateBinding MaxDropDownHeight}" ...
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/46268761

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档