// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using Microsoft.MixedReality.Toolkit.Utilities; using System.Collections.Generic; using Unity.Profiling; using UnityEngine; namespace Microsoft.MixedReality.Toolkit.Input { /// /// Defines the interactions and data that an articulated hand can provide. /// public class ArticulatedHandDefinition : BaseInputSourceDefinition { /// /// Constructor. /// /// The input source backing this definition instance. Used for raising events. /// The handedness that this definition instance represents. public ArticulatedHandDefinition(IMixedRealityInputSource source, Handedness handedness) : base(handedness) { InputSource = source; } /// /// The input source backing this definition instance. /// protected IMixedRealityInputSource InputSource { get; } private readonly float cursorBeamBackwardTolerance = 0.5f; private readonly float cursorBeamUpTolerance = 0.8f; private IDictionary unityJointPoseDictionary = new Dictionary(); private MixedRealityPose[] unityJointPoses = null; private MixedRealityPose currentIndexPose = MixedRealityPose.ZeroIdentity; private Vector3 currentPalmNormal = Vector3.zero; private const int PalmIndex = (int)TrackedHandJoint.Palm; private const int ThumbTipIndex = (int)TrackedHandJoint.ThumbTip; private const int IndexKnuckleIndex = (int)TrackedHandJoint.IndexKnuckle; private const int IndexTipIndex = (int)TrackedHandJoint.IndexTip; // Minimum distance between the index and the thumb tip required to enter a pinch private const float MinimumPinchDistance = 0.015f; // Maximum distance between the index and thumb tip required to exit the pinch gesture private const float MaximumPinchDistance = 0.1f; // Default enterPinchDistance value private float enterPinchDistance = 0.02f; /// /// The distance between the index finger tip and the thumb tip required to enter the pinch/air tap selection gesture. /// The pinch gesture enter will be registered for all values less than the EnterPinchDistance. The default EnterPinchDistance value is 0.02 and must be between 0.015 and 0.1. /// public float EnterPinchDistance { get => enterPinchDistance; set { if (value >= MinimumPinchDistance && value <= MaximumPinchDistance) { enterPinchDistance = value; } else { Debug.LogError($"EnterPinchDistance must be between {MinimumPinchDistance} and {MaximumPinchDistance}."); } } } // Default exitPinchDistance value private float exitPinchDistance = 0.05f; /// /// The distance between the index finger tip and the thumb tip required to exit the pinch/air tap gesture. /// The pinch gesture exit will be registered for all values greater than the ExitPinchDistance. The default ExitPinchDistance value is 0.05 and must be between 0.015 and 0.1. /// public float ExitPinchDistance { get => exitPinchDistance; set { if (value >= MinimumPinchDistance && value <= MaximumPinchDistance) { exitPinchDistance = value; } else { Debug.LogError($"ExitPinchDistance must be between {MinimumPinchDistance} and {MaximumPinchDistance}."); } } } /// /// The articulated hands default interactions. /// /// A single interaction mapping works for both left and right articulated hands. [System.Obsolete("Call GetDefaultMappings(Handedness) instead.")] public MixedRealityInteractionMapping[] DefaultInteractions { get { MixedRealityInteractionMapping[] defaultInteractions = new MixedRealityInteractionMapping[DefaultMappings.Length]; for (int i = 0; i < DefaultMappings.Length; i++) { defaultInteractions[i] = new MixedRealityInteractionMapping((uint)i, DefaultMappings[i]); } return defaultInteractions; } } /// /// The articulated hands default interactions. /// /// A single interaction mapping works for both left and right articulated hands. protected override MixedRealityInputActionMapping[] DefaultMappings => new[] { new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer), new MixedRealityInputActionMapping("Spatial Grip", AxisType.SixDof, DeviceInputType.SpatialGrip), new MixedRealityInputActionMapping("Select", AxisType.Digital, DeviceInputType.Select), new MixedRealityInputActionMapping("Grab", AxisType.SingleAxis, DeviceInputType.GripPress), new MixedRealityInputActionMapping("Index Finger Pose", AxisType.SixDof, DeviceInputType.IndexFinger), new MixedRealityInputActionMapping("Teleport Pose", AxisType.DualAxis, DeviceInputType.ThumbStick), }; // Internal calculation of what the HandRay should be // Useful as a fallback for hand ray data protected virtual IHandRay HandRay { get; } = new HandRay(); /// /// Calculates whether the current pose allows for pointing/distant interactions. /// Equivalent to the HandRay's ShouldShowRay implementation /// public bool IsInPointingPose { get { if (unityJointPoses != null) { if (cursorBeamBackwardTolerance >= 0 && CameraCache.Main != null && Vector3.Dot(currentPalmNormal.normalized, -CameraCache.Main.transform.forward) > cursorBeamBackwardTolerance) { return false; } if (cursorBeamUpTolerance >= 0 && Vector3.Dot(currentPalmNormal, Vector3.up) > cursorBeamUpTolerance) { return false; } } return !IsInTeleportPose; } } /// /// Calculates whether the current pose is the one to start a teleport action /// protected bool IsInTeleportPose { get { if (unityJointPoses == null) { return false; } Camera mainCamera = CameraCache.Main; if (mainCamera == null) { return false; } Transform cameraTransform = mainCamera.transform; // We check if the palm up is roughly in line with the camera up return Vector3.Dot(currentPalmNormal, cameraTransform.up) > 0.6f // Thumb must be extended, and middle must be grabbing && !isThumbGrabbing && isMiddleGrabbing; } } /// /// A bool tracking whether the hand definition is pinch or not /// private bool isPinching = false; /// /// Calculates whether the current the current joint pose is selecting (air tap gesture). /// public bool IsPinching { get { if (unityJointPoses != null) { float distance = Vector3.Distance(unityJointPoses[ThumbTipIndex].Position, currentIndexPose.Position); if (isPinching && distance > ExitPinchDistance) { isPinching = false; } else if (!isPinching && distance < EnterPinchDistance) { isPinching = true; } } else { isPinching = false; } return isPinching; } } public bool IsGrabbing => isIndexGrabbing && isMiddleGrabbing; private bool isIndexGrabbing; private bool isMiddleGrabbing; private bool isThumbGrabbing; // Velocity internal states private float deltaTimeStart; private const int VelocityUpdateInterval = 6; private int frameOn = 0; private readonly Vector3[] velocityPositionsCache = new Vector3[VelocityUpdateInterval]; private readonly Vector3[] velocityNormalsCache = new Vector3[VelocityUpdateInterval]; private Vector3 velocityPositionsSum = Vector3.zero; private Vector3 velocityNormalsSum = Vector3.zero; public Vector3 AngularVelocity { get; protected set; } public Vector3 Velocity { get; protected set; } private static readonly ProfilerMarker UpdateHandJointsPerfMarker = new ProfilerMarker("[MRTK] ArticulatedHandDefinition.UpdateHandJoints"); #region Hand Definition Update functions /// /// Updates the current hand joints with new data. /// /// The new joint poses. public void UpdateHandJoints(Dictionary jointPoses) { using (UpdateHandJointsPerfMarker.Auto()) { unityJointPoseDictionary = jointPoses; _ = unityJointPoseDictionary.TryGetValue(TrackedHandJoint.IndexTip, out currentIndexPose); if (unityJointPoseDictionary.TryGetValue(TrackedHandJoint.Palm, out MixedRealityPose palmPose)) { currentPalmNormal = palmPose.Rotation * Vector3.down; } if (unityJointPoses == null) { unityJointPoses = new MixedRealityPose[ArticulatedHandPose.JointCount]; } for (int i = 1; i < ArticulatedHandPose.JointCount; i++) { unityJointPoseDictionary.TryGetValue((TrackedHandJoint)i, out unityJointPoses[i]); } CoreServices.InputSystem?.RaiseHandJointsUpdated(InputSource, Handedness, unityJointPoseDictionary); } } /// /// Updates the current hand joints with new data. /// /// The new joint poses. public void UpdateHandJoints(MixedRealityPose[] jointPoses) { using (UpdateHandJointsPerfMarker.Auto()) { unityJointPoses = jointPoses; if (unityJointPoses == null) { return; } currentIndexPose = unityJointPoses[IndexTipIndex]; currentPalmNormal = unityJointPoses[PalmIndex].Rotation * Vector3.down; for (int i = 1; i < ArticulatedHandPose.JointCount; i++) { unityJointPoseDictionary[(TrackedHandJoint)i] = unityJointPoses[i]; } CoreServices.InputSystem?.RaiseHandJointsUpdated(InputSource, Handedness, unityJointPoseDictionary); } } private static readonly ProfilerMarker UpdateCurrentIndexPosePerfMarker = new ProfilerMarker("[MRTK] ArticulatedHandDefinition.UpdateCurrentIndexPose"); /// /// Updates the MixedRealityInteractionMapping with the latest index pose and fires a corresponding pose event. /// /// The index finger's interaction mapping. public void UpdateCurrentIndexPose(MixedRealityInteractionMapping interactionMapping) { using (UpdateCurrentIndexPosePerfMarker.Auto()) { if (unityJointPoses != null) { // Update the interaction data source interactionMapping.PoseData = currentIndexPose; // If our value changed raise it if (interactionMapping.Changed) { // Raise input system event if it's enabled CoreServices.InputSystem?.RaisePoseInputChanged(InputSource, Handedness, interactionMapping.MixedRealityInputAction, currentIndexPose); } } } } // Used to track the input that was last raised private bool previousReadyToTeleport = false; private IMixedRealityTeleportPointer teleportPointer; private static readonly ProfilerMarker UpdateCurrentTeleportPosePerfMarker = new ProfilerMarker("[MRTK] ArticulatedHandDefinition.UpdateCurrentTeleportPose"); /// /// Updates the MixedRealityInteractionMapping with the latest teleport pose status and fires an event when appropriate /// /// The teleport action's interaction mapping. public void UpdateCurrentTeleportPose(MixedRealityInteractionMapping interactionMapping) { using (UpdateCurrentTeleportPosePerfMarker.Auto()) { // Check if we're focus locked or near something interactive to avoid teleporting unintentionally. bool anyPointersLockedWithHand = false; for (int i = 0; i < InputSource?.Pointers?.Length; i++) { IMixedRealityPointer mixedRealityPointer = InputSource.Pointers[i]; if (mixedRealityPointer.IsNull()) continue; if (mixedRealityPointer is IMixedRealityNearPointer nearPointer) { anyPointersLockedWithHand |= nearPointer.IsNearObject; } anyPointersLockedWithHand |= mixedRealityPointer.IsFocusLocked; // If official teleport mode and we have a teleport pointer registered, we get the input action to trigger it. if (teleportPointer == null && mixedRealityPointer is IMixedRealityTeleportPointer pointer) { teleportPointer = pointer; } } // We close middle finger to signal spider-man gesture, and as being ready for teleport isIndexGrabbing = HandPoseUtils.IsIndexGrabbing(Handedness); isMiddleGrabbing = HandPoseUtils.IsMiddleGrabbing(Handedness); isThumbGrabbing = HandPoseUtils.IsThumbGrabbing(Handedness); bool isReadyForTeleport = !anyPointersLockedWithHand && IsInTeleportPose; // Tracks the input vector that should be sent out based on the gesture that is made Vector2 stickInput = (isReadyForTeleport && !isIndexGrabbing) ? Vector2.up : Vector2.zero; // The teleport event needs to be canceled if we have not completed the teleport motion and we were previously ready to teleport, but for some reason we // are no longer doing the ready to teleport gesture bool teleportCanceled = previousReadyToTeleport && !isReadyForTeleport && !isIndexGrabbing; if (teleportCanceled && teleportPointer != null) { CoreServices.TeleportSystem?.RaiseTeleportCanceled(teleportPointer, null); previousReadyToTeleport = isReadyForTeleport; return; } // Update the interaction data source interactionMapping.Vector2Data = stickInput; // If our value changed raise it if (interactionMapping.Changed) { CoreServices.InputSystem?.RaisePositionInputChanged(InputSource, Handedness, interactionMapping.MixedRealityInputAction, stickInput); } previousReadyToTeleport = isReadyForTeleport; } } /// /// Updates the MixedRealityInteractionMapping with the latest pointer pose status and fires a corresponding pose event. /// /// The pointer pose's interaction mapping. public void UpdatePointerPose(MixedRealityInteractionMapping interactionMapping) { if (unityJointPoses == null) return; Vector3 rayPosition = unityJointPoses[IndexKnuckleIndex].Position; HandRay.Update(rayPosition, currentPalmNormal, CameraCache.Main.transform, Handedness); Ray ray = HandRay.Ray; // Update the interaction data source interactionMapping.PoseData = new MixedRealityPose(ray.origin, Quaternion.LookRotation(ray.direction)); // If our value changed raise it if (interactionMapping.Changed) { // Raise input system event if it's enabled CoreServices.InputSystem?.RaisePoseInputChanged(InputSource, Handedness, interactionMapping.MixedRealityInputAction, interactionMapping.PoseData); } } /// /// Updates the hand definition with its velocity /// public void UpdateVelocity() { if (unityJointPoses != null) { Vector3 palmPosition = unityJointPoses[PalmIndex].Position; if (frameOn < VelocityUpdateInterval) { velocityPositionsCache[frameOn] = palmPosition; velocityPositionsSum += velocityPositionsCache[frameOn]; velocityNormalsCache[frameOn] = currentPalmNormal; velocityNormalsSum += velocityNormalsCache[frameOn]; } else { int frameIndex = frameOn % VelocityUpdateInterval; float deltaTime = Time.unscaledTime - deltaTimeStart; Vector3 newPositionsSum = velocityPositionsSum - velocityPositionsCache[frameIndex] + palmPosition; Vector3 newNormalsSum = velocityNormalsSum - velocityNormalsCache[frameIndex] + currentPalmNormal; Velocity = (newPositionsSum - velocityPositionsSum) / deltaTime / VelocityUpdateInterval; Quaternion rotation = Quaternion.FromToRotation(velocityNormalsSum / VelocityUpdateInterval, newNormalsSum / VelocityUpdateInterval); Vector3 rotationRate = rotation.eulerAngles * Mathf.Deg2Rad; AngularVelocity = rotationRate / deltaTime; velocityPositionsCache[frameIndex] = palmPosition; velocityNormalsCache[frameIndex] = currentPalmNormal; velocityPositionsSum = newPositionsSum; velocityNormalsSum = newNormalsSum; } deltaTimeStart = Time.unscaledTime; frameOn++; } } #endregion } }