Я обыгрываю идею передачи присвоения свойства методу в виде дерева выражения. Метод вызовет выражение, чтобы свойство было назначено правильно, а затем вынюхает имя свойства, которое только что было присвоено, чтобы я мог вызвать событие PropertyChanged. Идея состоит в том, что я хотел бы иметь возможность использовать тонкие автоматические свойства в моих моделях представления WPF и при этом иметь запускаемое событие PropertyChanged.
Я невежественен с ExpressionTrees, поэтому я надеюсь, что кто-то может указать мне в правильном направлении:
public class ViewModelBase {
public event Action<string> PropertyChanged = delegate { };
public int Value { get; set; }
public void RunAndRaise(MemberAssignment Exp) {
Expression.Invoke(Exp.Expression);
PropertyChanged(Exp.Member.Name);
}
}
Проблема в том, что я не знаю, как это назвать. Эта наивная попытка была отклонена компилятором по причинам, которые, я уверен, будут очевидны любому, кто сможет на это ответить:
ViewModelBase vm = new ViewModelBase();
vm.RunAndRaise(() => vm.Value = 1);
ИЗМЕНИТЬ
Спасибо @svick за прекрасный ответ. Я переместил одну мелочь и превратил ее в метод расширения. Вот полный пример кода с модульным тестом:
[TestClass]
public class UnitTest1 {
[TestMethod]
public void TestMethod1() {
MyViewModel vm = new MyViewModel();
bool ValuePropertyRaised = false;
vm.PropertyChanged += (s, e) => ValuePropertyRaised = e.PropertyName == "Value";
vm.SetValue(v => v.Value, 1);
Assert.AreEqual(1, vm.Value);
Assert.IsTrue(ValuePropertyRaised);
}
}
public class ViewModelBase : INotifyPropertyChanged {
public event PropertyChangedEventHandler PropertyChanged = delegate { };
public void OnPropertyChanged(string propertyName) {
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
public class MyViewModel : ViewModelBase {
public int Value { get; set; }
}
public static class ViewModelBaseExtension {
public static void SetValue<TViewModel, TProperty>(this TViewModel vm, Expression<Func<TViewModel, TProperty>> exp, TProperty value) where TViewModel : ViewModelBase {
var propertyInfo = (PropertyInfo)((MemberExpression)exp.Body).Member;
propertyInfo.SetValue(vm, value, null);
vm.OnPropertyChanged(propertyInfo.Name);
}
}