WPF DataGrid: привязка видимости DataGridColumn к ContextMenu MenuItems IsChecked (MVVM) - PullRequest
7 голосов
/ 13 октября 2009

Я хочу контролировать DataGrid видимость столбца через ContextMenu, доступную пользователю, щелкнув правой кнопкой мыши заголовок столбца. ContextMenu отображает имена всех доступных столбцов. Я использую шаблон проектирования MVVM.

У меня такой вопрос: как связать свойство DataGridColumn Visibility со свойством IsChecked MenuItem, расположенным в ContextMenu.

Код макета:

<UserControl.Resources>         
    <ContextMenu x:Key="ColumnHeaderContextMenu">  
        <MenuItem Header="Menu Item..1" IsCheckable="True" />  
    </ContextMenu>  
    <Style x:Key="ColumnHeaderStyle" 
           TargetType="{x:Type toolkit:DataGridColumnHeader}">  
        <Setter Property="ContextMenu" 
                Value="{StaticResource ColumnHeaderContextMenu}" />  
    </Style>  
    <BooleanToVisibilityConverter x:Key="booleanToVisibilityConverter" />  
</UserControl.Resources>  

... flaf flaf flaf

<toolkit:DataGrid x:Name="MyGrid" AutoGenerateColumns="False" 
    ItemsSource="{Binding MyCollection, Mode=Default}" 
    EnableColumnVirtualization="True" IsReadOnly="True" 
    ColumnHeaderStyle="{StaticResource ColumnHeaderStyle}">  
    <toolkit:DataGrid.Columns>  
        <toolkit:DataGridTextColumn Binding="{Binding Path=MyEntry}" 
            Header="MyEntry" Visibility="{Binding IsChecked, Converter=
                {StaticResource booleanToVisibilityConverter}.... />
    </toolkit:DataGrid.Columns>     
</toolkit:DataGrid>  

Если я неясен, пожалуйста, дайте мне знать, и я попытаюсь уточнить.

Приветствия

Ответы [ 6 ]

18 голосов
/ 12 января 2011

Я только что написал сообщение в блоге на эту тему. Он позволяет отображать или скрывать DataGridColumns через ContextMenu, который доступен щелчком правой кнопкой мыши по любому заголовку столбца. Эта задача решается исключительно с помощью прикрепленных свойств, поэтому она совместима с MVVM.

Смотри сообщение в блоге

13 голосов
/ 04 июля 2012

Я искал универсальный , XAML (т. Е. Без кода), автоматический и простой пример контекстного меню для выбора столбцов который привязывается к заголовку столбца WPF DataGrid. Я прочитал буквально сотни статей, но ни одна из них, кажется, не делает точно правильную вещь, или они не достаточно универсальны. Вот что я считаю лучшим комбинированным решением:

Сначала поместите их в словарь ресурсов. Я оставлю читателю в качестве упражнения написание конвертера Visibility / Boolean, чтобы флажки устанавливались, когда столбец виден, и наоборот. Обратите внимание, что, определив x: Shared = "False" для ресурса контекстного меню, он получит специфичное для экземпляра состояние, что означает, что вы можете использовать этот единственный шаблон / ресурс для всех ваших таблиц данных, и они будут поддерживать свое собственное состояние.

<Converters:VisiblityToInverseBooleanConverter x:Key="VisiblityToInverseBooleanConverter"/>

<ContextMenu x:Key="ColumnChooserMenu" x:Shared="False"
             DataContext="{Binding Path=PlacementTarget, RelativeSource={RelativeSource Self}}" 
             ItemsSource="{Binding Columns, RelativeSource={RelativeSource AncestorType={x:Type sdk:DataGrid}}}">
    <ContextMenu.ItemContainerStyle>
        <Style TargetType="MenuItem">
            <Setter Property="Header" Value="{Binding Header}"/>
            <Setter Property="AutomationProperties.Name" Value="{Binding Header}"/>
            <Setter Property="IsCheckable" Value="True" />
            <Setter Property="IsChecked" Value="{Binding Visibility, Mode=TwoWay, Converter={StaticResource VisiblityToInverseBooleanConverter}}" />
        </Style>
    </ContextMenu.ItemContainerStyle>
</ContextMenu>

<Style x:Key="ColumnHeaderStyle" TargetType="{x:Type Primitives:DataGridColumnHeader}">
    <Setter Property="ContextMenu" Value="{StaticResource ColumnChooserMenu}" />
</Style>

<ContextMenu x:Key="GridItemsContextMenu" >
    <MenuItem Header="Launch Do Some other action"/>
</ContextMenu>

Затем определите DataGrid следующим образом (где OrdersQuery - это некоторый источник данных, предоставляемый моделью представления):

<sdk:DataGrid ItemsSource="{Binding OrdersQuery}"
              AutoGenerateColumns="True" 
              ColumnHeaderStyle="{StaticResource ColumnHeaderStyle}"
              ContextMenu="{StaticResource GridItemsContextMenu}">

  <!-- rest of datagrid stuff goes here -->

</sdk:DataGrid>

Это даст вам следующее:

  1. Контекстное меню , связанное с заголовками столбцов , которое действует как средство выбора столбцов.
  2. Контекстное меню, привязанное к элементам в сетке (для выполнения действий над самими элементами - опять же, привязка действий является упражнением для читателя).

Надеюсь, это поможет людям, которые искали такой пример.

6 голосов
/ 25 июля 2011

Я знаю, что это немного старо. Но я смотрел на это, и этот пост намного проще: http://iimaginec.wordpress.com/2011/07/25/binding-wpf-toolkit%E2%80%99s-datagridcolumn-to-a-viewmodel-datacontext-propogation-for-datagrid-columns-the-mvvm-way-to-interact-with-datagridcolumn/

