Привязка данных WPF: проблема CollectionViewSource и ObjectDataProvider - PullRequest
1 голос
/ 01 декабря 2010

У меня есть файл MainWindow.xaml:

<Window.Resources>

  <CollectionViewSource x:Key="cvs" 
    Source="{Binding Source={StaticResource ResourceKey=DetailsCollection}}" />

  <CollectionViewSource x:Key="DetailScopes">
    <CollectionViewSource.Source>
      <ObjectDataProvider 
        MethodName="GetValues" 
        ObjectType="{x:Type system:Enum}">
        <ObjectDataProvider.MethodParameters>
          <x:Type TypeName="entities:DetailScope" />
        </ObjectDataProvider.MethodParameters>
      </ObjectDataProvider>
    </CollectionViewSource.Source>
  </CollectionViewSource>

  <DataTemplate x:Key="AccountDetail"
    DataType="{x:Type entities:AccountDetail}">
    <DockPanel>
      <ComboBox 
        DockPanel.Dock="Left" 
        ItemsSource="{Binding Source={StaticResource ResourceKey=DetailScopes}}" 
        SelectedItem="{Binding Path=Scope}">
        <ComboBox.ItemTemplate>
          <DataTemplate>
            <TextBlock 
              Text="{Binding Converter={StaticResource DetailScopeConverter}}" />
          </DataTemplate>
        </ComboBox.ItemTemplate>
      </ComboBox>
      <TextBox Text="{Binding Path=Value}" />
    </DockPanel>
  </DataTemplate>

</Window.Resources>

...

<ListBox 
  ItemTemplate="{StaticResource ResourceKey=AccountDetail}" 
  ItemsSource="{Binding Source={StaticResource ResourceKey=cvs}}" />

и его класс code-behind, где я определил фильтр для областей детализации:

public class MainWindow
{
    public MainWindow()
    {
        CollectionViewSource detailScopes;

        InitializeComponent();

        // Attach filter to the collection view source
        detailScopes = this.Resources["DetailScopes"] as CollectionViewSource;
        detailScopes.Filter += new FilterEventHandler(DetailScopesFilter);

        private void DetailScopesFilter(object sender, FilterEventArgs e)
        {
            DetailScope scope;

            scope = (DetailScope)e.Item;
            if (scope == DetailScope.Private ||
                scope == DetailScope.Business)
            {
                e.Accepted = true;
            }
            else
            {
                e.Accepted = false;
            }
        }
    }
}

Далее есть AccountDetail class:

public class AccountDetail
{
  public string Value
  {
    get { return this.value; }
    set { this.value = value; }
  }
  public DetailScope Scope
  {
    get { return scope; }
    set { scope = value; }
  }

  private string value;
  private DetailScope scope;
}

Наконец, enum:

public enum DetailScope
{
  Private, 
  Business, 
  Other
}

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

Когда я извлекаю ObjectDataProvider из CollectionViewSource DetailScopes и привязываю его непосредственно к ItemsSource поля со списком в DataTemplate AccountDetail, проблема исчезает.Однако он мне нужен внутри CollectionViewSource, потому что я применяю к нему некоторую фильтрацию и не могу применить фильтрацию к ObjectDataProvider.

Может кто-нибудь объяснить, почему это происходит и как я на самом деле должен соединить CollectionViewSource и ObjectDataProvider?Спасибо.

1 Ответ

5 голосов
/ 01 декабря 2010

.

Проблема с вашим кодом заключается в том, что каждый ComboBox использует один и тот же экземпляр CollectionViewSource ;это означает, что ресурс с ключом «DetailScopes» является общим для всех ComboBox, поэтому, когда вы выбираете одно значение из определенного ComboBox, он автоматически выбирает одно и то же значение во всех ComboBox. Это потому, что базовая коллекция, которая является общей, отслеживает выбранный элемент и поскольку она изменяется при выборе из одного ComboBox, CollectionViewSource уведомляет об изменении ALL ComboBox .

Итак, решениеэто очень просто.Все, что вам нужно, чтобы ресурс DetailScopes нерушимый .

Вот исправление:

<!-- Please note this x:Shared="False" just after x:Key="DetailsScopes" --->

<CollectionViewSource x:Key="DetailScopes"  x:Shared="False"> 
    <CollectionViewSource.Source>
      <ObjectDataProvider 
        MethodName="GetValues" 
        ObjectType="{x:Type system:Enum}">
        <ObjectDataProvider.MethodParameters>
          <x:Type TypeName="entities:DetailScope" />
        </ObjectDataProvider.MethodParameters>
      </ObjectDataProvider>
    </CollectionViewSource.Source>
  </CollectionViewSource>

Надеюсь, это решит вашу проблему!

Однако этоРешение вызовет другую проблему.Позвольте мне кое-что процитировать из MSDN, чтобы вы поняли, что делает x: Shared.

x: Общий атрибут

Если задано значение false, изменяет поведение поиска ресурсов WPF так,который запрашивает атрибутированный ресурс , создает новый экземпляр для каждого запроса вместо совместного использования одного и того же экземпляра для всех запросов.

, поскольку x: Shared вызывает создание нового экземпляра (aновая копия) ресурса всякий раз, когда вы пытаетесь получить к нему доступ, это означает, что метод обработчика фильтра присоединяется только к экземпляру, который вы получаете в коде, а не ко всем экземплярам.

Итак, чтобыДля правильной работы вашего обработчика вам необходимо подключить обработчик из самого XAML, например:

<!-- Now please note Filter="DetailsScopesFilter" --->

<CollectionViewSource x:Key="DetailScopes"  x:Shared="False"  Filter="DetailScopesFilter"> 
    <CollectionViewSource.Source>
      <ObjectDataProvider 
        MethodName="GetValues" 
        ObjectType="{x:Type system:Enum}">
        <ObjectDataProvider.MethodParameters>
          <x:Type TypeName="entities:DetailScope" />
        </ObjectDataProvider.MethodParameters>
      </ObjectDataProvider>
    </CollectionViewSource.Source>
  </CollectionViewSource>

Надеюсь, он решит все ваши проблемы.Дайте мне знать, если вы все еще сталкиваетесь. :-)

Да, кстати, следующий код больше не нужен.Поэтому, пожалуйста, удалите его.

    // Attach filter to the collection view source
    detailScopes = this.Resources["DetailScopes"] as CollectionViewSource;
    detailScopes.Filter += new FilterEventHandler(DetailScopesFilter);

.

...