m-y*_*m-y 5 c# data-binding wpf inotifypropertychanged
我有一个带有组合框和文本框的WPF表单(两者都是数据绑定到Object的属性).更改组合框或文本框输入会更新Object的属性,数据绑定将启动并更新UI.问题是,我实现了一种取消更改的方法,这种方式有效,但却搞砸了UI更新.如果我从组合框中进行更改并取消它,组合框不会将选定的值恢复为应该的值(由对象的值绑定).如果我从文本框进行更改并取消它,文本框和组合框都显示正确的数据,但然后焦点立即被给予组合框(当它应该留在文本框上,因为那是我的最后一个地方它).一世'
//User.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Text;
namespace MyTesting
{
public class User : AbstractEntity
{
public User()
{
Rankings = new Dictionary<int,string>();
Rankings.Add(1, "Newbie");
Rankings.Add(10, "Novice");
Rankings.Add(25, "Adept User");
Rankings.Add(50, "Power User");
Rankings.Add(100, "Admin God");
}
public Dictionary<Int32, String> Rankings { get; set; }
private Int32 _rank;
public Int32 Rank
{
get
{
return _rank;
}
set
{
SetProperty<Int32>("Rank", ref _rank, value);
}
}
}
}
//AbstractEntity.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Text;
namespace MyTesting
{
public abstract class AbstractEntity : INotifyPropertyChanging, INotifyPropertyChanged
{
protected void SetProperty<T>(String propertyName, ref T property, T value)
{
if (!Object.Equals(property, value))
{
if (OnPropertyChanging(propertyName, property, value))
{
T oldValue = (T)property;
property = value;
OnPropertyChanged(propertyName, property, value);
}
}
}
[field: NonSerialized]
public event PropertyChangingEventHandler PropertyChanging;
protected virtual Boolean OnPropertyChanging(String propertyName, Object oldValue = null, Object newValue = null)
{
CancellablePropertyChangingEventArgs e;
if ((oldValue != null) || (newValue != null))
e = new CancellablePropertyChangingEventArgs(propertyName, oldValue, newValue);
else
e = new CancellablePropertyChangingEventArgs(propertyName);
return OnPropertyChanging(e);
}
protected virtual Boolean OnPropertyChanging(CancellablePropertyChangingEventArgs e)
{
if (PropertyChanging != null)
PropertyChanging(this, e as PropertyChangingEventArgs);
return !e.IsCancelled;
}
[field: NonSerialized]
public event PropertyChangedEventHandler PropertyChanged;
protected virtual void OnPropertyChanged(String propertyName, Object oldValue = null, Object newValue = null)
{
ExtendedPropertyChangedEventArgs e;
if ((oldValue != null) || (newValue != null))
e = new ExtendedPropertyChangedEventArgs(propertyName, oldValue, newValue);
else
e = new ExtendedPropertyChangedEventArgs(propertyName);
OnPropertyChanged(e);
}
protected virtual void OnPropertyChanged(ExtendedPropertyChangedEventArgs e)
{
if (PropertyChanged != null)
PropertyChanged(this, e as PropertyChangedEventArgs);
}
}
public class ExtendedPropertyChangedEventArgs : PropertyChangedEventArgs
{
public ExtendedPropertyChangedEventArgs(String propertyName)
: base(propertyName)
{
}
public ExtendedPropertyChangedEventArgs(String propertyName, Object oldValue, Object newValue)
: base(propertyName)
{
OldValue = oldValue;
NewValue = newValue;
}
public Object OldValue { get; private set; }
public Object NewValue { get; private set; }
}
public class CancellablePropertyChangingEventArgs : PropertyChangingEventArgs
{
public CancellablePropertyChangingEventArgs(String propertyName, Boolean cancel = false)
: base(propertyName)
{
IsCancelled = cancel;
}
public CancellablePropertyChangingEventArgs(String propertyName, Object oldValue, Object newValue, Boolean cancel = false)
: base(propertyName)
{
OldValue = oldValue;
NewValue = newValue;
IsCancelled = cancel;
}
public Object OldValue { get; private set; }
public Object NewValue { get; private set; }
public Boolean IsCancelled { get; set; }
}
}
<!-- MainWindow.xaml -->
<Window x:Class="ObservableDictionaryBinding.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:src="clr-namespace:MyTesting"
Title="MainWindow" Height="350" Width="525" Loaded="OnLoaded">
<Grid>
<ComboBox x:Name="RankList" Height="23" HorizontalAlignment="Stretch" VerticalAlignment="Top" Margin="12,12,12,0" />
<TextBlock Height="23" Width="40" HorizontalAlignment="Left" VerticalAlignment="Top" Margin="13,100,0,0" Text="Rank:" />
<TextBox x:Name="RankBox" Height="23" HorizontalAlignment="Stretch" VerticalAlignment="Top" Margin="59,97,12,0" />
</Grid>
</Window>
//MainWindow.xaml.cs
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
namespace MyTesting
{
public partial class MainWindow : Window
{
public MainWindow()
{
MyUser = new User();
InitializeComponent();
MyUser.PropertyChanging += new PropertyChangingEventHandler(MyUser_PropertyChanging);
}
private User MyUser { get; set; }
private Binding RankListBinding { get; set; }
private Binding RankBinding { get; set; }
private Binding RankListRankBinding { get; set; }
private void OnLoaded(object sender, EventArgs e)
{
DataContext = MyUser;
RankListBinding = new Binding("Rankings");
RankListBinding.Source = MyUser;
RankList.SetBinding(ComboBox.ItemsSourceProperty, RankListBinding);
RankList.SelectedValuePath = "Key";
RankList.DisplayMemberPath = "Value";
RankBinding = new Binding("Rank");
RankBinding.Source = MyUser;
RankBox.SetBinding(TextBox.TextProperty, RankBinding);
RankListRankBinding = new Binding("Rank");
RankListRankBinding.Source = MyUser;
RankList.SetBinding(ComboBox.SelectedValueProperty, RankListRankBinding);
}
private void MyUser_PropertyChanging(Object sender, PropertyChangingEventArgs e)
{
CancellablePropertyChangingEventArgs ea = e as CancellablePropertyChangingEventArgs;
String text = String.Format("Would you like to change the property '{0}' from '{1}' to '{2}'?",
e.PropertyName,
(ea.OldValue == null) ? "<null>" : ea.OldValue.ToString(),
(ea.NewValue == null) ? "<null>" : ea.NewValue.ToString()
);
MessageBoxResult result = MessageBox.Show(this, text, "Property Changed",
MessageBoxButton.YesNo, MessageBoxImage.Question, MessageBoxResult.Yes);
if (result == MessageBoxResult.No)
ea.IsCancelled = true;
}
}
}
Run Code Online (Sandbox Code Playgroud)
更新的方法:这修复了绑定,但没有解决当用户尝试更改文本框中的值然后取消它时组合框被焦点窃取的问题.但是,至少UI在数据绑定值方面匹配.我找到了这个帮助我的链接.
protected void SetProperty<T>(String propertyName, ref T property, T value)
{
if (!Object.Equals(property, value))
{
bool cancelled = OnPropertyChanging<T>(propertyName, property, value);
if (cancelled)
{
Application.Current.Dispatcher.BeginInvoke(
new Action(() =>
{
OnPropertyChanged<T>(propertyName);
}),
DispatcherPriority.ContextIdle,
null
);
return;
}
T originalValue = property;
property = value;
OnPropertyChanged(propertyName, originalValue, property);
}
}
Run Code Online (Sandbox Code Playgroud)
这解决了 UI 显示正确的数据绑定数据的问题……它只是没有解决焦点被盗的问题:
protected void SetProperty<T>(String propertyName, ref T property, T value)
{
if (!Object.Equals(property, value))
{
bool cancelled = OnPropertyChanging<T>(propertyName, property, value);
if (cancelled)
{
Application.Current.Dispatcher.BeginInvoke(
new Action(() =>
{
OnPropertyChanged<T>(propertyName);
}),
DispatcherPriority.ContextIdle,
null
);
return;
}
T originalValue = property;
property = value;
OnPropertyChanged(propertyName, originalValue, property);
}
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
3199 次 |
| 最近记录: |