// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using Microsoft.MixedReality.Toolkit.Utilities; using System; using UnityEngine; namespace Microsoft.MixedReality.Toolkit.Input { [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Hand Tracking Profile", fileName = "MixedRealityHandTrackingProfile", order = (int)CreateProfileMenuItemIndices.HandTracking)] [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/hand-tracking")] public class MixedRealityHandTrackingProfile : BaseMixedRealityProfile { [SerializeField] [Tooltip("The joint prefab to use.")] private GameObject jointPrefab = null; [SerializeField] [Tooltip("The joint prefab to use for palm.")] private GameObject palmPrefab = null; [SerializeField] [Tooltip("The joint prefab to use for the index tip (point of interaction.")] private GameObject fingertipPrefab = null; /// /// The joint prefab to use. /// public GameObject JointPrefab => jointPrefab; /// /// The joint prefab to use for palm /// public GameObject PalmJointPrefab => palmPrefab; /// /// The joint prefab to use for finger tip /// public GameObject FingerTipPrefab => fingertipPrefab; [SerializeField] [Tooltip("The hand mesh material to use for system generated hand meshes")] private Material systemHandMeshMaterial; /// /// The hand mesh material to use for system generated hand meshes /// public Material SystemHandMeshMaterial => systemHandMeshMaterial; [SerializeField] [Tooltip("The hand mesh material to use for rigged hand meshes")] private Material riggedHandMeshMaterial; /// /// The hand mesh material to use for rigged hand meshes /// public Material RiggedHandMeshMaterial => riggedHandMeshMaterial; [SerializeField] [Tooltip("If this is not null and hand system supports hand meshes, use this mesh to render hand mesh.")] private GameObject handMeshPrefab = null; /// /// The hand mesh prefab to use to render the hand /// [Obsolete("The GameObject which generates the system handmesh is now created at runtime. This prefab is not used")] public GameObject HandMeshPrefab => handMeshPrefab; /// /// The hand mesh visualization enable/disable state of the current application mode. /// /// /// If this property is called while in-editor, this will only affect the in-editor settings /// (i.e. the SupportedApplicationModes.Editor flag of HandMeshVisualizationModes). /// If this property is called while in-player, this will only affect the in-player settings /// (i.e. the SupportedApplicationModes.Player flag of HandMeshVisualizationModes). /// public bool EnableHandMeshVisualization { get => IsSupportedApplicationMode(handMeshVisualizationModes); set => handMeshVisualizationModes = UpdateSupportedApplicationMode(value, handMeshVisualizationModes); } /// /// The hand joint visualization enable/disable state of the current application mode. /// /// /// If this property is called while in-editor, this will only affect the in-editor settings /// (i.e. the SupportedApplicationModes.Editor flag of HandJointVisualizationModes). /// If this property is called while in-player, this will only affect the in-player settings /// (i.e. the SupportedApplicationModes.Player flag of HandJointVisualizationModes). /// public bool EnableHandJointVisualization { get => IsSupportedApplicationMode(handJointVisualizationModes); set => handJointVisualizationModes = UpdateSupportedApplicationMode(value, handJointVisualizationModes); } [EnumFlags] [SerializeField] [Tooltip("The application modes in which hand mesh visualizations will display. " + "Will only show if the system provides hand mesh data. Note: this could reduce performance")] private SupportedApplicationModes handMeshVisualizationModes = 0; public SupportedApplicationModes HandMeshVisualizationModes { get => handMeshVisualizationModes; set => handMeshVisualizationModes = value; } [EnumFlags] [SerializeField] [Tooltip("The application modes in which hand joint visualizations will display. " + "Will only show if the system provides joint data. Note: this could reduce performance")] private SupportedApplicationModes handJointVisualizationModes = 0; public SupportedApplicationModes HandJointVisualizationModes { get => handJointVisualizationModes; set => handJointVisualizationModes = value; } /// /// Returns true if the modes specified by the specified SupportedApplicationModes matches /// the current mode that the code is running in. /// /// /// For example, if the code is currently running in editor mode (for testing in-editor /// simulation), this would return true if modes contained the SupportedApplicationModes.Editor /// bit. /// private static bool IsSupportedApplicationMode(SupportedApplicationModes modes) { #if UNITY_EDITOR return (modes & SupportedApplicationModes.Editor) != 0; #else // !UNITY_EDITOR return (modes & SupportedApplicationModes.Player) != 0; #endif } /// /// Updates the given SupportedApplicationModes by setting the bit associated with the /// currently active application mode. /// /// /// For example, if the code is currently running in editor mode (for testing in-editor /// simulation), and modes is currently SupportedApplicationModes.Player | SupportedApplicationModes.Editor /// and enabled is 'false', this would return SupportedApplicationModes.Player. /// private static SupportedApplicationModes UpdateSupportedApplicationMode(bool enabled, SupportedApplicationModes modes) { #if UNITY_EDITOR var bitValue = enabled ? SupportedApplicationModes.Editor : 0; return (modes & ~SupportedApplicationModes.Editor) | bitValue; #else // !UNITY_EDITOR var bitValue = enabled ? SupportedApplicationModes.Player : 0; return (modes & ~SupportedApplicationModes.Player) | bitValue; #endif } } }