Подтвердить что ты не робот

Отображение видимости столбцов Bind datagrid MVVM

.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>

Я прочитал почти все решения этой проблемы, и ничего не работает.

4b9b3361

Ответ 1

DataGridColumn не являются частью визуального дерева, поэтому они не связаны с контекстом данных DataGrid.

Для них connect вместе используйте подход прокси-элемента, подобный этому...

  • Добавьте прокси FrameworkElement в панель предков Resources.
  • Подключите его к невидимому ContentControl, привязанному к его Content.
  • Используйте этот 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> 
    

Помимо DataGridColumn, вышеприведенный подход отлично подходит для подключения DataContext к Popup и ContextMenu (т.е. к любому элементу, который не связан с визуальным деревом).

Пользователи Silverlight

Печально настроенное содержимое элементов управления содержимым с любыми элементами структуры не допускается в silverlight. Таким образом, обходной путь был бы (это всего лишь код наведения для silverlight)...

  • Измените ресурс элемента структуры на что-то легкое, как Textblock. (Silverlight не позволяет указать статический ресурс типа FrameworkElement.)

    <StackPanel.Resources>
        <TextBlock x:Key="MyTextBlock" />
    
  • Напишите прикрепленное свойство для хранения текстового блока для элемента управления содержимым.

    <ContentControl Visibility="Collapsed" 
                    local:MyAttachedBehavior.ProxyElement="{StaticResource MyTextBlock}" />
    
  • В зависимом атрибуте обработчика зависимых зависимостей задайте привязку контекста данных элемента управления содержимым к текстовому блоку.

     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);
           }
     }
    

Таким образом, текстовый блок не может быть связан с визуальным деревом, но будет вероятно знать изменения контекста данных.

Надеюсь, что это поможет.