Привязка свойства объекта ячейки к DataGridCell в WPF DataGrid

Используя WPF DataGrid, мне нужно изменить различные свойства отображения и связанные свойства DataGridCell, такие как Foreground, FontStyle, IsEnabled и т. Д., В зависимости от соответствующего значения свойства объекта ячейки.

Теперь это легко сделать в коде, например (используя Observable Collection of ObservableDictionaries):

  var b = new Binding("IsLocked") { Source = row[column], Converter = new BoolToFontStyleConverter() };
  cell.SetBinding(Control.FontStyleProperty, b);

и работает нормально, однако я не вижу, как это сделать в XAML, так как не могу найти способ установить Path для свойства объекта ячейки.

Одна попытка XAML:

<Setter Property="FontStyle">
    <Setter.Value>
        <MultiBinding Converter="{StaticResource IsLockedToFontStyleConverter}" Mode="OneWay" UpdateSourceTrigger="PropertyChanged">
              <Binding />
              <Binding RelativeSource="{x:Static RelativeSource.Self}"/>
        </MultiBinding>
    </Setter.Value>
</Setter>

но нет привязки к свойству IsLocked

public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
{
    var row = (RowViewModel) values[0];
    var cell = (DataGridCell) values[1];
    if (cell != null && row != null)
    {
        var column = DataGridMethods.GetColumn(cell);
        return row[column].IsLocked ? "Italic" : "Normal";
    }

    return DependencyProperty.UnsetValue;
}

Обратите внимание, что предыдущая версия вернула строку [col].IsLocked и установила FontStyle с помощью DataTrigger, но возвращенный объект не привязан к данным.

Обратите внимание, конечно, что приложение не знает, что столбцы во время разработки.

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

Конечно, это общая проблема, и я являюсь новичком WPF, пытающимся использовать все MVVM в моем проекте, но эта проблема сдерживает меня в связи с использованием WPF DataGrid.

1 ответ

Решение

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

Поместите ссылку на объект ячейки в свойстве Tag DataGridCell. Я делаю это с помощью комбинации XAML и привязки кода внутри конвертера следующим образом:

   <Setter Property="Tag">
       <Setter.Value>
           <MultiBinding Converter="{StaticResource CellViewModelToTagConverter}" Mode="OneWay" UpdateSourceTrigger="PropertyChanged">
              <Binding />
              <Binding RelativeSource="{x:Static RelativeSource.Self}"/>
          </MultiBinding>
       </Setter.Value>
   </Setter>

а также

 public class CellViewModelToTagConverter : IMultiValueConverter
 {
     public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
     {
        var row = values[0] as RowViewModel;
        var cell = values[1] as DataGridCell;
        if (row != null && cell != null)
        {
            var column = DataGridMethods.GetColumn(cell);
            // hack within hack!!! (using tag way is itself a hack?)
            var b = new Binding("Self") {Source = row[column]};
            cell.SetBinding(FrameworkElement.TagProperty, b);
            //...
            //return row[column];
            return DependencyProperty.UnsetValue;
        }
        return DependencyProperty.UnsetValue;
    }

    public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
    {
        throw new NotSupportedException();
    }
}

Что я думаю об этом решении, вы можете сказать по моим комментариям внутри конвертера (мне пришлось добавить свойство Self к объекту Cell и сделать Self=this в конструкторе).

Тем не менее, это позволяет моему кодированию Datagrid быть полностью MVVM - если вы согласитесь, что то, что я сделал внутри конвертера, согласуется с MVVM. Во всяком случае, это работает!

Делая это таким образом, я могу видеть и управлять всем из XAML, например, управлять таким связыванием только для определенных столбцов, помещая XAML в соответствующие стили ячеек столбцов (это не делается с помощью DataGrid.CellStyle).

Во всяком случае, пример использования

<Style.Triggers>
      <DataTrigger Value="true" Binding="{Binding RelativeSource={RelativeSource Self}, Path=Tag.IsLocked}">
            <Setter Property="FontStyle" Value="Italic"/>
            <Setter Property="IsEnabled" Value="False"/>
       </DataTrigger>
 </Style.Triggers>

На уровне XAML это просто и ИМХО элегантно (особенно для различных всплывающих подсказок и всплывающих окон, для которых я интенсивно использую свойства объекта ячейки). Однако я уверен, что есть лучший способ сделать это, не так ли?

Надеюсь, что все это уходит, когда я могу использовать Net 4.0 и динамические объекты, но для этого проекта я не могу.

Другие вопросы по тегам