绑定datagrid列可见性MVVM

时间:2011-10-10 10:12:16

标签: c# wpf xaml .net-3.5 wpftoolkit

.Net 3.5

我知道列不会继承datacontext并且通过阅读其他帖子我认为这样可行:

Visibility="{Binding RelativeSource={x:Static RelativeSource.Self},
                     Path=(FrameworkElement.DataContext).IsColumnNameVisible,
                     Converter={StaticResource boolToVisConverter}}"

但当然不会.. 输出窗口没有抱怨,似乎我找到的资源但是viewmodel属性被更新了。

这是整个DG:

<tk:DataGrid                                        
            VirtualizingStackPanel.IsVirtualizing="False"                                        
            Grid.Column="0"
            AlternationCount="2"
            AreRowDetailsFrozen="True"
            AutoGenerateColumns="False"
            Background="Transparent"
            BorderThickness="0"
            CanUserAddRows="False"
            CanUserReorderColumns="True"
            CanUserResizeRows="False"
            GridLinesVisibility="None"
            ItemsSource="{Binding Employees}"
            SelectionMode="Single"
            ColumnHeaderStyle="{StaticResource columnHeaderStyle}"
            RowHeaderStyle="{StaticResource rowHeaderStyle}"
            CellStyle="{StaticResource cellStyle}"
            RowStyle="{StaticResource rowStyle}" 
            ContextMenu="{StaticResource columnHeaderContextMenu}">
    <tk:DataGrid.Resources>
        <ContextMenu x:Key="columnHeaderContextMenu" ItemsSource="{Binding ColumnHeaderContextMenuItems}" />
        <Style TargetType="{x:Type ScrollBar}">
            <Setter Property="Background" Value="Transparent"/>
        </Style>                                    
        <Style TargetType="{x:Type tk:DataGridColumnHeader}">
            <Setter Property="Background" Value="Transparent"/>
        </Style>
    </tk:DataGrid.Resources>
    <tk:DataGrid.Triggers>
        <EventTrigger RoutedEvent="tk:DataGridRow.MouseDoubleClick">
            <EventTrigger.Actions>
                <BeginStoryboard Storyboard="{StaticResource showDetailGrid}"/>
            </EventTrigger.Actions>
        </EventTrigger>
    </tk:DataGrid.Triggers>
    <tk:DataGrid.Columns>
        <tk:DataGridTextColumn IsReadOnly="True" Header="test" Binding="{Binding Name, Mode=OneWay}" Visibility="{Binding RelativeSource={x:Static RelativeSource.Self}, Path=(FrameworkElement.DataContext).IsColumnNameVisible, Converter={StaticResource boolToVisConverter}}"  />
    </tk:DataGrid.Columns>
</tk:DataGrid>

我几乎已经阅读了这个问题的每一个解决方案,但没有任何作用。

1 个答案:

答案 0 :(得分:100)

DataGridColumn不属于可视树,因此它们不会连接到DataGrid的数据上下文。

对于他们连接一起使用像这样的代理元素方法......

  1. 在祖先面板的FrameworkElement中添加代理Resources
  2. 将其托管到与其ContentControl绑定的不可见Content
  3. 将此ProxyElement用作StaticResource用于可见性绑定中的数据上下文源。

    <StackPanel>
        <StackPanel.Resources>
           <local:BooleanToVisibilityConverter
                  x:Key="BooleanToVisibilityConverter" />
    
           <FrameworkElement x:Key="ProxyElement"
                             DataContext="{Binding}"/>
        </StackPanel.Resources>
        <ContentControl Visibility="Collapsed"
                    Content="{StaticResource ProxyElement}"/>
        <DataGrid AutoGenerateColumns="False">
            <DataGrid.Columns>
                <DataGridTextColumn
                       Visibility="{Binding DataContext.IsTextColumnVisibile,
                                            Source={StaticResource ProxyElement},
                                            Converter={StaticResource
                                                BooleanToVisibilityConverter}}"
                       Binding="{Binding Text}"/>
            </DataGrid.Columns>
        </DataGrid>
    </StackPanel> 
    
  4. 除了DataGridColumn之外,上述方法也很适合将DataContext连接到PopupContextMenu s(即任何未连接到可视树的元素) )。

    Silverlight用户

    遗憾的是,Silverlight中不允许使用任何框架元素设置内容控件的内容。所以解决方法是(这只是silverlight的指导代码)......

    1. 将框架元素资源更改为轻量级,如Textblock。 (Silverlight不允许指定FrameworkElement类型的静态资源。)

      <StackPanel.Resources>
          <TextBlock x:Key="MyTextBlock" />
      
    2. 编写附加属性以保留对内容控件的文本块。

      <ContentControl Visibility="Collapsed" 
                      local:MyAttachedBehavior.ProxyElement="{StaticResource MyTextBlock}" />
      
    3. 在附加的依赖项属性更改事件处理程序中,将内容控件的数据上下文绑定到文本块。

       private static void OnProxyElementPropertyChanged(
           DependencyObject depObj, DependencyPropertyChangedEventArgs e)
       {
             if (depObj is ContentControl && e.NewValue is TextBlock)
             {
                 var binding = new Binding("DataContext");
                 binding.Source = depObj;
                 binding.Mode = OneWay;
                 BindingOperations.SetBinding(
                     (TextBlock)e.NewValue, TextBlock.DataContextProperty, binding);
             }
       }
      
    4. 因此,文本块可能无法连接到可视树,但可能会知道数据上下文更改。

      希望这有帮助。