Назначить свойство с помощью ExpressionTree

Я обыгрываю идею передачи присвоения свойства методу в виде дерева выражения. Метод вызовет выражение, чтобы свойство было назначено правильно, а затем вынюхает имя свойства, которое только что было присвоено, чтобы я мог вызвать событие 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);
    }
}
13
задан Adam Rackis 24 May 2011 в 15:56
поделиться