Все, что вам нужно сделать, это установить DataContext для столбцов, а затем привязать Visibility к вашей ViewModel, как обычно! :) Просто и эффективно

1 голос
/ 16 октября 2009

Хорошо, это было довольно практичным упражнением для WPF n00b.

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

Вот то, что я придумал, если кто-то сможет найти более последовательный способ сделать это, я буду благодарен за любые комментарии:

Препятствия:

  1. DataGridColumnHeader не поддерживает контекстное меню. Поэтому контекстное меню необходимо применять как стиль.

  2. В контекстном меню есть собственный текст данных, поэтому мы должны использовать findancestor, чтобы связать его с текстом данных ViewModels.

  3. ATM, элемент управления DataGrid не выполняет синтаксический анализ своего текста данных со своими столбцами. Это может быть решено в коде, однако мы используем шаблон MVVM, поэтому я решил следовать подходу jamiers

Решение:

Поместите следующие два блока кода в Window.Resources

    <Style x:Key="ColumnHeaderStyle"
           TargetType="{x:Type toolkit:DataGridColumnHeader}">
        <Setter Property="ContextMenu" 
                Value="{StaticResource ColumnHeaderContextMenu}" />
    </Style>  

    <ContextMenu x:Key="ColumnHeaderContextMenu">
        <MenuItem x:Name="MyMenuItem"
                  IsCheckable="True"
                  IsChecked="{Binding DataContext.IsHidden, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type toolkit:DataGrid}}}"/>
    </ContextMenu>

Затем таблица данных выглядит примерно так в XAML

        <toolkit:DataGrid x:Name="MyGrid"
                          AutoGenerateColumns="False"
                          ItemsSource="{Binding SampleCollection, Mode=Default}"
                          EnableColumnVirtualization="True"
                          IsReadOnly="True"
                          ColumnHeaderStyle="{StaticResource ColumnHeaderStyle}">
            <toolkit:DataGrid.Columns>
                <toolkit:DataGridTextColumn Binding="{Binding Path=SamplingUser}"
                                            Header="{Binding (FrameworkElement.DataContext).IsHidden, RelativeSource={x:Static RelativeSource.Self}}"
                                            Visibility="{Binding (FrameworkElement.DataContext).IsHidden,
                                                RelativeSource={x:Static RelativeSource.Self},
                                                Converter={StaticResource booleanToVisibilityConverter}}"/>

Таким образом, свойство видимости в DataGridColumn и свойство ischeked связаны данными со свойством IsHidden в viewModel.

В ViewModel:

    public bool IsHidden
    {
        get { return isHidden; }
        set 
        { if (value != isHidden)
            {
                isHidden = value;
                OnPropertyChanged("IsHidden");
                OnPropertyChanged("IsVisible");
            }
        }
    }  

Класс Helper, определенный Jaimer:

class DataGridSupport
{
    static DataGridSupport() 
    {

        DependencyProperty dp = FrameworkElement.DataContextProperty.AddOwner(typeof(DataGridColumn)); 
        FrameworkElement.DataContextProperty.OverrideMetadata ( typeof(DataGrid), new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.Inherits, new PropertyChangedCallback(OnDataContextChanged)));

    }

    public static void OnDataContextChanged ( DependencyObject d, DependencyPropertyChangedEventArgs e)
    { 
        DataGrid grid = d as DataGrid ; 
        if ( grid != null  ) 
        {                 
            foreach ( DataGridColumn col in grid.Columns ) 
            { 
                col.SetValue ( FrameworkElement.DataContextProperty,  e.NewValue ); 
            } 
        } 
    }
}

Реализуется в viewmodel (просто чтобы показать, что сделано через Unity в реальном проекте)

    private static DataGridSupport dc = new DataGridSupport();

Приветствия

0 голосов
/ 14 мая 2010

Вместо booleanToVisibilityConverter вы можете использовать x: static

<Setter TargetName="UIElement"  Property="UIElement.Visibility" Value="x:Static Visibility.Hidden" />

Статика в XAML: http://msdn.microsoft.com/en-us/library/ms742135.aspx

0 голосов
/ 14 октября 2009

Я попытался получить это, чтобы привязать его к ContextMenu, используя «ElementName», но в конце концов заставил его работать, используя Свойства в ВМ, например,

bool _isHidden;
public bool IsHidden
{
  get { return _isHidden; }
  set
  {
    if (value != _isHidden)
    {
      _isHidden = value;
      RaisePropertyChanged("IsHidden");
      RaisePropertyChanged("IsVisible");
    }
  }
}

public Visibility IsVisible
{
  get { return IsHidden ? Visibility.Hidden : Visibility.Visible; }
}

и в XAML:

<Window.ContextMenu>
  <ContextMenu>
    <MenuItem Header="Hidden" IsCheckable="True" IsChecked="{Binding IsHidden}" />
  </ContextMenu>
</Window.ContextMenu>

<toolkit:DataGrid x:Name="MyGrid" AutoGenerateColumns="False" ItemsSource="{Binding MyCollection, Mode=Default}" EnableColumnVirtualization="True" IsReadOnly="True" ColumnHeaderStyle="{StaticResource ColumnHeaderStyle}">
  <toolkit:DataGrid.Columns>
    <toolkit:DataGridTextColumn Binding="{Binding Path=MyEntry}" Header="MyEntry" Visibility="{Binding Path=IsVisible, Mode=OneWay}" />
  </toolkit:DataGrid.Columns>
</toolkit:DataGrid>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...