Чтобы сгенерировать ошибку, выберите любой элемент в TopDataGrid
. В результате коллекция элементов будет загружена в BottomDataGrid
. Эта коллекция отсортирована по свойству Name
, как я указал! Затем выберите любой другой элемент в TopDataGrid
. В результате ItemsSource
из BottomDataGrid
будут перезагружены. И теперь коллекция несортированная! Коллекция выглядит так, как я указал в коде. Более того, если я проверю _customerView
отладчиком, то увижу отсортированную коллекцию.
Я знаю, что могу использовать List
с OrderBy
и INotifyPropertyChanged
, чтобы явно дать команду пользовательскому интерфейсу обновить себя вместо ObservableCollection
и ICollectionView
. Но я думаю, что это не правильный подход.
Вин 7, .Нет 4.0. Просто скопируйте и вставьте.
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="*"/>
<RowDefinition Height="*"/>
<RowDefinition Height="Auto"/>
</Grid.RowDefinitions>
<DataGrid Grid.Row="0"
AutoGenerateColumns="False"
ItemsSource="{Binding Items}"
SelectedItem="{Binding SelectedItem, Mode=TwoWay}"
CanUserAddRows="False"
CanUserDeleteRows="False"
CanUserResizeRows="False"
CanUserSortColumns="False"
SelectionMode="Single"
SelectionUnit="FullRow">
<DataGrid.Columns>
<DataGridTextColumn Binding="{Binding Name}"></DataGridTextColumn>
</DataGrid.Columns>
</DataGrid>
<DataGrid Grid.Row="1"
AutoGenerateColumns="False"
ItemsSource="{Binding SelectedItem.MyCollectionView}"
CanUserAddRows="False"
CanUserDeleteRows="False"
CanUserResizeRows="False"
CanUserSortColumns="False"
SelectionMode="Single"
SelectionUnit="FullRow">
<DataGrid.Columns>
<DataGridTextColumn Binding="{Binding Name}"></DataGridTextColumn>
<DataGridTextColumn Binding="{Binding Index}"></DataGridTextColumn>
</DataGrid.Columns>
</DataGrid>
<Grid Grid.Row="2">
<Grid.ColumnDefinitions>
<ColumnDefinition/>
<ColumnDefinition/>
</Grid.ColumnDefinitions>
<TextBox Grid.Column="0"
Text="{Binding Name1}"></TextBox>
<TextBox Grid.Column="1"
Text="{Binding Index}"></TextBox>
</Grid>
</Grid>
код
public class TopGridItem
{
private ObservableCollection<BottomGridItem> _collection;
public ObservableCollection<BottomGridItem> Collection
{
get { return _collection; }
}
public String Name { get; set; }
public ICollectionView MyCollectionView
{
get
{
ICollectionView _customerView = CollectionViewSource.GetDefaultView(Collection);
_customerView.SortDescriptions.Add(new SortDescription("Name", ListSortDirection.Ascending));
return _customerView;
}
}
public TopGridItem()
{
_collection = new ObservableCollection<BottomGridItem>();
_collection.Add(new BottomGridItem { Name = "bbbbbb" });
_collection.Add(new BottomGridItem { Name = "aaaaa" });
_collection.Add(new BottomGridItem { Name = "aaaaa" });
_collection.Add(new BottomGridItem { Name = "ccccc" });
_collection.Add(new BottomGridItem { Name = "dddddd" });
}
}
public class BottomGridItem
{
public String Name { get; set; }
public String Index { get; set; }
}
/// <summary>
/// Логика взаимодействия для NewWindow.xaml
/// </summary>
public partial class ProgressWindow : INotifyPropertyChanged
{
public TopGridItem _selectedItem;
public String Name1 { get; set; }
public String Index { get; set; }
public ObservableCollection<TopGridItem> Items { get; set; }
public TopGridItem SelectedItem
{
get { return _selectedItem; }
set
{
_selectedItem = value;
OnPropertyChanged("SelectedItem");
}
}
public ProgressWindow()
{
InitializeComponent();
DataContext = this;
Items = new ObservableCollection<TopGridItem>();
Items.Add(new TopGridItem {Name = "One"});
Items.Add(new TopGridItem {Name = "Two"});
Items.Add(new TopGridItem {Name = "Three"});
}
#region INotifyPropertyChanged
public event PropertyChangedEventHandler PropertyChanged;
[NotifyPropertyChangedInvocator]
protected virtual void OnPropertyChanged(string propertyName)
{
PropertyChangedEventHandler handler = PropertyChanged;
if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
}
#endregion
}
ОБНОВЛЕНИЕ
private void ClearSortDescriptionsOnItemsSourceChange()
{
this.Items.SortDescriptions.Clear();
this._sortingStarted = false;
List<int> descriptionIndices = this.GroupingSortDescriptionIndices;
if (descriptionIndices != null)
descriptionIndices.Clear();
foreach (DataGridColumn dataGridColumn in (Collection<DataGridColumn>) this.Columns)
dataGridColumn.SortDirection = new ListSortDirection?();
}
private static object OnCoerceItemsSourceProperty(DependencyObject d, object baseValue)
{
DataGrid dataGrid = (DataGrid) d;
if (baseValue != dataGrid._cachedItemsSource && dataGrid._cachedItemsSource != null)
dataGrid.ClearSortDescriptionsOnItemsSourceChange();
return baseValue;
}
Вроде как в ClearSortDescriptionsOnItemsSourceChange
методе сортировка очищается и не переопределяется заново. Я думаю, в этом дело.
_collection
только один раз, создайте экземплярListCollectionView
из_collection
в конструктореTopGridItem
(сSortDescriptions
) и всегда возвращать это вMyCollectionView
- person dkozl   schedule 29.01.2015DataGrid
. Я не могу воспроизвести проблему, если сетка данных отключена для чего-то вродеListBox
. Хотя я не знаю, почему это происходит. Вы определенно привязаны к использованиюDataGrid
для нижнего элемента управления? - person Steven Rands   schedule 29.01.2015