1. Как разрабатывать UI XAML приложений
(Prism) и веб-приложений (Angular2,
React) без Message Bus
Денис Цветцих
One Systems
DotNext Msk
9 декабря 2016
http://dotnext-moscow.ru/
2. 2
Message Bus – паттерн для интеграции
MessageBus
Application 1 Application 2
Application 3 Application 4
3. 3
Чем плох MessageBus на UI
• Взаимодействие объектов неявное
• Сложно дебажить
• В больших проектах MessageHell
• Простор для костылей
• Messages вместо Events
• Messages вместо Bindings
5. 5
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
6. 6
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
7. 7
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
8. 8
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
9. 9
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
10. 10
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
11. 11
Виноват Composite Application Blocks
public class MasterController : Controller
{
[EventPublication("GlobalEvent")]
public event EventHandler GlobalEvent;
}
public class DetailController : Controller
{
[EventSubscription("GlobalEvent")]
public void OnGlobalEvent(object sender, EventArgs e)
{
}
}
12. 12
CAB UI: скрытая угроза
• Нет класса EventAggregator
• Но
• Любой контроллер может подписаться на
• Любое событие
• Любого другого контроллера
• Неявно (при помощи атрибута)
13. 13
Message Bus есть во всех MVVM
• Prism – по инерции от Composite App Blocks
• Caliburn.Micro – для поддержки View-First и ViewModel-First
• MVVM от энтузиастов
• Не знают как по-другому
• Делают как у всех
14. 14
О чем мы поговорим?
• Типовые UI-задачи для MessageBus
• UI Composition
• Смена контекста
• Костыли
• Как без него обойтись
• MVVM-фреймворки для WPF и UWP без MessageBus
• Что c MessageBus на JS (Angular2, React)
15. 15
Опрос
• Кто использовал MessageBus на UI?
• И считает что без него обойтись нельзя
• И считает, что без него использовать можно
• Кто не использовал MessageBus на UI?
16. 16
Примеры с INPC кодом
public class DetailViewModel : BindableBase
{
public User User
{
get { return _user; }
set { SetProperty(ref _user, value); }
}
}
24. 24
View-First: по вьюхе создаем ViewModel
public partial class MasterView
{
[Import]
public MasterViewModel ViewModel
{
set { DataContext = value; }
}
}
27. 27
MasterViewModel
public class MasterViewModel
{
public User SelectedUser
{
get { return _selectedUser; }
set
{
SetProperty(ref _selectedUser, value);
EventAggregator.GetEvent<UserChangedMessage>()
.Publish(_selectedUser);
}
}
public ObservableCollection<User> Users { get; set; }
}
28. 28
MasterViewModel
public class MasterViewModel
{
public User SelectedUser
{
get { return _selectedUser; }
set
{
SetProperty(ref _selectedUser, value);
EventAggregator.GetEvent<UserChangedMessage>()
.Publish(_selectedUser);
}
}
public ObservableCollection<User> Users { get; set; }
}
29. 29
MasterViewModel
public class MasterViewModel
{
public User SelectedUser
{
get { return _selectedUser; }
set
{
SetProperty(ref _selectedUser, value);
EventAggregator.GetEvent<UserChangedMessage>()
.Publish(_selectedUser);
}
}
public ObservableCollection<User> Users { get; set; }
}
30. 30
MasterViewModel
public class MasterViewModel
{
public User SelectedUser
{
get { return _selectedUser; }
set
{
SetProperty(ref _selectedUser, value);
EventAggregator.GetEvent<UserChangedMessage>()
.Publish(_selectedUser);
}
}
public ObservableCollection<User> Users { get; set; }
}
31. 31
DetailViewModel
public class DetailViewModel : BindableBase
{
public DetailViewModel(IEventAggregator eventAggregator)
{
eventAggregator.GetEvent<UserChangedMessage>()
.Subscribe(OnUserChanged);
}
public User User { get; set; }
public void OnUserChanged(User user)
{
User = user;
}
}
32. 32
DetailViewModel
public class DetailViewModel : BindableBase
{
public DetailViewModel(IEventAggregator eventAggregator)
{
eventAggregator.GetEvent<UserChangedMessage>()
.Subscribe(OnUserChanged);
}
public User User { get; set; }
public void OnUserChanged(User user)
{
User = user;
}
}
33. 33
DetailViewModel
public class DetailViewModel : BindableBase
{
public DetailViewModel(IEventAggregator eventAggregator)
{
eventAggregator.GetEvent<UserChangedMessage>()
.Subscribe(OnUserChanged);
}
public User User { get; set; }
public void OnUserChanged(User user)
{
User = user;
}
}
34. 34
DetailViewModel
public class DetailViewModel : BindableBase
{
public DetailViewModel(IEventAggregator eventAggregator)
{
eventAggregator.GetEvent<UserChangedMessage>()
.Subscribe(OnUserChanged);
}
public User User { get; set; }
public void OnUserChanged(User user)
{
User = user;
}
}
35. 35
DetailViewModel
public class DetailViewModel : BindableBase
{
public DetailViewModel(IEventAggregator eventAggregator)
{
eventAggregator.GetEvent<UserChangedMessage>()
.Subscribe(OnUserChanged);
}
public User User { get; set; }
public void OnUserChanged(User user)
{
User = user;
}
}
36. 36
Это работает, но …
Чем плохо:
• Для UserChangedMessage только один получатель
• MessageBus – из пушки по воробьям
• Но без него никак
Чего хочется:
• Биндинг MasterVM.SelectedUser на DetailVM.User
47. 47
Подвох от Prism
Prism предлагает RegionManager (ViewFirt) для:
• View-Switching Navigation
• UI Composition
48. 48
Что делать?
• Разделить задачи ViewSwitching Navigation и UI Composition
• RegionManager – только для навигации
• Не использовать RegionManager для UI Composition
49. 49
UI Composition vs View-Switching Navigation
UI Composition View-Switching Navigation
Делим форму на части Да Да
Зачем делим Части проще и повторно
используются
Меняем регионы
на рантайме
Нет
Выделяем
статическую часть (меню) и
динамическую (регион)
Да
Нельзя (менять
подход к навигации)
Можно (скрепя сердце)Не использовать
MessageBus
52. 52
MasterDetailViewModel
public class MasterDetailViewModel
{
public MasterViewModel MasterViewModel { get; }
public DetailViewModel DetailViewModel { get; }
public MasterDetailViewModel()
{
MasterViewModel
.WhenAny(m => m.SelectedUser, user => user.Value)
.BindTo(DetailViewModel, d => d.User);
}
}
55. 55
View-Switching в разных MVVM
ViewModel-First (можно использовать для UI Composition)
• Caliburn.Micro
• MugenMvvmToolkit
View-First (нельзя использовать для UI Composition)
• Prism.WPF
• ReactiveUI (UI Composition сделать сложно)
56. 56
Не умеют View-Switching
• Prism.Windows (UWP)
• MvvmLight
• MvvmCross
• ReactiveUI (умеет, UI Composition сделать сложно)
72. 72
View-Switching
• Отдельная задача от Navigation
• Табы
• MDI
• Отдельная задача от UI Composition
• UI Composition не требует менять UI на рантайме
73. 73
UI Composition без MessageBus
• Разделить задачи View-Switching Navigation и UI Composition
• Не использовать ViewSwitching View-First для UI Composition
• RegionManager из Prism
• Можно использовать ViewSwitching ViewModel-First для UI Composition
• Caliburn.Micro
• MugenMvvmToolkit
• UI Composition без ViewSwitching (ViewModel в DataContext для View)
• MVVM без View-Switching (Prism.Windows)
• Если нужен ViewSwitching для UI Composition
• Отдельный View-ViewModel маппинг
• ViewModelPresenter
84. 84
Обычные события вместо широковещательных
public class ApplicationStateManager
{
public event EventHandler<ApplicationStateChangedEventArgs> ApplicationStateChanged;
public ApplicationState ApplicationState
{
get { return _applicationState; }
set
{
var oldValue = _applicationState;
_applicationState = value;
OnApplicationStateChanged(oldValue, _applicationState);
}
}
}
85. 85
Обычные события вместо широковещательных
public class ApplicationStateManager
{
public event EventHandler<ApplicationStateChangedEventArgs> ApplicationStateChanged;
public ApplicationState ApplicationState
{
get { return _applicationState; }
set
{
var oldValue = _applicationState;
_applicationState = value;
OnApplicationStateChanged(oldValue, _applicationState);
}
}
}
86. 86
Обычные события вместо широковещательных
public class ApplicationStateManager
{
public event EventHandler<ApplicationStateChangedEventArgs> ApplicationStateChanged;
public ApplicationState ApplicationState
{
get { return _applicationState; }
set
{
var oldValue = _applicationState;
_applicationState = value;
OnApplicationStateChanged(oldValue, _applicationState);
}
}
}
87. 87
Обычные события вместо широковещательных
public class ApplicationStateManager
{
public event EventHandler<ApplicationStateChangedEventArgs> ApplicationStateChanged;
public ApplicationState ApplicationState
{
get { return _applicationState; }
set
{
var oldValue = _applicationState;
_applicationState = value;
OnApplicationStateChanged(oldValue, _applicationState);
}
}
}
88. 88
ApplicationStateManager vs MessageBus
Частных решений не будет много (у нас - одно)
ApplicationStateManager MessageBus
Подписчик Любой Любой
Отправитель ApplicationStateManager Любой
Событие ApplicationStateChanged Любое
90. 90
Пример цепочки вызовов методов
Package Prism.Windows
Класс PrismApplication.cs
Метод OnSuspending
https://github.com/PrismLibrary/Prism/blob/master/Source/Windows10/Prism.Windows/PrismApplication.cs
91. 91
Цепочка вызовов vs частное решение
Цепочка вызовов
+ Не требует генерации событий
- Нужна продуманная архитектура
Частное решение
+ Их не будет много
- Хочется заменить частное на универсальное (MessageBus)
95. 95
DetailViewModel
public class DetailViewModel : IHandle<UserChangedMessage>
{
public User User { get; set; }
public void Handle(UserChangedMessage message)
{
User = message.User;
if (User?.IsNew)
{
EventAggregator.PublishOnUIThread(new FocusMessage());
}
}
}
96. 96
DetailViewModel
public class DetailViewModel : IHandle<UserChangedMessage>
{
public User User { get; set; }
public void Handle(UserChangedMessage message)
{
User = message.User;
if (User?.IsNew)
{
EventAggregator.PublishOnUIThread(new FocusMessage());
}
}
}
97. 97
DetailViewModel
public class DetailViewModel : IHandle<UserChangedMessage>
{
public User User { get; set; }
public void Handle(UserChangedMessage message)
{
User = message.User;
if (User?.IsNew)
{
EventAggregator.PublishOnUIThread(new FocusMessage());
}
}
}
98. 98
DetailView
public partial class DetailView : UserControl, IHandle<FocusMessage>
{
public void Handle(FocusMessage message)
{
FirstNameTextBox.Focus();
}
}
99. 99
DetailView
public partial class DetailView : UserControl, IHandle<FocusMessage>
{
public void Handle(FocusMessage message)
{
FirstNameTextBox.Focus();
}
}
100. 100
Как обойтись – Behavior (или Trigger)
public class FocusNewUserBehavior : Behavior<TextBox>
{
public static readonly DependencyProperty User /* код опущен */
private static void OnUserChanged
(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = (FocusNewUserBehavior) d;
if (behavior.User?.IsNew)
{
behavior.AssociatedObject.Focus();
}
}
}
101. 101
Как обойтись – Behavior (или Trigger)
public class FocusNewUserBehavior : Behavior<TextBox>
{
public static readonly DependencyProperty User /* код опущен */
private static void OnUserChanged
(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = (FocusNewUserBehavior) d;
if (behavior.User?.IsNew)
{
behavior.AssociatedObject.Focus();
}
}
}
102. 102
Как обойтись – Behavior (или Trigger)
public class FocusNewUserBehavior : Behavior<TextBox>
{
public static readonly DependencyProperty User /* код опущен */
private static void OnUserChanged
(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = (FocusNewUserBehavior) d;
if (behavior.User?.IsNew)
{
behavior.AssociatedObject.Focus();
}
}
}
103. 103
Как обойтись – Behavior (или Trigger)
public class FocusNewUserBehavior : Behavior<TextBox>
{
public static readonly DependencyProperty User /* код опущен */
private static void OnUserChanged
(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = (FocusNewUserBehavior) d;
if (behavior.User?.IsNew)
{
behavior.AssociatedObject.Focus();
}
}
}
104. 104
Как обойтись – Behavior (или Trigger)
public class FocusNewUserBehavior : Behavior<TextBox>
{
public static readonly DependencyProperty User /* код опущен */
private static void OnUserChanged
(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = (FocusNewUserBehavior) d;
if (behavior.User?.IsNew)
{
behavior.AssociatedObject.Focus();
}
}
}
105. 105
Как обойтись – Behavior (или Trigger)
public class FocusNewUserBehavior : Behavior<TextBox>
{
public static readonly DependencyProperty User /* код опущен */
private static void OnUserChanged
(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = (FocusNewUserBehavior) d;
if (behavior.User?.IsNew)
{
behavior.AssociatedObject.Focus();
}
}
}
109. 109
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
110. 110
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
111. 111
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
112. 112
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
113. 113
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
114. 114
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
115. 115
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
116. 116
Решение – Observable модель
public class ObservableDictionary<TKey, TValue>:
INotifyCollectionChanged,
INotifyPropertyChanged
{
}
https://code.msdn.microsoft.com/windowsdesktop/Samples-for-Parallel-b4b76364
117. 117
Не Observable модели (Prism)
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
118. 118
Observable модель
protected void UpdatePrice(string tickerSymbol, decimal newPrice)
{
_priceList[tickerSymbol] = newPrice;
OnPricesUpdated();
}
private void OnPricesUpdated()
{
var clonedPriceList = new Dictionary<string, decimal>(_priceList);
EventAggregator.GetEvent<MarketPricesUpdatedEvent>().Publish(clonedPriceList);
}
120. 120
MVVM фреймворки без MessageBus
• Таких нет
• В отдельном пакете
• Prism 5 – SubEvents (в Prism 6 уже неотделяем)
• MvvmCross – Messenger plugin
122. 122
Как жить без MessageBus на WPF и UWP
• Caliburn.Micro
• MugenMvvmToolkit
• MVVM без UI Composition
• + свой ViewModel-First UI Composition
123. 123
Не позволяет жить без MessageBus
WPF
• Prism.WPF
UWP
• Таких нет
• Prism.Windows без UI Composition
Как добавить UI Composition: https://github.com/denis-tsv/Prism.StoreApps.Extensions.Mvvm
124. 124
MessageBus на UI
Для чего используется Как обойтись
UI Composition Разделение UI Composition и ViewSwitching
Глобальные события
(смена контекста или lifecycle)
Цепочка вызовов методов
Частные решения с обычными событиями
ViewModel-View Message Triggers, Behaviors
Logic-ViewModel Message Observable Model
Избежать Messages между уровнями Разные сборки для View, ViewModel, Logic
Messages в сборке ViewModel
125. 125
Главная проблема MessageBus на UI – костыли!
Messages между слоями
• Messages вместо Events (Logic -> ViewModel Messages)
• Messages вместо Bindings (ViewModel -> View Messages)
126. 126
Как можно жить с MessageBus на UI
• UI Composition в стиле View-First (Prism.WPF)
• Отдельные сборки View, ViewModel, Logic
• Messages только в ViewModel
• Глобальные Messages
127. 127
Моя позиция
• Большинство бизнес-приложений маленькие и средние
• Команда 3-7 человек
• Срок пара месяцев – пара лет
• UI сложнее бэкэнда
• Не нужно CQRS, Microservices, DataAccess Abstraction
• MessageBus
• Ещё сильнее запутывает и так сложный UI
• Любую UI задачу можно решить без MessageBus
• Примеры Prism без EventAggregator полезная ссылка №2
• Строка «MessageBus» ломает билд полезная ссылка №9
128. 128
Paul Betts (ReactiveUI author) about MessageBus
… there are often
more correct ways
to solve problems,
given a bit of
ingenuity …
129. 129
В параллельной вселенной JS
Сходства
• Fontend не уступает Backend по сложности
• Развесистые JS-фреймворки (Angular, React)
• Компонент JS = Регион XAML
• Компонент (Шаблон + Поведение)
• Регион (View + ViewModel)
130. 130
Olical EventEmitter
var ee = new EventEmitter();
function listener() { }
ee.addListener('event', listener);
ee.emitEvent('event');
ee.removeListener('event', listener);
131. 131
Olical EventEmitter
var ee = new EventEmitter();
function listener() { }
ee.addListener('event', listener);
ee.emitEvent('event');
ee.removeListener('event', listener);
132. 132
Olical EventEmitter
var ee = new EventEmitter();
function listener() { }
ee.addListener('event', listener);
ee.emitEvent('event');
ee.removeListener('event', listener);
133. 133
Angular2 EventEmitter – Observer, а не MessageBus
class EventEmitter {
constructor(isAsync?: boolean)
emit(value?: T)
subscribe(generatorOrNext?: any, error?: any, complete?: any)
: any
}
134. 134
Angular2 EventEmitter – Observer, а не MessageBus
class EventEmitter {
constructor(isAsync?: boolean)
emit(value?: T)
subscribe(generatorOrNext?: any, error?: any, complete?: any)
: any
}
135. 135
Angular2 EventEmitter – Observer, а не MessageBus
class EventEmitter {
constructor(isAsync?: boolean)
emit(value?: T)
subscribe(generatorOrNext?: any, error?: any, complete?: any)
: any
}
136. 136
Angular2 EventEmitter – Observer, а не MessageBus
class EventEmitter {
constructor(isAsync?: boolean)
emit(value?: T)
subscribe(generatorOrNext?: any, error?: any, complete?: any)
: any
}
167. 167
React Context – не MessageBus
Задача
• Передача от родителя к вложенным компонентам
• С уровня 1 на уровень 3 минуя уровень 2
Особенности реализации
• Доступен родительскому и вложенным компонентам (не любым
компонентам)
• Обновление контекста во вложенных компонентах не работает
Это не MessageBus
168. 168
Резюме: MessageBus на JS
• MessageBus существует, но в виде отдельных библиотек
• MessageBus нет в коробке флагманов Angular2 и React
• Официальный гайд не рекомендует MesageBus
• Компонент != ViewModel
• Компонент = Шаблон + Поведение
• ViewModel = Поведение
• На JS нет разных видов композиции ViewFirst и ViewModelFirst
• Для JS MessageBus не является проблемой, как в XAML
• Продуманная архитектура JS фреймворков – аргумент за
переписывание UI на web-стек
169. 169
Полезные ссылки
1. MessageBus, врага нужно знать в лицо
https://msdn.microsoft.com/en-us/library/ff647328.aspx
2. Примеры StockTrader и UIComposition для Prism.WPF, откуда выпилен EventAggregator
https://github.com/denis-tsv/Prism-Samples-Wpf-NoEventAggregetor
3. Как добавить UI Composition для Prism.Windows (UWP)
https://github.com/denis-tsv/Prism.StoreApps.Extensions.Mvvm
4. Paul Betts. MessageBus and why you shouldn't use it
http://log.paulbetts.org/messagebus-and-why-you-shouldnt-use-it/
5, Caliburn.Micro Классный, минималистичный, кроссплатформенный и мультипарадигменный MVVM, в котором нет ничего лишнего
http://caliburnmicro.com/
6. MugenMvvmToolkit. Мощный UI фреймворк с офигительными кастомными XAML биндингами
https://github.com/MugenMvvmToolkit
https://habrahabr.ru/post/236745/
7. ReactiveUI. Биндинг свойств одной ViewModel на свойства другой ViewModel в родительской ViewModel
http://reactiveui.net/
8. Пример View-ViewModel mapping с использованием MEF
http://www.intuit.ru/EDI/14_12_14_2/1418505481-18353/tutorial/1050/objects/1/files/SampleSolution.zip
9. Как сломать билд (о, да!) словом MessageBus, EventAggregator, Messenger и любой другой реализацией MessageBus
http://hmemcpy.com/2013/03/even-more-resharper-annotations-hackery-marking-3rd-party-code-as-obsolete/
Паттерн, предназначенный для обмена сообщениями между процессами (бэкэнд)
Про него много говорят в области микросервисов
MessageBus позволяет любому процессу отправить сообщение любому другому процессу
Кто виноват и что делать?
Кто по картинке узнал виновника?
Как и при авиакатастрофах все списывают на мертвых
Нет явной подписки, она в инфраструктуре CAB UI
Мышки кололись и плакали но продолжали есть кактус
UI Composition – это разделение больших и сложных форм на более просты составляющие и настройка взаимодействия между ними.
Маленькие части проще писать и их можно повторно использовать
MessageBus – для настройки взаимодействия между формами
Содержимое правого региона меняется на рантайме
Только делим форму на части, на рантайме не меняем содержимое региона
Нет родительской ViewModel
Messages между ViewModel регионов
Родительская ViewModel
Взаимодействие ViewModel регионов в родительской ViewModel
Более продвинутое решение для View-Switching Navigation используется для более простой задачи UI Composition
Далее цепочка
View-Switching Navigation (View-First) для UI Composition
Message Bus для передачи сообщений между ViewModel
View-Switching Navigation и UI Composition – разные задачи
View-Switching Navigation – вторична
UI Composition – первична
Разделить задачи View-Switching Navigation и UI Composition
Это решение используется в Prism.Windows
TODO рассказать про навигацию и Composite UI, Что это разные задачи. CompositeUI не зависит от выбранного подхода к навигации
В списке слева – права и роли, при выборе элемента – отображение специализированной detail формы
Как это сделать при использовании MVVM без ViewSwitching?
Делает UI Composition более мощным (замена на рантайме вьюх) и используется для навигации