C# WPF Datagrid 不会在数据更新时动态排序
我的数据网格遇到了一些问题:
当我更新一些数据(来自模型)时,它会显示在我的数据网格上,但是如果单击标题对列进行排序,当我更新现有数据时,它会开始横向移动。
这是 2 1 示例s :
-
如果我添加新值,它不会出现在末尾(就像我不对数据网格进行排序时那样) )但它显示在错误的位置(每次都显示在同一位置)。 - 如果我更新现有值,则在需要时顺序永远不会改变。
我看过多个答案,但有些人说 DataGridTextColumn 不应该是问题... 所以我想知道这是否是因为字典...
我知道新数据的第一个问题与字典有关。
目的。
public class Player : INotifyPropertyChanged
{
private string _id;
private string _name;
private int _damage;
private int _heal;
private int _dps;
private int _hps;
private int _time = 1;
public Player(string id, string name)
{
_name = name;
_id = id;
}
public event PropertyChangedEventHandler PropertyChanged;
public string Id
{
get { return _id; }
private set
{
_id = value;
}
}
public string Name
{
get { return _name; }
private set
{
_name = value;
NotifyPropertyChanged("Name");
}
}
public int Damage
{
get { return _damage; }
set
{
_damage = value;
NotifyPropertyChanged("Damage");
Dps = _damage / _time;
}
}
public int Heal
{
get { return _heal; }
set
{
_heal = value;
NotifyPropertyChanged("Heal");
Hps = _heal / _time;
}
}
public int Dps
{
get { return _dps; }
private set
{
_dps = value;
NotifyPropertyChanged("Dps");
}
}
public int Hps
{
get {return _hps; }
private set
{
_hps = value;
NotifyPropertyChanged("Hps");
}
}
public int Time
{
get { return _time; }
set
{
_time = value;
Dps = _damage / _time;
Hps = _heal / _time;
}
}
private void NotifyPropertyChanged(string name)
{
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs(name));
}
}
ObservableCollection
public sealed class ShortList
{
private static readonly ShortList instance = new ShortList();
private ObservableCollection<Player> playerList = new ObservableCollection<Player>();
private ShortList()
{
}
public static ShortList getShortList
{
get
{
return instance;
}
}
public ObservableCollection<Player> getPlayerList
{
get
{
return playerList;
}
}
public void updatePlayer(string id, string name, int damage, int heal, int time)
{
Player player;
player = findPlayer(id);
if (player != null)
{
player.Damage = player.Damage + damage;
player.Heal = player.Heal + heal;
player.Time = player.Time + time;
}
else
{
player = new Player(id, name);
player.Damage = damage;
player.Heal = heal;
player.Time = time;
playerList.Add(player);
}
}
public void clear()
{
playerList.Clear();
}
private Player findPlayer(string id)
{
foreach (Player p in playerList)
{
if (p.Id == id)
{
return p;
}
}
return null;
}
}
XAML
<DataGrid AutoGenerateColumns="False"Name="playerDataGrid" IsReadOnly="True" ItemsSource="{Binding}">
<DataGrid.Columns>
<DataGridTextColumn Header="Nom" Binding="{Binding Name}" MinWidth="35"/>
<DataGridTextColumn Header="Degats" Binding="{Binding Damage}" MinWidth="45" />
<DataGridTextColumn Header="DPS" Binding="{Binding Dps}" MinWidth="29" />
<DataGridTextColumn Header="Soins" Binding="{Binding Heal}" MinWidth="35" />
<DataGridTextColumn Header="HPS" Binding="{Binding Hps}" MinWidth="29" />
</DataGrid.Columns>
</DataGrid>
窗口背后的代码
public partial class MiniParser : Window
{
public MiniParser()
{
InitializeComponent();
playerDataGrid.ItemsSource = ShortList.getShortList.getPlayerList;
temporyFill();
}
private void temporyFill()
{
ShortList.getShortList.updatePlayer("1234", "ABCD", 100, 0, 2);
ShortList.getShortList.updatePlayer("1234", "ABCD", 100, 0, 0);
ShortList.getShortList.updatePlayer("123", "ABC", 50, 0, 1);
ShortList.getShortList.updatePlayer("234", "ABC", 0, 50, 1);
ShortList.getShortList.updatePlayer("345", "BCD", 1000, 25, 25);
ShortList.getShortList.updatePlayer("456", "CDE", 250, 0, 25);
}
private void startMI_Click(object sender, RoutedEventArgs e)
{
ShortList.getShortList.updatePlayer("5678", "BABA", 100, 100, 100);
ShortList.getShortList.updatePlayer("1234", "ABCD", 100, 0, 0);
}
}
当然,背后的大部分代码都是为了测试目的...但想法是模型正在更新,视图需要反映更改(甚至排序)。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
当属性更新时,您不会引发
PropertyChanged
事件。这意味着name
从""
更改为"Test"
的新项目或发生更改的现有项目不会引发 PropertyChanged 事件,让 UI 知道值已更改并且需要更新。要修复此问题,请让您的属性在更改时引发 PropertyChanged 事件。
另外,我正在重新阅读您的问题,我不知道您在使用
ObservableDictionary
做什么,但我建议更改它,以便您的 DataGrid 直接绑定到ObservableCollection 。
。我怀疑这是你问题的一部分。You're not raising the
PropertyChanged
event when a property updates. This means that a new item which hasname
changed from""
to"Test"
, or an existing item that changes, is not going to raise a PropertyChanged event to let the UI know a value has changed and it needs to update.To fix it, make your properties raise the PropertyChanged event when they change.
Also, I was re-reading your question and I have no idea what you're doing with
ObservableDictionary
, but I'd recommend changing that so your DataGrid binds directly to anObservableCollection<Player>
. I suspect that's part of your problem.我找到了一个解决方案,到目前为止还不错:
问题是当您更改已存在于 ObservableCollection 中的对象中的字段值时,ObservableCollection 不会触发事件“CollectionChanged”。 (可能是因为您没有更改对它的引用)。
这是一个例子:
正如您可以猜到的,当我更改现有对象的字段时,CollectionChanged 没有触发第二部分...
所以我实现的解决方案如下:
您所要做的就是创建新类型集合,然后当您更改现有对象的字段值时,调用 UpdateCollection() 方法。
这个解法来自吴雪松。
我想说雷切尔也给了我很大的帮助。
I found a solution, so far so good :
The problem was that ObservableCollection doesn't trigger the event "CollectionChanged" when you change the value of a field in an object which is already in your ObservableCollection. (Possibly because you don't change the reference to it).
Here's an example :
As you can guess, the second part when I changed a field of my existing object the CollectionChanged didn't trigger...
So the solution I implemented is the following :
All you have to do is create the new type of collection and then when you change a field value of an existing object, call the UpdateCollection() method.
This solution is from Wu Xuesong.
I'd like to say Rachel as been also a big help.
将如下所示的任何代码移至
属性的 setter 方法中。这就是设置器的用途。
另外,我同意建议您使用
ObservableCollection
而不是ObservableDictionary
的答案。它们对于 WPF 绑定非常常见。Move any code that looks like this:
into the setter methods of your properties. That's what the setters are there for.
Also I second the answer suggestig you use an
ObservableCollection<T>
instead of yourObservableDictionary<TKey,TValue>
. They are very common for WPF bindings.