1 //*********************************************************
2 //
3 // Copyright (c) Microsoft. All rights reserved.
4 // THIS CODE IS PROVIDED *AS IS* WITHOUT WARRANTY OF
5 // ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING ANY
6 // IMPLIED WARRANTIES OF FITNESS FOR A PARTICULAR
7 // PURPOSE, MERCHANTABILITY, OR NON-INFRINGEMENT.
8 //
9 //*********************************************************
10 
11 //
12 // SuspensionManager.cpp
13 // Implementation of the SuspensionManager class
14 //
15 
16 #include "pch.h"
17 #include "SuspensionManager.h"
18 
19 #include <collection.h>
20 #include <algorithm>
21 
22 using namespace SDKSample::Common;
23 
24 using namespace Concurrency;
25 using namespace Platform;
26 using namespace Platform::Collections;
27 using namespace Windows::Foundation;
28 using namespace Windows::Foundation::Collections;
29 using namespace Windows::Storage;
30 using namespace Windows::Storage::FileProperties;
31 using namespace Windows::Storage::Streams;
32 using namespace Windows::UI::Xaml;
33 using namespace Windows::UI::Xaml::Controls;
34 using namespace Windows::UI::Xaml::Interop;
35 
36 namespace
37 {
38     Map<String^, Object^>^ _sessionState = ref new Map<String^, Object^>();
39     String^ sessionStateFilename = "_sessionState.dat";
40 
41     // Forward declarations for object object read / write support
42     void WriteObject(Windows::Storage::Streams::DataWriter^ writer, Platform::Object^ object);
43     Platform::Object^ ReadObject(Windows::Storage::Streams::DataReader^ reader);
44 }
45 
46 /// <summary>
47 /// Provides access to global session state for the current session.  This state is serialized by
48 /// <see cref="SaveAsync"/> and restored by <see cref="RestoreAsync"/> which require values to be
49 /// one of the following: boxed values including integers, floating-point singles and doubles,
50 /// wide characters, boolean, Strings and Guids, or Map<String^, Object^> where map values are
51 /// subject to the same constraints.  Session state should be as compact as possible.
52 /// </summary>
53 IMap<String^, Object^>^ SuspensionManager::SessionState::get(void)
54 {
55     return _sessionState;
56 }
57 
58 /// <summary>
59 /// Wrap a WeakReference as a reference object for use in a collection.
60 /// </summary>
61 private ref class WeakFrame sealed
62 {
63 private:
64     WeakReference _frameReference;
65 
66 internal:
67     WeakFrame(Frame^ frame) { _frameReference = frame; }
68     property Frame^ ResolvedFrame
69     {
70         Frame^ get(void) { return _frameReference.Resolve<Frame>(); }
71     };
72 };
73 
74 namespace
75 {
76     std::vector<WeakFrame^> _registeredFrames;
77     DependencyProperty^ FrameSessionStateKeyProperty =
78         DependencyProperty::RegisterAttached("_FrameSessionStateKeyProperty",
79         TypeName(String::typeid), TypeName(SuspensionManager::typeid), nullptr);
80     DependencyProperty^ FrameSessionStateProperty =
81         DependencyProperty::RegisterAttached("_FrameSessionStateProperty",
82         TypeName(IMap<String^, Object^>::typeid), TypeName(SuspensionManager::typeid), nullptr);
83 }
84 
85 /// <summary>
86 /// Registers a <see cref="Frame"/> instance to allow its navigation history to be saved to
87 /// and restored from <see cref="SessionState"/>.  Frames should be registered once
88 /// immediately after creation if they will participate in session state management.  Upon
89 /// registration if state has already been restored for the specified key
90 /// the navigation history will immediately be restored.  Subsequent invocations of
91 /// <see cref="RestoreAsync(String)"/> will also restore navigation history.
92 /// </summary>
93 /// <param name="frame">An instance whose navigation history should be managed by
94 /// <see cref="SuspensionManager"/></param>
95 /// <param name="sessionStateKey">A unique key into <see cref="SessionState"/> used to
96 /// store navigation-related information.</param>
97 void SuspensionManager::RegisterFrame(Frame^ frame, String^ sessionStateKey)
98 {
99     if (frame->GetValue(FrameSessionStateKeyProperty) != nullptr)
100     {
101         throw ref new FailureException("Frames can only be registered to one session state key");
102     }
103 
104     if (frame->GetValue(FrameSessionStateProperty) != nullptr)
105     {
106         throw ref new FailureException("Frames must be either be registered before accessing frame session state, or not registered at all");
107     }
108 
109     // Use a dependency property to associate the session key with a frame, and keep a list of frames whose
110     // navigation state should be managed
111     frame->SetValue(FrameSessionStateKeyProperty, sessionStateKey);
112     _registeredFrames.insert(_registeredFrames.begin(), ref new WeakFrame(frame));
113 
114     // Check to see if navigation state can be restored
115     RestoreFrameNavigationState(frame);
116 }
117 
118 /// <summary>
119 /// Disassociates a <see cref="Frame"/> previously registered by <see cref="RegisterFrame"/>
120 /// from <see cref="SessionState"/>.  Any navigation state previously captured will be
121 /// removed.
122 /// </summary>
123 /// <param name="frame">An instance whose navigation history should no longer be
124 /// managed.</param>
125 void SuspensionManager::UnregisterFrame(Frame^ frame)
126 {
127     // Remove session state and remove the frame from the list of frames whose navigation
128     // state will be saved (along with any weak references that are no longer reachable)
129     auto key = safe_cast<String^>(frame->GetValue(FrameSessionStateKeyProperty));
130     if (SessionState->HasKey(key)) SessionState->Remove(key);
131     _registeredFrames.erase(
132         std::remove_if(_registeredFrames.begin(), _registeredFrames.end(), [=](WeakFrame^& e)
133         {
134             auto testFrame = e->ResolvedFrame;
135             return testFrame == nullptr || testFrame == frame;
136         }),
137         _registeredFrames.end()
138     );
139 }
140 
141 /// <summary>
142 /// Provides storage for session state associated with the specified <see cref="Frame"/>.
143 /// Frames that have been previously registered with <see cref="RegisterFrame"/> have
144 /// their session state saved and restored automatically as a part of the global
145 /// <see cref="SessionState"/>.  Frames that are not registered have transient state
146 /// that can still be useful when restoring pages that have been discarded from the
147 /// navigation cache.
148 /// </summary>
149 /// <remarks>Apps may choose to rely on <see cref="LayoutAwarePage"/> to manage
150 /// page-specific state instead of working with frame session state directly.</remarks>
151 /// <param name="frame">The instance for which session state is desired.</param>
152 /// <returns>A collection of state subject to the same serialization mechanism as
153 /// <see cref="SessionState"/>.</returns>
154 IMap<String^, Object^>^ SuspensionManager::SessionStateForFrame(Frame^ frame)
155 {
156     auto frameState = safe_cast<IMap<String^, Object^>^>(frame->GetValue(FrameSessionStateProperty));
157 
158     if (frameState == nullptr)
159     {
160         auto frameSessionKey = safe_cast<String^>(frame->GetValue(FrameSessionStateKeyProperty));
161         if (frameSessionKey != nullptr)
162         {
163             // Registered frames reflect the corresponding session state
164             if (!_sessionState->HasKey(frameSessionKey))
165             {
166                 _sessionState->Insert(frameSessionKey, ref new Map<String^, Object^>());
167             }
168             frameState = safe_cast<IMap<String^, Object^>^>(_sessionState->Lookup(frameSessionKey));
169         }
170         else
171         {
172             // Frames that aren't registered have transient state
173             frameState = ref new Map<String^, Object^>();
174         }
175         frame->SetValue(FrameSessionStateProperty, frameState);
176     }
177     return frameState;
178 }
179 
180 void SuspensionManager::RestoreFrameNavigationState(Frame^ frame)
181 {
182     auto frameState = SessionStateForFrame(frame);
183     if (frameState->HasKey("Navigation"))
184     {
185         frame->SetNavigationState(safe_cast<String^>(frameState->Lookup("Navigation")));
186     }
187 }
188 
189 void SuspensionManager::SaveFrameNavigationState(Frame^ frame)
190 {
191     auto frameState = SessionStateForFrame(frame);
192     frameState->Insert("Navigation", frame->GetNavigationState());
193 }
194 
195 /// <summary>
196 /// Save the current <see cref="SessionState"/>.  Any <see cref="Frame"/> instances
197 /// registered with <see cref="RegisterFrame"/> will also preserve their current
198 /// navigation stack, which in turn gives their active <see cref="Page"/> an opportunity
199 /// to save its state.
200 /// </summary>
201 /// <returns>An asynchronous task that reflects when session state has been saved.</returns>
202 task<void> SuspensionManager::SaveAsync(void)
203 {
204     // Save the navigation state for all registered frames
205     for (auto&& weakFrame : _registeredFrames)
206     {
207         auto frame = weakFrame->ResolvedFrame;
208         if (frame != nullptr) SaveFrameNavigationState(frame);
209     }
210 
211     // Serialize the session state synchronously to avoid asynchronous access to shared
212     // state
213     auto sessionData = ref new InMemoryRandomAccessStream();
214     auto sessionDataWriter = ref new DataWriter(sessionData->GetOutputStreamAt(0));
215     WriteObject(sessionDataWriter, _sessionState);
216 
217     // Once session state has been captured synchronously, begin the asynchronous process
218     // of writing the result to disk
219     return task<unsigned int>(sessionDataWriter->StoreAsync()).then([=](unsigned int)
220     {
221         return sessionDataWriter->FlushAsync();
222     }).then([=](bool flushSucceeded)
223     {
224         (void)flushSucceeded; // Unused parameter
225         return ApplicationData::Current->LocalFolder->CreateFileAsync(sessionStateFilename,
226             CreationCollisionOption::ReplaceExisting);
227     }).then([=](StorageFile^ createdFile)
228     {
229         return createdFile->OpenAsync(FileAccessMode::ReadWrite);
230     }).then([=](IRandomAccessStream^ newStream)
231     {
232         return RandomAccessStream::CopyAndCloseAsync(
233             sessionData->GetInputStreamAt(0), newStream->GetOutputStreamAt(0));
234     }).then([=](UINT64 copiedBytes)
235     {
236         (void)copiedBytes; // Unused parameter
237         return;
238     });
239 }
240 
241 /// <summary>
242 /// Restores previously saved <see cref="SessionState"/>.  Any <see cref="Frame"/> instances
243 /// registered with <see cref="RegisterFrame"/> will also restore their prior navigation
244 /// state, which in turn gives their active <see cref="Page"/> an opportunity restore its
245 /// state.
246 /// </summary>
247 /// <param name="version">A version identifer compared to the session state to prevent
248 /// incompatible versions of session state from reaching app code.  Saved state with a
249 /// different version will be ignored, resulting in an empty <see cref="SessionState"/>
250 /// dictionary.</param>
251 /// <returns>An asynchronous task that reflects when session state has been read.  The
252 /// content of <see cref="SessionState"/> should not be relied upon until this task
253 /// completes.</returns>
254 task<void> SuspensionManager::RestoreAsync(void)
255 {
256     _sessionState->Clear();
257 
258     task<StorageFile^> getFileTask(ApplicationData::Current->LocalFolder->GetFileAsync(sessionStateFilename));
259     return getFileTask.then([=](StorageFile^ stateFile)
260     {
261         task<BasicProperties^> getBasicPropertiesTask(stateFile->GetBasicPropertiesAsync());
262         return getBasicPropertiesTask.then([=](BasicProperties^ stateFileProperties)
263         {
264             auto size = unsigned int(stateFileProperties->Size);
265             if (size != stateFileProperties->Size) throw ref new FailureException("Session state larger than 4GB");
266             task<IRandomAccessStreamWithContentType^> openReadTask(stateFile->OpenReadAsync());
267             return openReadTask.then([=](IRandomAccessStreamWithContentType^ stateFileStream)
268             {
269                 auto stateReader = ref new DataReader(stateFileStream);
270                 return task<unsigned int>(stateReader->LoadAsync(size)).then([=](unsigned int bytesRead)
271                 {
272                     (void)bytesRead; // Unused parameter
273                     // Deserialize the Session State
274                     Object^ content = ReadObject(stateReader);
275                     _sessionState = (Map<String^, Object^>^)content;
276 
277                     // Restore any registered frames to their saved state
278                     for (auto&& weakFrame : _registeredFrames)
279                     {
280                         auto frame = weakFrame->ResolvedFrame;
281                         if (frame != nullptr)
282                         {
283                             frame->ClearValue(FrameSessionStateProperty);
284                             RestoreFrameNavigationState(frame);
285                         }
286                     }
287                 }, task_continuation_context::use_current());
288             });
289         });
290     });
291 }
292 
293 #pragma region Object serialization for a known set of types
294 
295 namespace
296 {
297     // Codes used for identifying serialized types
298     enum StreamTypes {
299         NullPtrType = 0,
300 
301         // Supported IPropertyValue types
302         UInt8Type, UInt16Type, UInt32Type, UInt64Type, Int16Type, Int32Type, Int64Type,
303         SingleType, DoubleType, BooleanType, Char16Type, GuidType, StringType,
304 
305         // Additional supported types
306         StringToObjectMapType,
307 
308         // Marker values used to ensure stream integrity
309         MapEndMarker
310     };
311 
312     void WriteString(DataWriter^ writer, String^ string)
313     {
314         writer->WriteByte(StringType);
315         writer->WriteUInt32(writer->MeasureString(string));
316         writer->WriteString(string);
317     }
318 
319     void WriteProperty(DataWriter^ writer, IPropertyValue^ propertyValue)
320     {
321         switch (propertyValue->Type)
322         {
323         case PropertyType::UInt8:
324             writer->WriteByte(UInt8Type);
325             writer->WriteByte(propertyValue->GetUInt8());
326             return;
327         case PropertyType::UInt16:
328             writer->WriteByte(UInt16Type);
329             writer->WriteUInt16(propertyValue->GetUInt16());
330             return;
331         case PropertyType::UInt32:
332             writer->WriteByte(UInt32Type);
333             writer->WriteUInt32(propertyValue->GetUInt32());
334             return;
335         case PropertyType::UInt64:
336             writer->WriteByte(UInt64Type);
337             writer->WriteUInt64(propertyValue->GetUInt64());
338             return;
339         case PropertyType::Int16:
340             writer->WriteByte(Int16Type);
341             writer->WriteUInt16(propertyValue->GetInt16());
342             return;
343         case PropertyType::Int32:
344             writer->WriteByte(Int32Type);
345             writer->WriteUInt32(propertyValue->GetInt32());
346             return;
347         case PropertyType::Int64:
348             writer->WriteByte(Int64Type);
349             writer->WriteUInt64(propertyValue->GetInt64());
350             return;
351         case PropertyType::Single:
352             writer->WriteByte(SingleType);
353             writer->WriteSingle(propertyValue->GetSingle());
354             return;
355         case PropertyType::Double:
356             writer->WriteByte(DoubleType);
357             writer->WriteDouble(propertyValue->GetDouble());
358             return;
359         case PropertyType::Boolean:
360             writer->WriteByte(BooleanType);
361             writer->WriteBoolean(propertyValue->GetBoolean());
362             return;
363         case PropertyType::Char16:
364             writer->WriteByte(Char16Type);
365             writer->WriteUInt16(propertyValue->GetChar16());
366             return;
367         case PropertyType::Guid:
368             writer->WriteByte(GuidType);
369             writer->WriteGuid(propertyValue->GetGuid());
370             return;
371         case PropertyType::String:
372             WriteString(writer, propertyValue->GetString());
373             return;
374         default:
375             throw ref new InvalidArgumentException("Unsupported property type");
376         }
377     }
378 
379     void WriteStringToObjectMap(DataWriter^ writer, IMap<String^, Object^>^ map)
380     {
381         writer->WriteByte(StringToObjectMapType);
382         writer->WriteUInt32(map->Size);
383         for (auto&& pair : map)
384         {
385             WriteObject(writer, pair->Key);
386             WriteObject(writer, pair->Value);
387         }
388         writer->WriteByte(MapEndMarker);
389     }
390 
391     void WriteObject(DataWriter^ writer, Object^ object)
392     {
393         if (object == nullptr)
394         {
395             writer->WriteByte(NullPtrType);
396             return;
397         }
398 
399         auto propertyObject = dynamic_cast<IPropertyValue^>(object);
400         if (propertyObject != nullptr)
401         {
402             WriteProperty(writer, propertyObject);
403             return;
404         }
405 
406         auto mapObject = dynamic_cast<IMap<String^, Object^>^>(object);
407         if (mapObject != nullptr)
408         {
409             WriteStringToObjectMap(writer, mapObject);
410             return;
411         }
412 
413         throw ref new InvalidArgumentException("Unsupported data type");
414     }
415 
416     String^ ReadString(DataReader^ reader)
417     {
418         int length = reader->ReadUInt32();
419         String^ string = reader->ReadString(length);
420         return string;
421     }
422 
423     IMap<String^, Object^>^ ReadStringToObjectMap(DataReader^ reader)
424     {
425         auto map = ref new Map<String^, Object^>();
426         auto size = reader->ReadUInt32();
427         for (unsigned int index = 0; index < size; index++)
428         {
429             auto key = safe_cast<String^>(ReadObject(reader));
430             auto value = ReadObject(reader);
431             map->Insert(key, value);
432         }
433         if (reader->ReadByte() != MapEndMarker)
434         {
435             throw ref new InvalidArgumentException("Invalid stream");
436         }
437         return map;
438     }
439 
440     Object^ ReadObject(DataReader^ reader)
441     {
442         auto type = reader->ReadByte();
443         switch (type)
444         {
445         case NullPtrType:
446             return nullptr;
447         case UInt8Type:
448             return reader->ReadByte();
449         case UInt16Type:
450             return reader->ReadUInt16();
451         case UInt32Type:
452             return reader->ReadUInt32();
453         case UInt64Type:
454             return reader->ReadUInt64();
455         case Int16Type:
456             return reader->ReadInt16();
457         case Int32Type:
458             return reader->ReadInt32();
459         case Int64Type:
460             return reader->ReadInt64();
461         case SingleType:
462             return reader->ReadSingle();
463         case DoubleType:
464             return reader->ReadDouble();
465         case BooleanType:
466             return reader->ReadBoolean();
467         case Char16Type:
468             return (char16_t)reader->ReadUInt16();
469         case GuidType:
470             return reader->ReadGuid();
471         case StringType:
472             return ReadString(reader);
473         case StringToObjectMapType:
474             return ReadStringToObjectMap(reader);
475         default:
476             throw ref new InvalidArgumentException("Unsupported property type");
477         }
478     }
479 }
480 
481 #pragma endregion
482