关于itemstemplate的Scrollviewer拾取错误的itemtemplate

时间:2015-12-21 19:41:39

标签: wpf scrollviewer itemscontrol

我需要一个ItemsControl周围的scrollviewer,其ItemsPaneltemplate依赖于属性IsClusterSupported。 如果没有ScrollViewer,ItemsControl总是显示正确的ItemsPanel,但是当我添加scrollviewer代码时,即使IsClusterSupported = False,也会显示ClusterMapItemsPanelTemplate。

这里可能出现什么问题?我附上了以下代码。

<Grid Name="MapGrid">
    <ScrollViewer>
        <ScrollViewer.Style>
            <Style TargetType="{x:Type ScrollViewer}">
                <Setter Property="VerticalScrollBarVisibility" Value="Hidden"/>
                <Setter Property="HorizontalScrollBarVisibility" Value="Hidden"/>
                <Style.Triggers>
                    <DataTrigger Binding="{Binding IsScrollingSupported}" Value="True">
                        <Setter Property="VerticalScrollBarVisibility" Value="Auto"/>
                        <Setter Property="HorizontalScrollBarVisibility" Value="Auto"/>
                    </DataTrigger>
                </Style.Triggers>
            </Style>
        </ScrollViewer.Style>
        <ItemsControl Name="ItemsControlInnerMap" Loaded="ItemsControlInnerMap_Loaded"
              ItemsSource="{Binding ItemsCol}" Tag="{Binding IsClusterSupported, PresentationTraceSources.TraceLevel=High}">
            <ItemsControl.Style>
                <Style TargetType="{x:Type ItemsControl}">
                    <Setter Property="ItemsPanel" Value="{StaticResource MapItemsPanelTemplate}"/>
                    <Style.Triggers>
                        <DataTrigger Binding="{Binding IsClusterSupported}" Value="True">
                            <Setter Property="ItemsPanel" Value="{StaticResource ClusterMapItemsPanelTemplate}"/>
                            <Setter Property="ItemContainerStyle" Value="{StaticResource ClusterMapGridContainerStyle}"/>
                        </DataTrigger>
                    </Style.Triggers>
                </Style>
            </ItemsControl.Style>

            <ItemsControl.ItemTemplate>
                <DataTemplate>
                  <Border>
                    <TextBlock Text="{Binding LabelTxt}"/>
                  </Border>
                </DataTemplate>
            </ItemsControl.ItemTemplate>

            <ItemsControl.GroupStyle>
                <GroupStyle x:Name="MyGroup">

                    <GroupStyle.ContainerStyle>
                        <Style TargetType="{x:Type GroupItem}">
                            <Setter Property="Template">
                                <Setter.Value>
                                    <ControlTemplate>
                                        <Border Name="BD" BorderBrush="Black" BorderThickness="2">
                                            <ItemsPresenter Name="ClusterPresenter" Tag="{Binding Name}"/>
                                        </Border>
                                        <ControlTemplate.Triggers>
                                            <DataTrigger Binding="{Binding Name, Converter={StaticResource BorderVisibilityConvertor}}" Value="false">
                                                <Setter TargetName="BD" Property="BorderBrush" Value="Transparent"/>
                                            </DataTrigger>
                                        </ControlTemplate.Triggers>
                                    </ControlTemplate>
                                </Setter.Value>
                            </Setter>
                            <!--http://stackoverflow.com/questions/17863271/wpf-using-grid-as-itemshost-stacking-multiple-items-in-single-cell-automatical-->
                            <Setter Property="Grid.Row" Value="{Binding Items[0].ClusterGridRow}"/>
                            <Setter Property="Grid.Column" Value="{Binding Items[0].ClusterGridColumn}"/>
                            <Setter Property="Margin" Value="5"/>
                        </Style>
                    </GroupStyle.ContainerStyle>

                    <GroupStyle.Panel>
                        <ItemsPanelTemplate>
                            <Grid Name="ClusterMapGrid" HorizontalAlignment="Stretch" VerticalAlignment="Stretch"
                                  Loaded="ClusterMapGrid_Loaded"
                                  MaxHeight="{Binding AllowedMaxHeight}" MaxWidth="{Binding AllowedMaxWidth}"
                                  MinHeight="{Binding ReqMinHeight}" MinWidth="{Binding ReqMinWidth}">
                            </Grid>ReqMinHeight
                        </ItemsPanelTemplate>
                    </GroupStyle.Panel>

                </GroupStyle>
            </ItemsControl.GroupStyle>
        </ItemsControl>
    </ScrollViewer>
</Grid>
RDV(2016年4月1日):我在这里发现了一些有趣的东西,希望能够更接近解决方案,但仍然需要帮助。

  1. 我的所有绑定工作正常;我用&amp;调试了没有scrollviewer并在ItemsControl中看到了正确的ItemsPanel。 ItemsControl的IsGrouping属性也设置正确。

  2. 当我在ItemsControl周围有一个scrollviewer时,无论我有没有集群,都会加载ClusterMapGrid(GroupStyle.Panel)。加载后,不会加载非群集模板(MapsItemsPanelTemplate)。

  3. 为了解决这个问题,我尝试将ClusterMapGrid可见性绑定到ItemsControl的IsGrouping属性,但是在滚动查看器的情况下再次以某种方式加载了ClusterMapGrid,并且一旦加载,非群集模板就无法加载。

    < / LI>
  4. 我使用snoop检查ClusterMapGrid的可见性,我可以看到它已折叠,但是如果手动将其更改为snoop到任何其他值,则非群集模板加载,一切正常。我尝试将可见性设置为在ClusterMapGrid_Loaded事件中折叠,但它也没有帮助。

  5. 进一步调试我发现ScrollViewer正在加载其中的所有项目 - 因此加载了ClusterMapGrid,一旦我使用Snoop更改可见性,就会在非集群情况下卸载它。

  6. 问题陈述:如何在加载ClusterMapGrid时卸载它,我知道这是一个非集群场景。

1 个答案:

答案 0 :(得分:0)

问题是ScrollViewer初始化&amp;加载XAML中的所有内容,即使它不是要加载的(例如,如果ItemsControl上的IsGrouping为true,则应加载GroupStyle.Panel,但是使用scrollviewer,它仍然会被加载)。

解决方案:从XAML中注释掉GroupStyle.Panel并在参考资料中将其添加为ItemsPanelTemplate:

    <ItemsPanelTemplate x:Key="ClusterPanelTemplate">
                <Grid Name="ClusterMapGrid" HorizontalAlignment="Stretch" VerticalAlignment="Stretch"                                                                      
                      Loaded="ClusterMapGrid_Loaded"
                      MaxHeight="{Binding AllowedMaxHeight}" MaxWidth="{Binding AllowedMaxWidth}"
                      MinHeight="{Binding ReqMinHeight}" MinWidth="{Binding ReqMinWidth}"/>
   </ItemsPanelTemplate>

在ItemsControlInnerMap_Loaded事件代码隐藏中添加以下内容:

private void ItemsControlInnerMap_Loaded(object sender, RoutedEventArgs e)
        {
            if ((sender as ItemsControl).IsGrouping)
            {
                (sender as ItemsControl).GroupStyle[0].Panel = (ItemsPanelTemplate)this.FindResource("ClusterPanelTemplate");
            }
            else
            {
                (sender as ItemsControl).GroupStyle.Clear();
            }
        }