ReactiveUI 6.0 и привязка WinForms

Теперь, когда ReactiveUI 6.0 был выпущен, у меня возник вопрос к сообществу: каков наилучший или наиболее эффективный способ связывания ReactiveObject (s) и Windows Form (s).

Вот что у меня так далеко:

Моя модель

namespace WindowsFormsApplication1
{
    #region

    using System;
    using System.Reactive.Linq;

    using ReactiveUI;

    #endregion

    public class ClockModel : ReactiveObject
    {
        #region Fields

        private DateTime currentDateTime;

        #endregion

        #region Constructors and Destructors

        public ClockModel() { Observable.Interval(TimeSpan.FromSeconds(1)).Subscribe(s => this.CurrentDateTime = DateTime.Now); }

        #endregion

        #region Public Properties

        public DateTime CurrentDateTime
        {
            get { return currentDateTime; }
            set { this.RaiseAndSetIfChanged(ref currentDateTime, value); }
        }

        #endregion
    }
}

Моя единственная форма

using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    public partial class Form1 : Form
    {
        readonly ClockModel model = new ClockModel();
        public Form1()
        {
            InitializeComponent();
            this.label1.DataBindings.Add("Text", model, "CurrentDateTime");
        }
    }
}

Это работает, но я не в восторге от ссылки на свойства с каждой стороны по их строковым именам. Есть ли лучший способ сделать это?

ОБНОВЛЕНИЕ 1

Учитывая это расширение:

public class NameOf<T>
{
    public static string Property<TProp>(Expression<Func<T, TProp>> expression)
    {
        var body = expression.Body as MemberExpression;
        if (body == null)
            throw new ArgumentException("'expression' should be a member expression");
        return body.Member.Name;
    }
}

Является ли эта форма более желательной, чем предыдущая?

this.label1.DataBindings.Add(NameOf<Form1>.Property(e => e.label1.Text), model, NameOf<ClockModel>.Property(p => p.CurrentDateTime));

ОБНОВЛЕНИЕ 2

Еще одно расширение:

public static class Extensions
{
    public static string GetPropertyName<T,TProp>(this T obj, Expression<Func<T, TProp>> expression)
    {
        var body = expression.Body as MemberExpression;
        if (body == null)
            throw new ArgumentException("'expression' should be a member expression");
        return body.Member.Name;
    }
}

Теперь я могу использовать:

this.label1.DataBindings.Add(this.GetPropertyName(e => e.label1.Text), model, model.GetPropertyName(p => p.CurrentDateTime));

или это:

this.label2.DataBindings.Add(this.label2.GetPropertyName(e => e.Text), model, model.GetPropertyName(p => p.Ticks));

Мне начинает нравиться этот, но что другие думают о любом из трех подходов?

Технически говоря, Форма стала Представлением и Контроллером или Представителем, в то время как Модель все еще отделена и не знает своих потребителей. Будет ли это предпочтительным подходом?

ОБНОВЛЕНИЕ 3

Это может вызвать у некоторых пуристов легкий сердечный приступ... На основании совета Пола Беттса:

namespace WindowsFormsApplication1
{
    #region

    using System.Windows.Forms;

    using ReactiveUI;

    #endregion

    public partial class Form1 : Form, IViewFor<ClockModel>
    {
        #region Constructors and Destructors

        public Form1()
        {
            InitializeComponent();
            this.ViewModel = new ClockModel();
            this.Bind(ViewModel, vm => vm.CurrentDateTime, v => v.label1.Text);
            this.Bind(ViewModel, vm => vm.Ticks, v => v.label2.Text);
        }

        #endregion

        #region Public Properties

        public ClockModel ViewModel { get; set; }

        #endregion

        #region Explicit Interface Properties

        object IViewFor.ViewModel
        {
            get { return ViewModel; }
            set { ViewModel = value as ClockModel; }
        }

        #endregion
    }
}

Интересное изменение в поведении: до обновления 3 время отображалось с секундами. Я могу только предположить, что это происходит из-за конвертера ReactiveUI по умолчанию, используемого в третьем подходе. Я должен узнать больше об этом.

public class DateTimeStringConverter : IBindingTypeConverter
{
    public int GetAffinityForObjects(Type fromType, Type toType) { return 1; }

    public bool TryConvert(object @from, Type toType, object conversionHint, out object result) { 
        result = ((DateTime)@from).ToString("F");
        return true;
    }
}

Мне нравится элегантность подхода.

Есть другие мнения?

1 ответ

Решение

Самый простой способ - использовать привязки ReactiveUI - реализовать IViewFor<TViewModel> в вашей форме, тогда вы получите несколько новых методов, таких как Bind а также OneWayBind,

Зайдите на https://github.com/reactiveui/ReactiveUI/blob/docs/docs/basics/bindings.md для получения дополнительной информации.

Другие вопросы по тегам