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 для получения дополнительной информации.