using System; using System.Collections.Generic; using System.Runtime.CompilerServices; using UnityEngine.Experimental.XR.Interaction; #if ENABLE_AR || ENABLE_VR using UnityEngine.XR; #endif [assembly: InternalsVisibleTo("UnityEditor.SpatialTracking")] namespace UnityEngine.SpatialTracking { internal class TrackedPoseDriverDataDescription { internal struct PoseData { public List PoseNames; public List Poses; } internal static List DeviceData = new List { // Generic XR Device new PoseData { PoseNames = new List { "Left Eye", "Right Eye", "Center Eye - HMD Reference", "Head", "Color Camera" }, Poses = new List { TrackedPoseDriver.TrackedPose.LeftEye, TrackedPoseDriver.TrackedPose.RightEye, TrackedPoseDriver.TrackedPose.Center, TrackedPoseDriver.TrackedPose.Head, TrackedPoseDriver.TrackedPose.ColorCamera } }, // generic controller new PoseData { PoseNames = new List { "Left Controller", "Right Controller" }, Poses = new List { TrackedPoseDriver.TrackedPose.LeftPose, TrackedPoseDriver.TrackedPose.RightPose } }, // generic remote new PoseData { PoseNames = new List { "Device Pose" }, Poses = new List { TrackedPoseDriver.TrackedPose.RemotePose, } }, }; } /// /// Bitflag enum which represents what data was set on an associated Pose struct /// [Flags] public enum PoseDataFlags { /// /// No data was actually set on the pose /// NoData = 0, /// /// If this flag is set, position data was updated on the associated pose struct /// Position = 1 << 0, /// /// If this flag is set, rotation data was updated on the associated pose struct /// Rotation = 1 << 1, } /// /// The PoseDataSource class acts as a container for the GetDataFromSource method call that should be used by PoseProviders wanting to query data for a particular pose. /// public static class PoseDataSource { #if ENABLE_AR || ENABLE_VR static internal List nodeStates = new List(); static internal PoseDataFlags GetNodePoseData(XR.XRNode node, out Pose resultPose) { PoseDataFlags retData = PoseDataFlags.NoData; XR.InputTracking.GetNodeStates(nodeStates); foreach (XR.XRNodeState nodeState in nodeStates) { if (nodeState.nodeType == node) { if (nodeState.TryGetPosition(out resultPose.position)) { retData |= PoseDataFlags.Position; } if (nodeState.TryGetRotation(out resultPose.rotation)) { retData |= PoseDataFlags.Rotation; } return retData; } } resultPose = Pose.identity; return retData; } #endif /// The GetDataFromSource method is used to query data from the XRNode subsystem based on the provided pose source. /// The pose source to request data for. /// The resulting pose data. /// True, if the pose source is valid, otherwise false. public static bool TryGetDataFromSource(TrackedPoseDriver.TrackedPose poseSource, out Pose resultPose) { return GetDataFromSource(poseSource, out resultPose) == (PoseDataFlags.Position | PoseDataFlags.Rotation); } /// The GetDataFromSource method is used to query data from the XRNode subsystem based on the provided pose source. /// The pose source to request data for. /// The resulting pose data. This function will return the Center Eye pose if the Color Camera pose is not available. /// Returns a bitflag which represents which data has been retrieved from the provided pose source public static PoseDataFlags GetDataFromSource(TrackedPoseDriver.TrackedPose poseSource, out Pose resultPose) { #if ENABLE_AR || ENABLE_VR switch (poseSource) { case TrackedPoseDriver.TrackedPose.RemotePose: { PoseDataFlags retFlags = GetNodePoseData(XR.XRNode.RightHand, out resultPose); if (retFlags == PoseDataFlags.NoData) return GetNodePoseData(XR.XRNode.LeftHand, out resultPose); return retFlags; } case TrackedPoseDriver.TrackedPose.LeftEye: { return GetNodePoseData(XR.XRNode.LeftEye, out resultPose); } case TrackedPoseDriver.TrackedPose.RightEye: { return GetNodePoseData(XR.XRNode.RightEye, out resultPose); } case TrackedPoseDriver.TrackedPose.Head: { return GetNodePoseData(XR.XRNode.Head, out resultPose); } case TrackedPoseDriver.TrackedPose.Center: { return GetNodePoseData(XR.XRNode.CenterEye, out resultPose); } case TrackedPoseDriver.TrackedPose.LeftPose: { return GetNodePoseData(XR.XRNode.LeftHand, out resultPose); } case TrackedPoseDriver.TrackedPose.RightPose: { return GetNodePoseData(XR.XRNode.RightHand, out resultPose); } case TrackedPoseDriver.TrackedPose.ColorCamera: { // We fall back to CenterEye because we can't currently extend the XRNode structure, nor are we ready to replace it. return GetNodePoseData(XR.XRNode.CenterEye, out resultPose); } default: { Debug.LogWarningFormat("Unable to retrieve pose data for poseSource: {0}", poseSource.ToString()); break; } } #endif resultPose = Pose.identity; return PoseDataFlags.NoData; } } // The DefaultExecutionOrder is needed because TrackedPoseDriver does some // of its work in regular Update and FixedUpdate calls, but this needs to // be done before regular user scripts have their own Update and // FixedUpdate calls, in order that they correctly get the values for this // frame and not the previous. // -32000 is the minimal possible execution order value; -30000 makes it // unlikely users chose lower values for their scripts by accident, but // still allows for the possibility. /// /// The TrackedPoseDriver component applies the current Pose value of a tracked device to the transform of the GameObject. /// TrackedPoseDriver can track multiple types of devices including XR HMDs, controllers, and remotes. /// [DefaultExecutionOrder(-30000)] [Serializable] [AddComponentMenu("XR/Tracked Pose Driver")] [HelpURL("https://docs.unity3d.com/Packages/com.unity.xr.legacyinputhelpers@2.1/manual/index.html")] public class TrackedPoseDriver : MonoBehaviour { /// /// The device being tracked by the tracked pose driver /// public enum DeviceType { /// /// An XR Controller, use this value for controllers /// GenericXRDevice = 0, /// /// An Generic XR Devices, use this value for HMD and AR Mobile device tracking /// GenericXRController = 1, /// /// An XR Remote, use this value for mobile remotes /// GenericXRRemote = 2 } /// /// The list of endpoints that users can track with the /// public enum TrackedPose { /// /// The left eye of a HMD style device /// LeftEye = 0, /// /// The right eye of a HMD style device /// RightEye = 1, /// /// The center eye of a HMD style device, this is usually the default for most HMDs /// Center = 2, /// /// The head eye of a HMD style device /// Head = 3, /// /// The left hand controller pose /// LeftPose = 4, /// /// The right hand controller pose /// RightPose = 5, /// /// The color camera of a mobile device /// ColorCamera = 6, /// /// No Longer Used /// DepthCameraDeprecated = 7, /// /// No Longer Used /// FisheyeCameraDeprected = 8, /// /// No Longer Used /// DeviceDeprecated = 9, /// /// The pose of a mobile remote /// RemotePose = 10, } [SerializeField] DeviceType m_Device; /// /// This is used to indicate which pose the TrackedPoseDriver is currently tracking. /// public DeviceType deviceType { get { return m_Device; } internal set { m_Device = value; } } [SerializeField] TrackedPose m_PoseSource = TrackedPoseDriver.TrackedPose.Center; /// /// The pose being tracked by the tracked pose driver /// public TrackedPose poseSource { get { return m_PoseSource; } internal set { m_PoseSource = value; } } /// /// This method is used to set the device / pose pair for the SpatialTracking.TrackedPoseDriver. setting an invalid combination of these values will return false. /// /// The device type that we wish to track /// The pose source that we wish to track /// true if the values provided are sensible, otherwise false public bool SetPoseSource(DeviceType deviceType, TrackedPose pose) { if ((int)deviceType < TrackedPoseDriverDataDescription.DeviceData.Count) { TrackedPoseDriverDataDescription.PoseData val = TrackedPoseDriverDataDescription.DeviceData[(int)deviceType]; for (int i = 0; i < val.Poses.Count; ++i) { if (val.Poses[i] == pose) { this.deviceType = deviceType; poseSource = pose; return true; } } } return false; } [SerializeField] BasePoseProvider m_PoseProviderComponent; /// /// Optional: This field holds the reference to the BasePoseProvider instance that, if set, will be used to override the behavior of /// the TrackedPoseDriver. When this field is empty, the TrackedPoseDriver will operate as per usual, with pose data being /// retrieved from the device or pose settings of the TrackedPoseDriver. When this field is set, the pose data will be /// provided by the attached BasePoseProvider. The device or pose fields will be hidden as they are no longer used to /// control the parent GameObject Transform. /// public BasePoseProvider poseProviderComponent { get { return m_PoseProviderComponent; } set { m_PoseProviderComponent = value; } } PoseDataFlags GetPoseData(DeviceType device, TrackedPose poseSource, out Pose resultPose) { return m_PoseProviderComponent != null ? m_PoseProviderComponent.GetPoseFromProvider(out resultPose) : PoseDataSource.GetDataFromSource(poseSource, out resultPose); } /// /// This enum is used to indicate which parts of the pose will be applied to the parent transform /// public enum TrackingType { /// /// With this setting, both the pose's rotation and position will be applied to the parent transform /// RotationAndPosition, /// /// With this setting, only the pose's rotation will be applied to the parent transform /// RotationOnly, /// /// With this setting, only the pose's position will be applied to the parent transform /// PositionOnly } [SerializeField] TrackingType m_TrackingType; /// /// The tracking type being used by the tracked pose driver /// public TrackingType trackingType { get { return m_TrackingType; } set { m_TrackingType = value; } } /// /// The update type being used by the tracked pose driver /// public enum UpdateType { /// /// Sample input at both update, and directly before rendering. For smooth head pose tracking, /// we recommend using this value as it will provide the lowest input latency for the device. /// This is the default value for the UpdateType option /// UpdateAndBeforeRender, /// /// Only sample input during the update phase of the frame. /// Update, /// /// Only sample input directly before rendering /// BeforeRender, } [SerializeField] UpdateType m_UpdateType = UpdateType.UpdateAndBeforeRender; /// /// The update type being used by the tracked pose driver /// public UpdateType updateType { get { return m_UpdateType; } set { m_UpdateType = value; } } [SerializeField] bool m_UseRelativeTransform = false; /// /// This is used to indicate whether the TrackedPoseDriver will use the object's original transform as its basis. /// public bool UseRelativeTransform { get { return m_UseRelativeTransform; } set { m_UseRelativeTransform = value; } } /// /// The origin pose is the offset applied to any tracking data. This is only used when in legacy compatibility mode. /// protected Pose m_OriginPose; /// /// originPose is an offset applied to any tracking data read from this object. /// Setting this value should be reserved for dealing with edge-cases, such as /// achieving parity between room-scale (floor centered) and stationary (head centered) /// tracking - without having to alter the transform hierarchy. /// For user locomotion and gameplay purposes you are usually better off just /// moving the parent transform of this object. /// public Pose originPose { get { return m_OriginPose; } set { m_OriginPose = value; } } private void CacheLocalPosition() { m_OriginPose.position = transform.localPosition; m_OriginPose.rotation = transform.localRotation; } private void ResetToCachedLocalPosition() { SetLocalTransform(m_OriginPose.position, m_OriginPose.rotation, PoseDataFlags.Position | PoseDataFlags.Rotation); } /// protected virtual void Awake() { CacheLocalPosition(); #if UNITY_2019_3_OR_NEWER // deprecated functionality in 2020.1 #elif ENABLE_AR || ENABLE_VR if (HasStereoCamera()) { XRDevice.DisableAutoXRCameraTracking(GetComponent(), true); } #endif } /// protected virtual void OnDestroy() { #if UNITY_2019_3_OR_NEWER // deprecated functionality in 2020.1 #elif ENABLE_AR || ENABLE_VR if (HasStereoCamera()) { XRDevice.DisableAutoXRCameraTracking(GetComponent(), false); } #endif } /// protected virtual void OnEnable() { Application.onBeforeRender += OnBeforeRender; } /// protected virtual void OnDisable() { // remove delegate registration ResetToCachedLocalPosition(); Application.onBeforeRender -= OnBeforeRender; } /// protected virtual void FixedUpdate() { if (m_UpdateType == UpdateType.Update || m_UpdateType == UpdateType.UpdateAndBeforeRender) { PerformUpdate(); } } /// protected virtual void Update() { if (m_UpdateType == UpdateType.Update || m_UpdateType == UpdateType.UpdateAndBeforeRender) { PerformUpdate(); } } /// // For the same reason as DefaultExecutionOrder, a callback order is specified to // apply the pose to the Transform before default user scripts execute. [BeforeRenderOrder(-30000)] protected virtual void OnBeforeRender() { if (m_UpdateType == UpdateType.BeforeRender || m_UpdateType == UpdateType.UpdateAndBeforeRender) { PerformUpdate(); } } /// /// Sets the transform that is being driven by the . will only correct set the rotation or position depending on the /// /// The position to apply. /// The rotation to apply. /// The flags indiciating which of the position/rotation values are provided by the calling code. protected virtual void SetLocalTransform(Vector3 newPosition, Quaternion newRotation, PoseDataFlags poseFlags) { if ((m_TrackingType == TrackingType.RotationAndPosition || m_TrackingType == TrackingType.RotationOnly) && (poseFlags & PoseDataFlags.Rotation) > 0) { transform.localRotation = newRotation; } if ((m_TrackingType == TrackingType.RotationAndPosition || m_TrackingType == TrackingType.PositionOnly) && (poseFlags & PoseDataFlags.Position) > 0) { transform.localPosition = newPosition; } } /// /// This is only used when running in legacy mode, and will fake the behavior of the old implicit camera tracking. This will transform by the origin pose if necessary. /// /// Pose to transform by the origin if in relative transform mode. /// The pose, with the applied transform if in Relative Transform mode. protected Pose TransformPoseByOriginIfNeeded(Pose pose) { if (m_UseRelativeTransform) { return pose.GetTransformedBy(m_OriginPose); } else { return pose; } } private bool HasStereoCamera() { Camera camera = GetComponent(); return camera != null && camera.stereoEnabled; } /// /// PerformUpdate queries the data from the selected pose source, and then calls to apply the pose. /// protected virtual void PerformUpdate() { if (!enabled) return; Pose currentPose; PoseDataFlags poseFlags = GetPoseData(m_Device, m_PoseSource, out currentPose); if (poseFlags != PoseDataFlags.NoData) { Pose localPose = TransformPoseByOriginIfNeeded(currentPose); SetLocalTransform(localPose.position, localPose.rotation, poseFlags); } } } }