4
4
// See the LICENSE file in the project root for full license information.
5
5
6
6
using System ;
7
+ using System . Reactive ;
7
8
using System . Reactive . Disposables ;
8
9
using System . Reactive . Linq ;
10
+ using System . Threading ;
9
11
using Splat ;
10
12
11
13
namespace ReactiveUI ;
@@ -16,38 +18,50 @@ namespace ReactiveUI;
16
18
public static class SuspensionHostExtensions
17
19
{
18
20
/// <summary>
19
- /// Observe changes to the AppState of a class derived from ISuspensionHost .
21
+ /// Func used to load app state exactly once .
20
22
/// </summary>
21
- /// <typeparam name="T">The observable type.</typeparam>
23
+ private static Func < IObservable < Unit > > ? ensureLoadAppStateFunc ;
24
+
25
+ /// <summary>
26
+ /// Supsension driver reference field to prevent introducing breaking change.
27
+ /// </summary>
28
+ private static ISuspensionDriver ? suspensionDriver ;
29
+
30
+ /// <summary>
31
+ /// Get the current App State of a class derived from ISuspensionHost.
32
+ /// </summary>
33
+ /// <typeparam name="T">The app state type.</typeparam>
22
34
/// <param name="item">The suspension host.</param>
23
- /// <returns>An observable of the app state.</returns>
24
- public static IObservable < T > ObserveAppState < T > ( this ISuspensionHost item )
25
- where T : class
35
+ /// <returns>The app state.</returns>
36
+ public static T GetAppState < T > ( this ISuspensionHost item )
26
37
{
27
38
if ( item is null )
28
39
{
29
40
throw new ArgumentNullException ( nameof ( item ) ) ;
30
41
}
31
42
32
- return item . WhenAny ( suspensionHost => suspensionHost . AppState , observedChange => observedChange . Value )
33
- . WhereNotNull ( )
34
- . Cast < T > ( ) ;
43
+ Interlocked . Exchange ( ref ensureLoadAppStateFunc , null ) ? . Invoke ( ) ;
44
+
45
+ return ( T ) item . AppState ! ;
35
46
}
36
47
37
48
/// <summary>
38
- /// Get the current App State of a class derived from ISuspensionHost.
49
+ /// Observe changes to the AppState of a class derived from ISuspensionHost.
39
50
/// </summary>
40
- /// <typeparam name="T">The app state type.</typeparam>
51
+ /// <typeparam name="T">The observable type.</typeparam>
41
52
/// <param name="item">The suspension host.</param>
42
- /// <returns>The app state.</returns>
43
- public static T GetAppState < T > ( this ISuspensionHost item )
53
+ /// <returns>An observable of the app state.</returns>
54
+ public static IObservable < T > ObserveAppState < T > ( this ISuspensionHost item )
55
+ where T : class
44
56
{
45
57
if ( item is null )
46
58
{
47
59
throw new ArgumentNullException ( nameof ( item ) ) ;
48
60
}
49
61
50
- return ( T ) item . AppState ! ;
62
+ return item . WhenAny ( suspensionHost => suspensionHost . AppState , observedChange => observedChange . Value )
63
+ . WhereNotNull ( )
64
+ . Cast < T > ( ) ;
51
65
}
52
66
53
67
/// <summary>
@@ -65,32 +79,64 @@ public static IDisposable SetupDefaultSuspendResume(this ISuspensionHost item, I
65
79
}
66
80
67
81
var ret = new CompositeDisposable ( ) ;
68
- driver ??= Locator . Current . GetService < ISuspensionDriver > ( ) ;
82
+ suspensionDriver ??= driver ?? Locator . Current . GetService < ISuspensionDriver > ( ) ;
69
83
70
- if ( driver is null )
84
+ if ( suspensionDriver is null )
71
85
{
72
86
item . Log ( ) . Error ( "Could not find a valid driver and therefore cannot setup Suspend/Resume." ) ;
73
87
return Disposable . Empty ;
74
88
}
75
89
90
+ ensureLoadAppStateFunc = ( ) => EnsureLoadAppState ( item , suspensionDriver ) ;
91
+
76
92
ret . Add ( item . ShouldInvalidateState
77
- . SelectMany ( _ => driver . InvalidateState ( ) )
93
+ . SelectMany ( _ => suspensionDriver . InvalidateState ( ) )
78
94
. LoggedCatch ( item , Observables . Unit , "Tried to invalidate app state" )
79
95
. Subscribe ( _ => item . Log ( ) . Info ( "Invalidated app state" ) ) ) ;
80
96
81
97
ret . Add ( item . ShouldPersistState
82
- . SelectMany ( x => driver . SaveState ( item . AppState ! ) . Finally ( x . Dispose ) )
98
+ . SelectMany ( x => suspensionDriver . SaveState ( item . AppState ! ) . Finally ( x . Dispose ) )
83
99
. LoggedCatch ( item , Observables . Unit , "Tried to persist app state" )
84
100
. Subscribe ( _ => item . Log ( ) . Info ( "Persisted application state" ) ) ) ;
85
101
86
102
ret . Add ( item . IsResuming . Merge ( item . IsLaunchingNew )
87
- . SelectMany ( _ => driver . LoadState ( ) )
88
- . LoggedCatch (
89
- item ,
90
- Observable . Defer ( ( ) => Observable . Return ( item . CreateNewAppState ? . Invoke ( ) ) ) ,
91
- "Failed to restore app state from storage, creating from scratch" )
92
- . Subscribe ( x => item . AppState = x ?? item . CreateNewAppState ? . Invoke ( ) ) ) ;
103
+ . Do ( _ => Interlocked . Exchange ( ref ensureLoadAppStateFunc , null ) ? . Invoke ( ) )
104
+ . Subscribe ( ) ) ;
93
105
94
106
return ret ;
95
107
}
96
- }
108
+
109
+ /// <summary>
110
+ /// Ensures one time app state load from storage.
111
+ /// </summary>
112
+ /// <param name="item">The suspension host.</param>
113
+ /// <param name="driver">The suspension driver.</param>
114
+ /// <returns>A completed observable.</returns>
115
+ private static IObservable < Unit > EnsureLoadAppState ( this ISuspensionHost item , ISuspensionDriver ? driver = null )
116
+ {
117
+ if ( item . AppState is not null )
118
+ {
119
+ return Observable . Return ( Unit . Default ) ;
120
+ }
121
+
122
+ suspensionDriver ??= driver ?? Locator . Current . GetService < ISuspensionDriver > ( ) ;
123
+
124
+ if ( suspensionDriver is null )
125
+ {
126
+ item . Log ( ) . Error ( "Could not find a valid driver and therefore cannot load app state." ) ;
127
+ return Observable . Return ( Unit . Default ) ;
128
+ }
129
+
130
+ try
131
+ {
132
+ item . AppState = suspensionDriver . LoadState ( ) . Wait ( ) ;
133
+ }
134
+ catch ( Exception ex )
135
+ {
136
+ item . Log ( ) . Warn ( ex , "Failed to restore app state from storage, creating from scratch" ) ;
137
+ item . AppState = item . CreateNewAppState ? . Invoke ( ) ;
138
+ }
139
+
140
+ return Observable . Return ( Unit . Default ) ;
141
+ }
142
+ }
0 commit comments