forked from reactiveui/ReactiveUI
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRxRouting.cs
136 lines (120 loc) · 5.37 KB
/
RxRouting.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reactive;
using System.Reactive.Linq;
using System.Text;
using ReactiveUI;
namespace ReactiveUI.Routing
{
public static class RxRouting
{
public static Func<string, string> ViewModelToViewFunc { get; set; }
static RxRouting()
{
ViewModelToViewFunc = (vm) => interfaceifyTypeName(vm.Replace("ViewModel", "View"));
}
/// <summary>
/// Returns the View associated with a ViewModel, deriving the name of
/// the Type via ViewModelToViewFunc, then discovering it via
/// ServiceLocator.
/// </summary>
/// <param name="viewModel">The ViewModel for which to find the
/// associated View.</param>
/// <returns>The View for the ViewModel.</returns>
public static IViewFor ResolveView<T>(T viewModel)
where T : class
{
// Given IFooBarViewModel (whose name we derive from T), we'll look
// for a few things:
// * IFooBarView that implements IViewFor
// * IViewFor<IFooBarViewModel>
// * IViewFor<FooBarViewModel> (the original behavior in RxUI 3.1)
var attrs = viewModel.GetType().GetCustomAttributes(typeof (ViewContractAttribute), true);
string key = null;
if (attrs.Any()) {
key = ((ViewContractAttribute) attrs.First()).Contract;
}
// IFooBarView that implements IViewFor (or custom ViewModelToViewFunc)
var typeToFind = ViewModelToViewFunc(viewModel.GetType().AssemblyQualifiedName);
try {
var type = Reflection.ReallyFindType(typeToFind, true);
var ret = RxApp.GetService(type, key) as IViewFor;
if (ret != null) return ret;
} catch (Exception ex) {
LogHost.Default.DebugException("Couldn't instantiate " + typeToFind, ex);
}
var viewType = typeof (IViewFor<>);
// IViewFor<IFooBarViewModel>
try {
var ifn = interfaceifyTypeName(viewModel.GetType().AssemblyQualifiedName);
var type = Reflection.ReallyFindType(ifn, true);
var ret = RxApp.GetService(viewType.MakeGenericType(type), key) as IViewFor;
if (ret != null) return ret;
} catch (Exception ex) {
LogHost.Default.DebugException("Couldn't instantiate View via pure interface type", ex);
}
// IViewFor<FooBarViewModel> (the original behavior in RxUI 3.1)
return (IViewFor) RxApp.GetService(viewType.MakeGenericType(viewModel.GetType()), key);
}
static string interfaceifyTypeName(string typeName)
{
var typeVsAssembly = typeName.Split(',');
var parts = typeVsAssembly[0].Split('.');
parts[parts.Length - 1] = "I" + parts[parts.Length - 1];
var newType = String.Join(".", parts, 0, parts.Length);
return newType + "," + String.Join(",", typeVsAssembly.Skip(1));
}
}
public static class RoutableViewModelMixin
{
/// <summary>
/// This Observable fires whenever the current ViewModel is navigated to.
/// Note that this method is difficult to use directly without leaking
/// memory, you most likely want to use WhenNavigatedTo.
/// </summary>
public static IObservable<Unit> NavigatedToMe(this IRoutableViewModel This)
{
return This.HostScreen.Router.ViewModelObservable()
.Where(x => x == This)
.Select(_ => Unit.Default);
}
/// <summary>
/// This Observable fires whenever the current ViewModel is navigated
/// away from. Note that this method is difficult to use directly
/// without leaking memory, you most likely want to use WhenNavigatedTo.
/// </summary>
public static IObservable<Unit> NavigatedFromMe(this IRoutableViewModel This)
{
return This.HostScreen.Router.ViewModelObservable()
.Where(x => x != This)
.Select(_ => Unit.Default);
}
/// <summary>
/// This method allows you to set up connections that only operate
/// while the ViewModel has focus, and cleans up when the ViewModel
/// loses focus.
/// </summary>
/// <param name="onNavigatedTo">Called when the ViewModel is navigated
/// to - return an IDisposable that cleans up all of the things that are
/// configured in the method.</param>
/// <returns>An IDisposable that lets you disconnect the entire process
/// earlier than normal.</returns>
public static IDisposable WhenNavigatedTo(this IRoutableViewModel This, Func<IDisposable> onNavigatedTo)
{
IDisposable inner = null;
var router = This.HostScreen.Router;
return router.NavigationStack.CollectionCountChanged.Subscribe(_ => {
if (router.GetCurrentViewModel() == This) {
if (inner != null) inner.Dispose();
inner = onNavigatedTo();
} else {
if (inner != null) {
inner.Dispose();
inner = null;
}
}
});
}
}
}