// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using UnityEngine; namespace Microsoft.MixedReality.OpenXR { /// /// Represents different possible hand poses. /// /// See https://www.khronos.org/registry/OpenXR/specs/1.0/html/xrspec.html#XrHandPoseTypeMSFT for more information. public enum HandPoseType { /// /// Represents a hand pose provided by actual tracking of the user's hand. /// Tracked = 0, /// /// Represents a stable reference hand pose in a relaxed open hand shape. /// ReferenceOpenPalm, } /// /// Represents a user's hand and the ability to render a hand mesh representation of it. /// public class HandMeshTracker { /// /// The user's left hand. /// public static HandMeshTracker Left { get; } = new HandMeshTracker(Handedness.Left); /// /// The user's right hand. /// public static HandMeshTracker Right { get; } = new HandMeshTracker(Handedness.Right); private HandTrackingFeaturePlugin Feature => OpenXRFeaturePlugin.Feature; private readonly Handedness m_handedness; private Vector3[] m_handMeshVertices = null; private Vector3[] m_handMeshNormals = null; private int[] m_handMeshIndices = null; private Mesh m_currentMesh = null; private uint m_indexBufferKey = 0; private ulong m_vertexBufferkey = 0; private HandMeshTracker(Handedness trackerHandedness) { m_handedness = trackerHandedness; } /// /// Tries to get the current location in world-space of the specified hand mesh. /// /// Specify the to locate the hand mesh. /// The current pose of the specified hand mesh. /// The type of hand mesh pose to request. The tracked pose represents the actively tracked hand. The reference pose represents a stable hand pose in a relaxed open hand shape. /// Returns true when the returned pose is tracking and valid to be used. /// Returns false when the hand mesh tracker lost tracking or it's not properly set up. public bool TryLocateHandMesh(FrameTime frameTime, out Pose pose, HandPoseType handPoseType = HandPoseType.Tracked) { pose = Pose.identity; return Feature.IsValidAndEnabled() && OpenXRContext.Current.IsSessionRunning && NativeLib.TryLocateHandMesh(m_handedness, frameTime, handPoseType, out pose); } /// /// Retrieves the latest hand mesh information and build the current hand mesh in the passed-in mesh parameter. /// /// Specify the to locate the hand mesh. /// The mesh object to build the hand mesh in. /// The type of hand mesh to request. The tracked pose represents the actively tracked hand. The reference pose represents a stable hand pose in a relaxed open hand shape. /// True if the mesh was retrievable. public bool TryGetHandMesh(FrameTime frameTime, Mesh handMesh, HandPoseType handPoseType = HandPoseType.Tracked) { if (!Feature.IsValidAndEnabled() || !OpenXRContext.Current.IsSessionRunning) { return false; // Hand tracking feature is not enabled. Return the tracker not tracking. } try { if (m_handMeshVertices == null || m_handMeshNormals == null || m_handMeshIndices == null) { if (NativeLib.TryGetHandMeshBufferSizes(out uint maxVertexCount, out uint maxIndexCount)) { m_handMeshVertices = new Vector3[maxVertexCount]; m_handMeshNormals = new Vector3[maxVertexCount]; m_handMeshIndices = new int[maxIndexCount]; } else { return false; } } if (m_currentMesh != handMesh) { m_currentMesh = handMesh; m_indexBufferKey = 0; m_vertexBufferkey = 0; } if (NativeLib.TryGetHandMesh(m_handedness, frameTime, handPoseType, ref m_vertexBufferkey, out uint vertexCount, m_handMeshVertices, m_handMeshNormals, ref m_indexBufferKey, out uint indexCount, m_handMeshIndices)) { // The NativeLib call will return a count of 0 if no change was made if (vertexCount > 0) { handMesh.SetVertices(m_handMeshVertices, 0, (int)vertexCount); handMesh.SetNormals(m_handMeshNormals, 0, (int)vertexCount); } if (indexCount > 0) { handMesh.SetTriangles(m_handMeshIndices, 0, (int)indexCount, 0); } return true; } else { return false; } } catch (System.DllNotFoundException) { return false; } } } }