// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using System; using System.Collections.Generic; using System.Threading.Tasks; namespace Microsoft.MixedReality.OpenXR { /// /// Type of tracking maps /// public enum TrackingMapType : uint { /// /// Default tracking map on HoloLens 2, shared between all applications. /// Shared = 0, /// /// New tracking environment private to application, which can be persisted /// and restored with some limitations. /// ApplicationExclusive = 0x00000001 } /// /// TrackingMapManager class allows an application to opt into running in an Application-Exclusive tracking mode instead of default shared environment. /// /// /// /// Activating an Application-Exclusive tracking map creates a brand-new environment for the application, unencumbered by any Device Space /// tracking inaccuracies as the result of degradation over time. /// This is equivalent to using the "Remove All Holograms" command from Settings, but only applicable to the running application. /// Holograms for all other applications (including the HoloLens Shell) remain intact and available as before. /// Returning to the Shell or activating another application will return the HoloLens to the Device Shared tracking mode automatically. /// /// /// When first entering the Application-Exclusive tracking mode, the calling application will be issued a unique identifier /// that can be used to resume tracking the app-specific map in future sessions of the application /// (like if the user switches away from the application and it is terminated in the background due to system resource constraints). /// However, if the device simply goes to sleep or the user briefly interacts with the Shell, /// the application will automatically resume in the Application-Exclusive tracking mode once it is reactivated (and all application state will remain available). /// /// /// There are two limitations to be aware of when using the Application-Exclusive tracking mode: /// /// /// /// Only a single Application-Exclusive tracking map can exist on the HoloLens at one time. If an application requests a new Application-Exclusive tracking mode, /// then any previous Application-Exclusive tracking data would be erased and all SpatialAnchor objects (and attached holograms) would be lost, /// even if the data was created by a completely different application using its own Application-Exclusive tracking mode. /// Therefore, attempting to return to a previous Application-Exclusive map (by specifying the identifier received when this map was created) /// may result in a return value indicating that the previous map was not found. /// Applications must be prepared to handle the scenario where a previous Application-Exclusive tracking map is not available. /// /// /// The disk storage available to the Application-Exclusive tracking mode is limited to one third of what is available for the Device Shared tracking mode, /// although this is unlikely to be an issue for most users. When this limit is reached, HoloLens will begin erasing its least valuable tracking data, /// which will eventually result in poorer tracking accuracy. /// The smaller limit is still large enough to maintain good accuracy for house-sized environments and is unlikely to be a concern for most application scenarios. /// /// /// Given these limitations, the target scenario for the Application-Exclusive tracking mode is for applications with high accuracy requirements that are task oriented, /// where a task may be interrupted by the user returning the HoloLens Shell or the device going to sleep. /// However, once the user's task is complete, nothing about the task (with respect to the 3D environment) needs to be saved and so can be erased. /// /// Examples: /// /// /// High-accuracy alignment of holograms to a real-world object, using QR codes to bootstrap the scenario. /// /// /// Editing a 3D model with high-accuracy requirements when no 3D spatial persistence of the model needs to occur after the session ends. /// /// /// Tracking in places that have a lot of environmental churn (like people moving around), which sometimes results in poorer tracking quality than more static environments. /// /// /// public class TrackingMapManager { private TrackingMapSubsystem m_trackingMapSubsystem; private TrackingMapManager(TrackingMapSubsystem trackingMapSubsystem) { m_trackingMapSubsystem = trackingMapSubsystem; } /// /// Gets an instance of TrackingMapManager. /// /// Task<TrackingMapManager> which is completed when the Tracking Map Manager is fully initialized. /// The Result property of the task is a TrackingMapManager instance. /// Use IsSupported to check if the HoloLens 2 /// on which the application is currently executing supports Application-Exclusive maps. public static Task GetAsync() { Task result = Task.Run(() => { TrackingMapSubsystem trackingMapSubsystem = TrackingMapSubsystem.TryCreateTrackingMapSubsystem(); return new TrackingMapManager(trackingMapSubsystem); }); return result; } /// /// Indicates if a tracking map type is supported /// /// Tracking map type to check for support /// True if the tracking map type is supported. /// /// TrackingMapType.Shared is always supported. /// public bool IsSupported(TrackingMapType trackingMapType) { switch (trackingMapType) { case TrackingMapType.Shared: return true; case TrackingMapType.ApplicationExclusive: return (m_trackingMapSubsystem != null) && m_trackingMapSubsystem.SupportsApplicationExclusiveMaps(); default: return false; } } /// /// Indicates the active tracking map type. /// public TrackingMapType ActiveTrackingMapType { get => (m_trackingMapSubsystem != null) ? m_trackingMapSubsystem.ActiveTrackingMapType : TrackingMapType.Shared; } /// /// Creates and activates a new application-exclusive map. /// /// Optional identifier to try reactivating a previously created map. If null is passed, a new map is created. /// If a non empty guid map is passed, the tracking manager tries to load the corresponding map. /// If the map cannot be found or loaded, a new map is created. /// Task<Guid> which is completed when the system has created and activated an application-exclusive map. /// The Result property contains the guid of the activated map. If it is equal to existingMapId, it means that the existing map has been reloaded. /// If not, it means that a new map has been created. /// Thrown when the TrackingMapManager does not support application-exclusive Maps /// or if a map change is already in progress. /// In order to take effect, the application must be the active, immersive 3D application. /// ActivateApplicationExclusiveMapAsync should only be called once the application has started rendering its 3D user interface (and not, for example, when the application first starts). /// public Task ActivateApplicationExclusiveMapAsync(Guid? existingMapId = null) { if (m_trackingMapSubsystem == null) throw new InvalidOperationException("Feature not supported."); Task result = Task.Run(() => { return m_trackingMapSubsystem.ActivateApplicationExclusiveMap(existingMapId); }); return result; } /// /// Leaves the current application-exclusive map and returns to the default shared map. /// If the device was already in the default shared map, this method does nothing. /// /// Task which is completed when the system has returned to the default map. /// Thrown when a map change is already in progress. public Task ActivateSharedMapAsync() { if (m_trackingMapSubsystem == null) throw new InvalidOperationException("Feature not supported."); Task result = Task.Run(() => { m_trackingMapSubsystem.ActivateSharedMapAsync(); }); return result; } } }