如何通过单击 Header 对带有 ObservableCollection 的 ListView 进行排序
How to sort ListView With ObservableCollection by Clicking Header
我是 WPF 初学者。我正在尝试通过单击列对 (asc/desc) listview
进行排序。我已将一个可观察的 collection 绑定到我的列表视图。我已经尝试了我在这里找到的东西 https://msdn.microsoft.com/en-us/library/ms745786(v=vs.110).aspx 出于某种原因它对我不起作用。我的代码在 link 这里几乎相同。不过我还是写到这里了,抱歉有点长。
我进行了一些搜索,许多答案已经准确地表明了我所做的。所以我想我在某个地方犯了一些错误。
这是 WPF:
<ListView Grid.Row="3" Name="lvPurchaseSummaryList" ItemsSource="{Binding purchaseSummaryCol, Mode=Default}" Width="1700" HorizontalAlignment="Left" GridViewColumnHeader.Click="purchaseSummaryListColumn_click">
<ListView.View>
<GridView>
<GridViewColumn Width="100" DisplayMemberBinding="{Binding Path=originalName}" Header="Original Name" />
<GridViewColumn Width="100" DisplayMemberBinding="{Binding Path=turkishName}" Header="Local Name"/>
// List goes on like this...
</GridView>
</ListView.View>
</ListView>
这里是class(几乎和msdn一样link):
public partial class PurchaseSummaryPage : Page
{
GridViewColumnHeader _lastHeaderClicked = null;
ListSortDirection _lastDirection = ListSortDirection.Ascending;
public PurchaseSummaryPage()
{
InitializeComponent();
}
void purchaseSummaryListColumn_click(object sender, RoutedEventArgs e)
{
GridViewColumnHeader headerClicked = e.OriginalSource as GridViewColumnHeader;
ListSortDirection direction;
if (headerClicked != null)
{
if (headerClicked.Role != GridViewColumnHeaderRole.Padding)
{
if (headerClicked != _lastHeaderClicked)
{
direction = ListSortDirection.Ascending;
}
else
{
if (_lastDirection == ListSortDirection.Ascending)
{
direction = ListSortDirection.Descending;
}
else
{
direction = ListSortDirection.Ascending;
}
}
string header = headerClicked.Column.Header as string;
Sort(header, direction);
if (direction == ListSortDirection.Ascending)
{
headerClicked.Column.HeaderTemplate =
Resources["HeaderTemplateArrowUp"] as DataTemplate;
}
else
{
headerClicked.Column.HeaderTemplate =
Resources["HeaderTemplateArrowDown"] as DataTemplate;
}
// Remove arrow from previously sorted header
if (_lastHeaderClicked != null && _lastHeaderClicked != headerClicked)
{
_lastHeaderClicked.Column.HeaderTemplate = null;
}
_lastHeaderClicked = headerClicked;
_lastDirection = direction;
}
}
}
private void Sort(string sortBy, ListSortDirection direction)
{
ICollectionView dataView =
CollectionViewSource.GetDefaultView(lvPurchaseSummaryList.ItemsSource);
dataView.SortDescriptions.Clear();
SortDescription sd = new SortDescription(sortBy, direction);
dataView.SortDescriptions.Add(sd);
dataView.Refresh();
}
}
这是来自主要 window 的一些示例...
public ObservableCollection<PurchaseSummary> purchaseSummaryCol { get; set; }
public MainWindow()
{
InitializeComponent();
purchaseSummaryCol = new ObservableCollection<PurchaseSummary>();
purchaseSummaryCol.Add(new PurchaseSummary("100", "basd", "asd", "basd", "asd", "basd", "asd", "basd", "asd", new Pay(DateTime.Today, DateTime.Today, 3), new PayRestrict(), new Vod()));
purchaseSummaryCol.Add(new PurchaseSummary("105", "asd", "basd", "asd", "basd", "asd", "basd", "asd", "basd", new Pay(DateTime.Today, DateTime.Today, 3), new PayRestrict(), new Vod()));
purchaseSummaryCol.Add(new PurchaseSummary("101", "casd", "cbasd", "casd", "cbasd", "casd", "cbasd", "casd", "cbasd", new Pay(DateTime.Today, DateTime.Today, 3), new PayRestrict(), new Vod()));
PurchaseSummaryPage purchaseSummaryPage = new PurchaseSummaryPage();
purchaseSummaryPage.DataContext = this;
AttachPageToFrame(_f, purchaseSummaryPage);
}
原因来自这段代码:
string header = headerClicked.Column.Header as string;
Sort(header, direction);
它使用 header 文本,在您的情况下它与 属性 名称不同,简单的修复方法是尝试将 Header 设置为与 [=25= 相同] 姓名 Header="originalName"
为了更好的解决方案,可以将需要排序的propertyName存储在Header的Tag中,但是我们需要做的更多的是:
<GridViewColumn Width="100" DisplayMemberBinding="{Binding Path=originalName}" >
<GridViewColumn.Header>
<TextBlock Text="Original Name" Tag="originalName"/>
</GridViewColumn.Header>
</GridViewColumn>
然后在代码中:
GridViewColumnHeader headerClicked = e.OriginalSource as GridViewColumnHeader;
var tb = headerClicked.Column.Header as TextBlock;
var propertyName = tb.Tag as string;
Sort(propertyName, direction);
我是 WPF 初学者。我正在尝试通过单击列对 (asc/desc) listview
进行排序。我已将一个可观察的 collection 绑定到我的列表视图。我已经尝试了我在这里找到的东西 https://msdn.microsoft.com/en-us/library/ms745786(v=vs.110).aspx 出于某种原因它对我不起作用。我的代码在 link 这里几乎相同。不过我还是写到这里了,抱歉有点长。
我进行了一些搜索,许多答案已经准确地表明了我所做的。所以我想我在某个地方犯了一些错误。
这是 WPF:
<ListView Grid.Row="3" Name="lvPurchaseSummaryList" ItemsSource="{Binding purchaseSummaryCol, Mode=Default}" Width="1700" HorizontalAlignment="Left" GridViewColumnHeader.Click="purchaseSummaryListColumn_click">
<ListView.View>
<GridView>
<GridViewColumn Width="100" DisplayMemberBinding="{Binding Path=originalName}" Header="Original Name" />
<GridViewColumn Width="100" DisplayMemberBinding="{Binding Path=turkishName}" Header="Local Name"/>
// List goes on like this...
</GridView>
</ListView.View>
</ListView>
这里是class(几乎和msdn一样link):
public partial class PurchaseSummaryPage : Page
{
GridViewColumnHeader _lastHeaderClicked = null;
ListSortDirection _lastDirection = ListSortDirection.Ascending;
public PurchaseSummaryPage()
{
InitializeComponent();
}
void purchaseSummaryListColumn_click(object sender, RoutedEventArgs e)
{
GridViewColumnHeader headerClicked = e.OriginalSource as GridViewColumnHeader;
ListSortDirection direction;
if (headerClicked != null)
{
if (headerClicked.Role != GridViewColumnHeaderRole.Padding)
{
if (headerClicked != _lastHeaderClicked)
{
direction = ListSortDirection.Ascending;
}
else
{
if (_lastDirection == ListSortDirection.Ascending)
{
direction = ListSortDirection.Descending;
}
else
{
direction = ListSortDirection.Ascending;
}
}
string header = headerClicked.Column.Header as string;
Sort(header, direction);
if (direction == ListSortDirection.Ascending)
{
headerClicked.Column.HeaderTemplate =
Resources["HeaderTemplateArrowUp"] as DataTemplate;
}
else
{
headerClicked.Column.HeaderTemplate =
Resources["HeaderTemplateArrowDown"] as DataTemplate;
}
// Remove arrow from previously sorted header
if (_lastHeaderClicked != null && _lastHeaderClicked != headerClicked)
{
_lastHeaderClicked.Column.HeaderTemplate = null;
}
_lastHeaderClicked = headerClicked;
_lastDirection = direction;
}
}
}
private void Sort(string sortBy, ListSortDirection direction)
{
ICollectionView dataView =
CollectionViewSource.GetDefaultView(lvPurchaseSummaryList.ItemsSource);
dataView.SortDescriptions.Clear();
SortDescription sd = new SortDescription(sortBy, direction);
dataView.SortDescriptions.Add(sd);
dataView.Refresh();
}
}
这是来自主要 window 的一些示例...
public ObservableCollection<PurchaseSummary> purchaseSummaryCol { get; set; }
public MainWindow()
{
InitializeComponent();
purchaseSummaryCol = new ObservableCollection<PurchaseSummary>();
purchaseSummaryCol.Add(new PurchaseSummary("100", "basd", "asd", "basd", "asd", "basd", "asd", "basd", "asd", new Pay(DateTime.Today, DateTime.Today, 3), new PayRestrict(), new Vod()));
purchaseSummaryCol.Add(new PurchaseSummary("105", "asd", "basd", "asd", "basd", "asd", "basd", "asd", "basd", new Pay(DateTime.Today, DateTime.Today, 3), new PayRestrict(), new Vod()));
purchaseSummaryCol.Add(new PurchaseSummary("101", "casd", "cbasd", "casd", "cbasd", "casd", "cbasd", "casd", "cbasd", new Pay(DateTime.Today, DateTime.Today, 3), new PayRestrict(), new Vod()));
PurchaseSummaryPage purchaseSummaryPage = new PurchaseSummaryPage();
purchaseSummaryPage.DataContext = this;
AttachPageToFrame(_f, purchaseSummaryPage);
}
原因来自这段代码:
string header = headerClicked.Column.Header as string;
Sort(header, direction);
它使用 header 文本,在您的情况下它与 属性 名称不同,简单的修复方法是尝试将 Header 设置为与 [=25= 相同] 姓名 Header="originalName"
为了更好的解决方案,可以将需要排序的propertyName存储在Header的Tag中,但是我们需要做的更多的是:
<GridViewColumn Width="100" DisplayMemberBinding="{Binding Path=originalName}" >
<GridViewColumn.Header>
<TextBlock Text="Original Name" Tag="originalName"/>
</GridViewColumn.Header>
</GridViewColumn>
然后在代码中:
GridViewColumnHeader headerClicked = e.OriginalSource as GridViewColumnHeader;
var tb = headerClicked.Column.Header as TextBlock;
var propertyName = tb.Tag as string;
Sort(propertyName, direction);