Определить порядок привязки DependencyProperties?

Что определяет порядок вычисления нескольких DepdencyProperties одного и того же элемента управления?

Я использую расширенный набор инструментов WPF PropertyGrid и имеет оба свойства SelectedObject и PropertyDefinitions:

<extToolkit:PropertyGrid AutoGenerateProperties="False" SelectedObject="{Binding ActiveDataPoint}" PropertyDefinitions="{Binding ActiveDataPoint.Properties}">

Проблема заключается в том, что OnSelectedObjectChanged запускается из свойства зависимостей, и в этом измененном обработчике он ссылается на PropertyDefinitions, который он видит как null. Если я комментирую обработчик OnSelectedObjectChanged, тогда я могу видеть, когда отладка вызывает OnPropertyDefinitionsChanged ПОСЛЕ вызова OnSelectedObjectChanged.

public static readonly DependencyProperty PropertyDefinitionsProperty = DependencyProperty.Register( "PropertyDefinitions", typeof( PropertyDefinitionCollection ), typeof( PropertyGrid ), new UIPropertyMetadata( null, OnPropertyDefinitionsChanged ) );
public PropertyDefinitionCollection PropertyDefinitions
{
  get
  {
    return ( PropertyDefinitionCollection )GetValue( PropertyDefinitionsProperty );
  }
  set
  {
    SetValue( PropertyDefinitionsProperty, value );
  }
}

private static void OnPropertyDefinitionsChanged(DependencyObject o, DependencyPropertyChangedEventArgs e)
{
    Console.Write("I changed!");
}

public static readonly DependencyProperty SelectedObjectProperty = DependencyProperty.Register( "SelectedObject", typeof( object ), typeof( PropertyGrid ), new UIPropertyMetadata( null, OnSelectedObjectChanged ) );
public object SelectedObject
{
  get
  {
    return ( object )GetValue( SelectedObjectProperty );
  }
  set
  {
    SetValue( SelectedObjectProperty, value );
  }
}

private static void OnSelectedObjectChanged( DependencyObject o, DependencyPropertyChangedEventArgs e )
{
  PropertyGrid propertyInspector = o as PropertyGrid;
  if( propertyInspector != null )
    propertyInspector.OnSelectedObjectChanged( ( object )e.OldValue, ( object )e.NewValue );
}

Проблема, с которой я столкнулась, обсуждается на этой теме форума, но я задаю более общий вопрос WPF о том, как я могу изменить порядок обновления этих свойств.

Я попытался иметь несколько вызовов NotifyPropertyChanged в разных порядках, но это, похоже, не влияет на это. Могу ли я заставить порядок отличаться или я должен просто изменить PropertyGrid, чтобы он работал для любого заказа?

Ответ 1

Короткий ответ заключается в том, что все это черный ящик, и вы не должны полагаться на то, что он оценивается до или после другого. Таким образом, наилучшим подходом было бы изменить PropertyGrid, чтобы он работал независимо от порядка, в котором установлены свойства.

Долгий ответ: похоже, это зависит от того, как указывается порядок привязки. Итак, вы можете сделать:

<extToolkit:PropertyGrid AutoGenerateProperties="False"
    PropertyDefinitions="{Binding ActiveDataPoint.Properties}"
    SelectedObject="{Binding ActiveDataPoint}"
    >

Вместо:

<extToolkit:PropertyGrid AutoGenerateProperties="False"
    SelectedObject="{Binding ActiveDataPoint}"
    PropertyDefinitions="{Binding ActiveDataPoint.Properties}"
    >

Опять же, было бы плохой практикой полагаться на это. И эта причуда может работать только при инициализации элемента управления. Изменения в ActiveDataPoint или DataContext позже могут привести к другому порядку.

Ответ 2

И еще один противоположный пример, чтобы подтвердить уже сказанное

... никогда не полагаться на порядок применяемых свойств

В пользовательском UserControl с определенными DependencyProperty -ies (.NET 4.5 и т.д.) - поскольку PropertyChangedCallbacks вызывается при инициализации...

фактический порядок определяется по порядку "определения кода позади" (статические поля)

... Я предполагаю, что это связано с порядком регистрации.

В некоторых других случаях порядок зависит от того, как свойства выстроены в XAML.