123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647 |
-
- namespace NRKernal
- {
- using UnityEngine;
- using System.Collections;
- using System;
- using System.Collections.Generic;
- #if USING_XR_SDK
- using UnityEngine.XR;
- #endif
-
- public delegate void HMDPoseTrackerEvent();
- public delegate void HMDPoseTrackerModeChangeEvent(NRHMDPoseTracker.TrackingType origin, NRHMDPoseTracker.TrackingType target);
- public delegate void OnTrackingModeChanged(NRHMDPoseTracker.TrackingModeChangedResult result);
- public delegate void OnWorldPoseResetEvent();
-
-
-
- public interface IExternSlamProvider
- {
-
-
-
-
-
- Pose GetHeadPoseAtTime(UInt64 timeStamp);
- }
-
-
-
-
-
- [HelpURL("https://developer.nreal.ai/develop/discover/introduction-nrsdk")]
- public class NRHMDPoseTracker : MonoBehaviour
- {
-
- public static event HMDPoseTrackerEvent OnHMDPoseReady;
-
- public static event HMDPoseTrackerEvent OnHMDLostTracking;
-
- public static event HMDPoseTrackerModeChangeEvent OnChangeTrackingMode;
-
- public static event OnWorldPoseResetEvent OnWorldPoseReset;
- public struct TrackingModeChangedResult
- {
- public bool success;
- public TrackingType trackingType;
- }
-
- public enum TrackingType
- {
-
-
-
- Tracking6Dof = 0,
-
-
-
- Tracking3Dof = 1,
-
-
-
- Tracking0Dof = 2,
-
-
-
- Tracking0DofStable = 3
- }
-
- [SerializeField]
- private TrackingType m_TrackingType = TrackingType.Tracking6Dof;
-
-
- public TrackingType TrackingMode
- {
- get
- {
- return m_TrackingType;
- }
- }
-
- public bool TrackingModeAutoAdapt = true;
-
- public bool UseRelative = false;
-
- private LostTrackingReason m_LastReason = LostTrackingReason.INITIALIZING;
- private IExternSlamProvider m_externSlamProvider = null;
-
- public Camera leftCamera;
-
- public Camera centerCamera;
- public Transform centerAnchor;
- Pose HeadRotFromCenter = Pose.identity;
-
- public Camera rightCamera;
- private bool m_ModeChangeLock = false;
- public bool IsTrackModeChanging
- {
- get { return m_ModeChangeLock; }
- }
- #if USING_XR_SDK
- static internal List<XRNodeState> nodeStates = new List<XRNodeState>();
- static internal void GetNodePoseData(XRNode node, out Pose resultPose)
- {
- InputTracking.GetNodeStates(nodeStates);
- for (int i = 0; i < nodeStates.Count; i++)
- {
- var nodeState = nodeStates[i];
- if (nodeState.nodeType == node)
- {
- nodeState.TryGetPosition(out resultPose.position);
- nodeState.TryGetRotation(out resultPose.rotation);
- return;
- }
- }
- resultPose = Pose.identity;
- }
- #endif
-
- void Awake()
- {
- #if UNITY_EDITOR || USING_XR_SDK
- if (leftCamera != null)
- leftCamera.enabled = false;
- if (rightCamera != null)
- rightCamera.enabled = false;
- centerCamera.depth = 1;
- centerCamera.enabled = true;
- #else
- if (leftCamera != null)
- leftCamera.enabled = true;
- if (rightCamera != null)
- rightCamera.enabled = true;
- centerCamera.enabled = false;
- #endif
- StartCoroutine(Initialize());
- }
-
- void OnEnable()
- {
- #if USING_XR_SDK && !UNITY_EDITOR
- Application.onBeforeRender += OnUpdate;
- #else
- NRKernalUpdater.OnUpdate += OnUpdate;
- #endif
- }
-
- void OnDisable()
- {
- #if USING_XR_SDK && !UNITY_EDITOR
- Application.onBeforeRender -= OnUpdate;
- #else
- NRKernalUpdater.OnUpdate -= OnUpdate;
- #endif
- }
-
- void OnUpdate()
- {
- CheckHMDPoseState();
- UpdatePoseByTrackingType();
- }
-
- public void AutoAdaptTrackingType()
- {
- if (TrackingModeAutoAdapt)
- {
- TrackingType adjustTrackingType = AdaptTrackingType(m_TrackingType);
- if (adjustTrackingType != m_TrackingType)
- {
- NRDebugger.Warning("[NRHMDPoseTracker] AutoAdaptTrackingType : {0} => {1}", m_TrackingType, adjustTrackingType);
- m_TrackingType = adjustTrackingType;
- }
- }
- }
-
-
- public static TrackingType AdaptTrackingType(TrackingType mode)
- {
- switch (mode)
- {
- case TrackingType.Tracking6Dof:
- {
- if (NRDevice.Subsystem.IsFeatureSupported(NRSupportedFeature.NR_FEATURE_TRACKING_6DOF))
- return TrackingType.Tracking6Dof;
- else if (NRDevice.Subsystem.IsFeatureSupported(NRSupportedFeature.NR_FEATURE_TRACKING_3DOF))
- return TrackingType.Tracking3Dof;
- else
- return TrackingType.Tracking0Dof;
- }
- case TrackingType.Tracking3Dof:
- {
- if (NRDevice.Subsystem.IsFeatureSupported(NRSupportedFeature.NR_FEATURE_TRACKING_3DOF))
- return TrackingType.Tracking3Dof;
- else
- return TrackingType.Tracking0Dof;
- }
- case TrackingType.Tracking0DofStable:
- {
- if (NRDevice.Subsystem.IsFeatureSupported(NRSupportedFeature.NR_FEATURE_TRACKING_3DOF))
- return TrackingType.Tracking0DofStable;
- else
- return TrackingType.Tracking0Dof;
- }
- }
- return mode;
- }
-
-
-
- private bool ChangeMode(TrackingType trackingtype, OnTrackingModeChanged OnModeChanged)
- {
- NRDebugger.Info("[NRHMDPoseTracker] Begin ChangeMode to:" + trackingtype);
- TrackingModeChangedResult result = new TrackingModeChangedResult();
- if (trackingtype == m_TrackingType || m_ModeChangeLock)
- {
- result.success = false;
- result.trackingType = m_TrackingType;
- OnModeChanged?.Invoke(result);
- NRDebugger.Warning("[NRHMDPoseTracker] Change tracking mode faild...");
- return false;
- }
- OnChangeTrackingMode?.Invoke(m_TrackingType, trackingtype);
- NRSessionManager.OnChangeTrackingMode?.Invoke(m_TrackingType, trackingtype);
- #if !UNITY_EDITOR
- m_ModeChangeLock = true;
- AsyncTaskExecuter.Instance.RunAction(() =>
- {
- result.success = NRSessionManager.Instance.NativeAPI.NativeTracking.SwitchTrackingMode((TrackingMode)trackingtype);
- if (result.success)
- {
- m_TrackingType = trackingtype;
- }
- result.trackingType = m_TrackingType;
- OnModeChanged?.Invoke(result);
- m_ModeChangeLock = false;
- NRDebugger.Info("[NRHMDPoseTracker] End ChangeMode, result:" + result.success);
- });
- #else
- m_TrackingType = trackingtype;
- result.success = true;
- result.trackingType = m_TrackingType;
- OnModeChanged?.Invoke(result);
- #endif
- return true;
- }
- private void OnApplicationPause(bool pause)
- {
- NRDebugger.Info("[NRHMDPoseTracker] OnApplicationPause : pause={0}, headPos={1}, cachedWorldMatrix={2}",
- pause, NRFrame.HeadPose.ToString("F2"), cachedWorldMatrix.ToString());
- if (pause)
- {
- this.CacheWorldMatrix();
- }
- }
- Pose GetCachPose()
- {
- Pose cachePose;
- if (UseRelative)
- cachePose = new Pose(transform.localPosition, transform.localRotation);
- else
- cachePose = new Pose(transform.position, transform.rotation);
-
-
- if (m_TrackingType != TrackingType.Tracking6Dof)
- {
- cachePose.position = ConversionUtility.GetPositionFromTMatrix(cachedWorldMatrix);
- }
- return cachePose;
- }
-
-
-
- public bool ChangeTo6Dof(OnTrackingModeChanged OnModeChanged, bool autoAdapt = false)
- {
- var trackType = TrackingType.Tracking6Dof;
- if (autoAdapt)
- trackType = AdaptTrackingType(trackType);
- Pose cachePose = GetCachPose();
- return ChangeMode(trackType, (NRHMDPoseTracker.TrackingModeChangedResult result) =>
- {
- if (result.success)
- CacheWorldMatrix(cachePose);
- if (OnModeChanged != null)
- OnModeChanged(result);
- });
- }
-
-
-
- public bool ChangeTo3Dof(OnTrackingModeChanged OnModeChanged, bool autoAdapt = false)
- {
- var trackType = TrackingType.Tracking3Dof;
- if (autoAdapt)
- trackType = AdaptTrackingType(trackType);
- Pose cachePose = GetCachPose();
- return ChangeMode(trackType, (NRHMDPoseTracker.TrackingModeChangedResult result) =>
- {
- if (result.success)
- CacheWorldMatrix(cachePose);
- if (OnModeChanged != null)
- OnModeChanged(result);
- });
- }
-
-
-
- public bool ChangeTo0Dof(OnTrackingModeChanged OnModeChanged, bool autoAdapt = false)
- {
- var trackType = TrackingType.Tracking0Dof;
- if (autoAdapt)
- trackType = AdaptTrackingType(trackType);
- Pose cachePose = GetCachPose();
- return ChangeMode(trackType, (NRHMDPoseTracker.TrackingModeChangedResult result) =>
- {
- if (result.success)
- CacheWorldMatrix(cachePose);
- if (OnModeChanged != null)
- OnModeChanged(result);
- });
- }
-
-
-
- public bool ChangeTo0DofStable(OnTrackingModeChanged OnModeChanged, bool autoAdapt = false)
- {
- var trackType = TrackingType.Tracking0DofStable;
- if (autoAdapt)
- trackType = AdaptTrackingType(trackType);
- Pose cachePose = GetCachPose();
- return ChangeMode(trackType, (NRHMDPoseTracker.TrackingModeChangedResult result) =>
- {
- if (result.success)
- CacheWorldMatrix(cachePose);
- if (OnModeChanged != null)
- OnModeChanged(result);
- });
- }
- private Matrix4x4 cachedWorldMatrix = Matrix4x4.identity;
- private float cachedWorldPitch = 0;
-
- public void CacheWorldMatrix()
- {
- Pose cachePose = GetCachPose();
- CacheWorldMatrix(cachePose);
- NRFrame.ResetHeadPose();
- }
- public void CacheWorldMatrix(Pose pose)
- {
- NRDebugger.Info("[NRHMDPoseTracker] CacheWorldMatrix : UseRelative={0}, trackType={1}, pos={2}", UseRelative, m_TrackingType, pose.ToString("F2"));
- Plane horizontal_plane = new Plane(Vector3.up, Vector3.zero);
- Vector3 forward_use_gravity = horizontal_plane.ClosestPointOnPlane(pose.forward).normalized;
- Quaternion rotation_use_gravity = Quaternion.LookRotation(forward_use_gravity, Vector3.up);
- cachedWorldMatrix = ConversionUtility.GetTMatrix(pose.position, rotation_use_gravity);
- cachedWorldPitch = 0;
- NRDebugger.Info("CacheWorldMatrix Adjust : pos={0}, {1}, cachedWorldMatrix={2}",
- pose.position.ToString("F2"), rotation_use_gravity.ToString("F2"), cachedWorldMatrix.ToString());
- OnWorldPoseReset?.Invoke();
- }
-
-
-
-
-
-
-
-
- public void ResetWorldMatrix(bool resetPitch = false)
- {
- Pose pose;
- if (UseRelative)
- pose = new Pose(transform.localPosition, transform.localRotation);
- else
- pose = new Pose(transform.position, transform.rotation);
-
- Matrix4x4 cachedWorldInverse = Matrix4x4.Inverse(cachedWorldMatrix);
- var worldMatrix = ConversionUtility.GetTMatrix(pose.position, pose.rotation);
- var objectMatrix = cachedWorldInverse * worldMatrix;
- var srcPose = new Pose(ConversionUtility.GetPositionFromTMatrix(objectMatrix), ConversionUtility.GetRotationFromTMatrix(objectMatrix));
- Quaternion rotation = Quaternion.identity;
- if (resetPitch)
- {
-
- Vector3 forward = srcPose.forward;
- Vector3 right = Vector3.Cross(forward, Vector3.up);
- Vector3 up = Vector3.Cross(right, forward);
- rotation = Quaternion.LookRotation(forward, up);
- Debug.LogFormat("ResetWorldMatrix: forward={0}, right={1}, up={2}", forward.ToString("F4"), right.ToString("F4"), up.ToString("F4"));
- }
- else
- {
-
- Plane horizontal_plane = new Plane(Vector3.up, Vector3.zero);
- Vector3 forward_use_gravity = horizontal_plane.ClosestPointOnPlane(srcPose.forward).normalized;
- rotation = Quaternion.LookRotation(forward_use_gravity, Vector3.up);
- }
- var matrix = ConversionUtility.GetTMatrix(srcPose.position, rotation);
- cachedWorldMatrix = Matrix4x4.Inverse(matrix);
- cachedWorldPitch = -rotation.eulerAngles.x;
-
-
- UpdatePoseByTrackingType();
-
-
- {
- var correctPos = ConversionUtility.GetPositionFromTMatrix(cachedWorldMatrix);
- var correctRot = ConversionUtility.GetRotationFromTMatrix(cachedWorldMatrix);
- NRDebugger.Info("[NRHMDPoseTracker] ResetWorldMatrix : resetPitch={0}, curPos={1},{2} srcPose={3},{4}, cachedWorldPitch={5}, correctPos={6},{7}",
- resetPitch, pose.position.ToString("F4"), pose.rotation.eulerAngles.ToString("F4"),
- srcPose.position.ToString("F4"), srcPose.rotation.eulerAngles.ToString("F4"),
- cachedWorldPitch, correctPos.ToString("F4"), correctRot.eulerAngles.ToString("F4"));
- }
- OnWorldPoseReset?.Invoke();
- }
-
-
-
-
- public Matrix4x4 GetWorldOffsetMatrixFromNative()
- {
- Matrix4x4 parentTransformMatrix;
- if (UseRelative)
- {
- if (transform.parent == null)
- {
- parentTransformMatrix = Matrix4x4.TRS(Vector3.zero, Quaternion.identity, Vector3.one);
- }
- else
- {
- parentTransformMatrix = Matrix4x4.TRS(transform.parent.position, transform.parent.rotation, Vector3.one);
- }
- }
- else
- {
- parentTransformMatrix = Matrix4x4.TRS(Vector3.zero, Quaternion.identity, Vector3.one);
- }
- return parentTransformMatrix * cachedWorldMatrix;
- }
- public float GetCachedWorldPitch()
- {
- return cachedWorldPitch;
- }
- private Pose ApplyWorldMatrix(Pose pose)
- {
- var objectMatrix = ConversionUtility.GetTMatrix(pose.position, pose.rotation);
- var object_in_world = cachedWorldMatrix * objectMatrix;
- return new Pose(ConversionUtility.GetPositionFromTMatrix(object_in_world),
- ConversionUtility.GetRotationFromTMatrix(object_in_world));
- }
-
-
- private IEnumerator Initialize()
- {
- while (NRFrame.SessionStatus != SessionState.Running)
- {
- NRDebugger.Debug("[NRHMDPoseTracker] Waitting to initialize.");
- yield return new WaitForEndOfFrame();
- }
- #if !UNITY_EDITOR && !USING_XR_SDK
- leftCamera.enabled = false;
- rightCamera.enabled = false;
- bool result;
- var matrix_data = NRFrame.GetEyeProjectMatrix(out result, leftCamera.nearClipPlane, leftCamera.farClipPlane);
- Debug.Assert(result, "[NRHMDPoseTracker] GetEyeProjectMatrix failed.");
- if (result)
- {
- NRDebugger.Info("[NRHMDPoseTracker] Left Camera Project Matrix : {0}", matrix_data.LEyeMatrix.ToString());
- NRDebugger.Info("[NRHMDPoseTracker] RightCamera Project Matrix : {0}", matrix_data.REyeMatrix.ToString());
- leftCamera.projectionMatrix = matrix_data.LEyeMatrix;
- rightCamera.projectionMatrix = matrix_data.REyeMatrix;
-
- centerCamera.projectionMatrix = matrix_data.LEyeMatrix;
- var eyeposeFromHead = NRFrame.EyePoseFromHead;
- leftCamera.transform.localPosition = eyeposeFromHead.LEyePose.position;
- leftCamera.transform.localRotation = eyeposeFromHead.LEyePose.rotation;
- rightCamera.transform.localPosition = eyeposeFromHead.REyePose.position;
- rightCamera.transform.localRotation = eyeposeFromHead.REyePose.rotation;
- centerAnchor.localPosition = centerCamera.transform.localPosition = (leftCamera.transform.localPosition + rightCamera.transform.localPosition) * 0.5f;
- centerAnchor.localRotation = centerCamera.transform.localRotation = Quaternion.Lerp(leftCamera.transform.localRotation, rightCamera.transform.localRotation, 0.5f);
- var centerRotMatrix = ConversionUtility.GetTMatrix(Vector3.zero, centerAnchor.localRotation).inverse;
- HeadRotFromCenter = new Pose(Vector3.zero, ConversionUtility.GetRotationFromTMatrix(centerRotMatrix));
- }
- #endif
- NRDebugger.Info("[NRHMDPoseTracker] Initialized success.");
- }
- public void RegisterSlamProvider(IExternSlamProvider provider)
- {
- NRDebugger.Info("[NRHMDPoseTracker] RegisterSlamProvider");
- m_externSlamProvider = provider;
- }
- public bool GetHeadPoseByTimeInUnityWorld(ref Pose headPose, ulong timeStamp)
- {
- var nativeHeadPose = Pose.identity;
- if (m_TrackingType == TrackingType.Tracking0Dof)
- {
- nativeHeadPose = HeadRotFromCenter;
- }
- else
- {
- if (m_externSlamProvider != null)
- {
- nativeHeadPose = m_externSlamProvider.GetHeadPoseAtTime(timeStamp);
- }
- else
- {
- if (timeStamp == NRFrame.CurrentPoseTimeStamp)
- {
- nativeHeadPose = NRFrame.HeadPose;
- }
- else
- {
- if (!NRFrame.GetHeadPoseByTime(ref nativeHeadPose, timeStamp))
- return false;
- }
- }
- if (m_TrackingType == TrackingType.Tracking0DofStable)
- {
- nativeHeadPose.rotation = HeadRotFromCenter.rotation * nativeHeadPose.rotation;
- }
- }
- headPose = nativeHeadPose;
- headPose = cachedWorldMatrix.Equals(Matrix4x4.identity) ? headPose : ApplyWorldMatrix(headPose);
-
- return true;
- }
-
- private void UpdatePoseByTrackingType()
- {
- Pose headPose = Pose.identity;
- #if USING_XR_SDK && !UNITY_EDITOR
- Pose centerPose;
- GetNodePoseData(XRNode.Head, out headPose);
- headPose = cachedWorldMatrix.Equals(Matrix4x4.identity) ? headPose : ApplyWorldMatrix(headPose);
- GetNodePoseData(XRNode.CenterEye, out centerPose);
- centerCamera.transform.localPosition = Vector3.zero;
- centerCamera.transform.localRotation = Quaternion.identity;
- centerAnchor.localPosition = centerPose.position;
- centerAnchor.localRotation = centerPose.rotation;
- #else
- if (!GetHeadPoseByTimeInUnityWorld(ref headPose, NRFrame.CurrentPoseTimeStamp))
- return;
-
- #endif
- #if !UNITY_EDITOR
- if (UseRelative)
- {
- transform.localRotation = headPose.rotation;
- transform.localPosition = headPose.position;
- }
- else
- {
- transform.rotation = headPose.rotation;
- transform.position = headPose.position;
- }
- #endif
- }
-
- private void CheckHMDPoseState()
- {
- if (NRFrame.SessionStatus != SessionState.Running
- || TrackingMode == TrackingType.Tracking0Dof
- || TrackingMode == TrackingType.Tracking0DofStable
- || IsTrackModeChanging)
- {
- return;
- }
- var currentReason = NRFrame.LostTrackingReason;
-
- if (currentReason != m_LastReason)
- {
- if (currentReason != LostTrackingReason.NONE && m_LastReason == LostTrackingReason.NONE)
- {
- OnHMDLostTracking?.Invoke();
- NRSessionManager.OnHMDLostTracking?.Invoke();
- }
- else if (currentReason == LostTrackingReason.NONE && m_LastReason != LostTrackingReason.NONE)
- {
- OnHMDPoseReady?.Invoke();
- NRSessionManager.OnHMDPoseReady?.Invoke();
- }
- m_LastReason = currentReason;
- }
- }
- }
- }
|