I have been using ReactiveUI for a while with Xamarin Forms, but I've hit a brick wall when trying to use a ReactiveTabbedPage. I can't figure out how the ViewModel will get bound to the ReactiveContentPage's that are the children of the ReactiveTabbedPage.
So, as an example, I might have the following XAML:
<ReactiveTabbedPage x:Name="TabbedPage">
<local:Page1View x:Name="Page1" />
<local:Page2View x:Name="Page2" />
</ReactiveTabbedPage>
Where Page1View and Page2View are both of type ReactiveContentPage and T is the associated ViewModel.
What I expected to happen was that when the ReactiveTabbedPage was navigated to, Page1View would be displayed, and the ViewModel would be loaded (in the same way it would if I navigated to the Page1View directly). However, the ViewModel never gets called (the constructor is never fired and no data binding occurs).
However, both Page1View and Page2View do render and I can see the initial data that is created in those views (e.g. default text for labels etc.).
I know that the ViewModel stuff is working correctly, because if I navigate to Page1View directly (e.g. not in the ReactiveTabbedPage) everything displays as I expect.
Have I missed something, or am I going about this the wrong way? Or is this just not supported in the current version of RxUI?
Any advice is greatly appreciated!
The responsibility for tying the VM to the child pages lies with the host page (i.e. the
ReactiveTabbedPage
). It alone knows which VM corresponds to which view.Let's take this one step at a time. First of all, the
MainViewModel
:This code obviously isn't realistic because you wouldn't want to recreate the child VMs upon every property access. It's more the API that's pertinent here.
ChildViewModel1
looks like this:And
ChildViewModel2
looks much the same.Now we can go about setting the views up. Our
MainView.xaml
looks like this:Notice it declares each of the child views. We need to hook up the VMs to those views, which we do in the code-behind for
MainView
:I've done this the safest way by using
WhenActivated
andOneWayBind
calls. In reality, it's unlikely your child VMs will change, so directly assigning them rather than binding is totally fine.Now our child views can be thrown together. Here's
ChildView1.xaml
:And the code-behind:
Once again we're doing the usual RxUI binding goodness to associate properties in the VM with controls in the UI. And once again you could optimize this for properties that don't mutate.
For the purposes of this example,
ChildView2
is much the same asChildView1
, but obviously it could be totally different.The end result is as you'd expect:
What's not evident from the screenshot but is very important is that each tab is deactivating when you switch away from it (as would its associated view model if it implemented
ISupportsActivation
). This means you can clean up any bindings and subscriptions for that tab when it's not in use, reducing memory pressure and improving performance.