diff --git a/com.microsoft.mixedreality.toolkit.foundation/CHANGELOG.md b/com.microsoft.mixedreality.toolkit.foundation/CHANGELOG.md
new file mode 100644
index 0000000..f412bc8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/CHANGELOG.md
@@ -0,0 +1,21 @@
+# Changelog
+
+## 2.5.1
+
+[MRTK 2.5.1 changes](https://github.com/microsoft/MixedRealityToolkit-Unity/milestone/15?closed=1)
+
+## 2.5.0
+
+[MRTK 2.5.0 changes](https://github.com/microsoft/MixedRealityToolkit-Unity/milestone/12?closed=1)
+
+## 2.4.0
+
+[MRTK 2.4.0 changes](https://github.com/microsoft/MixedRealityToolkit-Unity/milestone/11?closed=1)
+
+## 2.3.0
+
+[MRTK 2.3.0 changes](https://github.com/microsoft/MixedRealityToolkit-Unity/milestone/10?closed=1)
+
+## 2.2.0
+
+[MRTK 2.2.0 changes](https://github.com/microsoft/MixedRealityToolkit-Unity/milestone/9?closed=1)
diff --git a/com.microsoft.mixedreality.toolkit.foundation/CHANGELOG.md.meta b/com.microsoft.mixedreality.toolkit.foundation/CHANGELOG.md.meta
new file mode 100644
index 0000000..915d405
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/CHANGELOG.md.meta
@@ -0,0 +1,7 @@
+fileFormatVersion: 2
+guid: 4d3d415154baf7f429edd9dc550990e6
+TextScriptImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core.meta b/com.microsoft.mixedreality.toolkit.foundation/Core.meta
new file mode 100644
index 0000000..4815025
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 4d9628b89018421fa5adeaabed593632
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/AssemblyInfo.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/AssemblyInfo.cs
new file mode 100644
index 0000000..9a27b11
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/AssemblyInfo.cs
@@ -0,0 +1,12 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System.Runtime.CompilerServices;
+
+[assembly: InternalsVisibleTo("Microsoft.MixedReality.Toolkit.Tests.EditModeTests")]
+[assembly: InternalsVisibleTo("Microsoft.MixedReality.Toolkit.Tests.PlayModeTests")]
+[assembly: System.Reflection.AssemblyVersion("2.8.3.0")]
+[assembly: System.Reflection.AssemblyFileVersion("2.8.3.0")]
+
+[assembly: System.Reflection.AssemblyProduct("Microsoft® Mixed Reality Toolkit")]
+[assembly: System.Reflection.AssemblyCopyright("Copyright © Microsoft Corporation")]
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/AssemblyInfo.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/AssemblyInfo.cs.meta
new file mode 100644
index 0000000..192ef2b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/AssemblyInfo.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 26cee117733e84e409a3020951c8cfb2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes.meta
new file mode 100644
index 0000000..2c09443
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 3976310d79db4c879d874d7c9e89e0fb
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/DocLinkAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/DocLinkAttribute.cs
new file mode 100644
index 0000000..df27059
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/DocLinkAttribute.cs
@@ -0,0 +1,20 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines a documentation link for a service.
+ /// Used primarily by service inspector facades.
+ ///
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false)]
+ [Obsolete("Use HelpURLAttribute from Unity instead")]
+ public class DocLinkAttribute : Attribute
+ {
+ public DocLinkAttribute(string url) { URL = url; }
+
+ public string URL { get; private set; }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/DocLinkAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/DocLinkAttribute.cs.meta
new file mode 100644
index 0000000..deaab76
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/DocLinkAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d68183ede17f6b74ca38173e4b40aff3
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/EnumFlagsAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/EnumFlagsAttribute.cs
new file mode 100644
index 0000000..332beb8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/EnumFlagsAttribute.cs
@@ -0,0 +1,21 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// An attribute that allows a particular field to be rendered as multi-selectable
+ /// set of flags.
+ ///
+ ///
+ /// From https://answers.unity.com/questions/486694/default-editor-enum-as-flags-.html
+ ///
+ [AttributeUsage(AttributeTargets.Field)]
+ public sealed class EnumFlagsAttribute : PropertyAttribute
+ {
+ public EnumFlagsAttribute() { }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/EnumFlagsAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/EnumFlagsAttribute.cs.meta
new file mode 100644
index 0000000..fc440bb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/EnumFlagsAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8406c3890cd44f8bb71fdf288e018fd2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExperimentalAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExperimentalAttribute.cs
new file mode 100644
index 0000000..f35ef07
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExperimentalAttribute.cs
@@ -0,0 +1,36 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// A PropertyAttribute for showing a warning box that the tagged implementation is experimental.
+ ///
+ [AttributeUsage(AttributeTargets.Field, Inherited = true)]
+ public class ExperimentalAttribute : PropertyAttribute
+ {
+ ///
+ /// The text to display in the warning box.
+ ///
+ public string Text;
+
+ private const string defaultText = "This is an experimental feature.\n" +
+ "Parts of the MRTK appear to have a lot of value even if the details " +
+ "haven’t fully been fleshed out. For these types of features, we want " +
+ "the community to see them and get value out of them early. Because " +
+ "they are early in the cycle, we label them as experimental to indicate " +
+ "that they are still evolving, and subject to change over time.";
+
+ ///
+ /// Constructor.
+ ///
+ /// The experimental text to display in the warning box.
+ public ExperimentalAttribute(string experimentalText = defaultText)
+ {
+ Text = experimentalText;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExperimentalAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExperimentalAttribute.cs.meta
new file mode 100644
index 0000000..3f91066
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExperimentalAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 21560d9bd9bdf4c41a9ad9c366b81b1c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExtendsAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExtendsAttribute.cs
new file mode 100644
index 0000000..c2b8554
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExtendsAttribute.cs
@@ -0,0 +1,41 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using Microsoft.MixedReality.Toolkit;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Constraint that allows selection of classes that extend a specific class when
+ /// selecting a with the Unity inspector.
+ ///
+ public sealed class ExtendsAttribute : SystemTypeAttribute
+ {
+ ///
+ /// Gets the type of class that selectable classes must derive from.
+ ///
+ public Type BaseType { get; private set; }
+
+ ///
+ /// Initializes a new instance of the class.
+ ///
+ /// Type of class that selectable classes must derive from.
+ /// Gets or sets grouping of selectable classes. Defaults to unless explicitly specified.
+ public ExtendsAttribute(Type baseType, TypeGrouping grouping) : base(baseType, grouping)
+ {
+ BaseType = baseType;
+ }
+
+ ///
+ public override bool IsConstraintSatisfied(Type type)
+ {
+ return base.IsConstraintSatisfied(type) &&
+ BaseType.IsAssignableFrom(type) &&
+ type != BaseType;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExtendsAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExtendsAttribute.cs.meta
new file mode 100644
index 0000000..6048085
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ExtendsAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e78caa572e154a2b8532d8cb9f0d339c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/HelpAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/HelpAttribute.cs
new file mode 100644
index 0000000..03a6935
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/HelpAttribute.cs
@@ -0,0 +1,46 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// A PropertyAttribute for showing a collapsible Help section.
+ ///
+ [AttributeUsage(AttributeTargets.Field | AttributeTargets.Property, AllowMultiple = false)]
+ public class HelpAttribute : PropertyAttribute
+ {
+ ///
+ /// The help text
+ ///
+ public string Text;
+
+ ///
+ /// The help header foldout text
+ ///
+ ///
+ /// If Collapsible is false, then this header text will not be shown.
+ ///
+ public string Header;
+
+ ///
+ /// If true, this will be a collapsible help section. Defaults to true.
+ ///
+ public bool Collapsible;
+
+ ///
+ /// Constructor
+ ///
+ /// The help text to display
+ /// The help header foldout text
+ /// If true, this help drawer will be collapsible
+ public HelpAttribute(string helpText, string helpHeader = "Help", bool collapsible = true)
+ {
+ Text = helpText;
+ Header = helpHeader;
+ Collapsible = collapsible;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/HelpAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/HelpAttribute.cs.meta
new file mode 100644
index 0000000..afe64fc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/HelpAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2f5319af93ddc2143b74d7d7f0b08830
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ImplementsAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ImplementsAttribute.cs
new file mode 100644
index 0000000..a195b80
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ImplementsAttribute.cs
@@ -0,0 +1,51 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using Microsoft.MixedReality.Toolkit;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Constraint that allows selection of classes that implement a specific interface
+ /// when selecting a with the Unity inspector.
+ ///
+ public sealed class ImplementsAttribute : SystemTypeAttribute
+ {
+ ///
+ /// Gets the type of interface that selectable classes must implement.
+ ///
+ public Type InterfaceType { get; private set; }
+
+ ///
+ /// Initializes a new instance of the class.
+ ///
+ /// Type of interface that selectable classes must implement.
+ /// Gets or sets grouping of selectable classes. Defaults to unless explicitly specified.
+ public ImplementsAttribute(Type interfaceType, TypeGrouping grouping) : base(interfaceType, grouping)
+ {
+ InterfaceType = interfaceType;
+ }
+
+ ///
+ public override bool IsConstraintSatisfied(Type type)
+ {
+ if (base.IsConstraintSatisfied(type))
+ {
+ var interfaces = type.GetInterfaces();
+ for (var i = 0; i < interfaces.Length; i++)
+ {
+ if (interfaces[i] == InterfaceType)
+ {
+ return true;
+ }
+ }
+ }
+
+ return false;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ImplementsAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ImplementsAttribute.cs.meta
new file mode 100644
index 0000000..cc0f944
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ImplementsAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8fcf9f0ee6ff42d98ad11a39aa40693f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityControllerAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityControllerAttribute.cs
new file mode 100644
index 0000000..2973e83
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityControllerAttribute.cs
@@ -0,0 +1,71 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Linq;
+
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using System.Reflection;
+using Microsoft.MixedReality.Toolkit;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Attach to a controller device class to make it show up in the controller mapping profile.
+ ///
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false)]
+ public class MixedRealityControllerAttribute : Attribute
+ {
+ ///
+ /// The SupportedControllerType to which the controller device belongs to.
+ ///
+ public SupportedControllerType SupportedControllerType { get; }
+
+ ///
+ /// List of handedness values supported by the respective controller.
+ ///
+ public Handedness[] SupportedHandedness { get; }
+
+ ///
+ /// Path to image file used when displaying an icon in the UI.
+ ///
+ public string TexturePath { get; }
+
+ ///
+ /// Additional flags for configuring controller capabilities.
+ ///
+ public MixedRealityControllerConfigurationFlags Flags { get; }
+
+ ///
+ /// The supported Unity XR pipelines for this controller.
+ ///
+ public SupportedUnityXRPipelines SupportedUnityXRPipelines { get; }
+
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityControllerAttribute(
+ SupportedControllerType supportedControllerType,
+ Handedness[] supportedHandedness,
+ string texturePath = "",
+ MixedRealityControllerConfigurationFlags flags = 0,
+ SupportedUnityXRPipelines supportedUnityXRPipelines = (SupportedUnityXRPipelines)(-1))
+ {
+ SupportedControllerType = supportedControllerType;
+ SupportedHandedness = supportedHandedness;
+ TexturePath = texturePath;
+ Flags = flags;
+ SupportedUnityXRPipelines = supportedUnityXRPipelines;
+ }
+
+ ///
+ /// Convenience function for retrieving the attribute given a certain class type.
+ ///
+ public static MixedRealityControllerAttribute Find(Type type)
+ {
+ return type.GetCustomAttributes(typeof(MixedRealityControllerAttribute), true).FirstOrDefault() as MixedRealityControllerAttribute;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityControllerAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityControllerAttribute.cs.meta
new file mode 100644
index 0000000..976511b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityControllerAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 68bd9743c3da6de4fb0640d1b9db8f35
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityDataProviderAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityDataProviderAttribute.cs
new file mode 100644
index 0000000..f185727
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityDataProviderAttribute.cs
@@ -0,0 +1,39 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute that defines the properties of a Mixed Reality Toolkit data provider.
+ ///
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)]
+ public class MixedRealityDataProviderAttribute : MixedRealityExtensionServiceAttribute
+ {
+ ///
+ /// The interface type of the IMixedRealityService for which the data provider is supported.
+ ///
+ public Type ServiceInterfaceType { get; }
+
+ ///
+ /// The supported Unity XR pipelines for this data provider.
+ ///
+ public SupportedUnityXRPipelines SupportedUnityXRPipelines { get; }
+
+ public MixedRealityDataProviderAttribute(
+ Type serviceInterfaceType,
+ SupportedPlatforms runtimePlatforms,
+ string name = "",
+ string profilePath = "",
+ string packageFolder = "MixedRealityToolkit",
+ bool requiresProfile = false,
+ SupportedUnityXRPipelines supportedUnityXRPipelines = (SupportedUnityXRPipelines)(-1))
+ : base(runtimePlatforms, name, profilePath, packageFolder, requiresProfile)
+ {
+ ServiceInterfaceType = serviceInterfaceType;
+ SupportedUnityXRPipelines = supportedUnityXRPipelines;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityDataProviderAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityDataProviderAttribute.cs.meta
new file mode 100644
index 0000000..a464285
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityDataProviderAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f6afe3f00c06b8242a915125257240fc
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityExtensionServiceAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityExtensionServiceAttribute.cs
new file mode 100644
index 0000000..ef86fe3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityExtensionServiceAttribute.cs
@@ -0,0 +1,109 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Linq;
+using UnityEngine;
+
+#if UNITY_EDITOR
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using UnityEditor;
+#endif
+
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using Microsoft.MixedReality.Toolkit;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute that defines the properties of a Mixed Reality Toolkit extension service.
+ ///
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)]
+ public class MixedRealityExtensionServiceAttribute : Attribute
+ {
+ ///
+ /// The friendly name for this service.
+ ///
+ public virtual string Name { get; }
+
+ ///
+ /// The runtime platform(s) to run this service.
+ ///
+ public virtual SupportedPlatforms RuntimePlatforms { get; }
+
+ ///
+ /// Is a profile explicitly required?
+ ///
+ public virtual bool RequiresProfile { get; }
+
+ ///
+ /// The file path to the default profile asset relative to the package folder.
+ ///
+ public virtual string DefaultProfilePath { get; }
+
+ ///
+ /// The package where the default profile asset resides.
+ ///
+ public virtual string PackageFolder { get; }
+
+ ///
+ /// The default profile.
+ ///
+ public virtual BaseMixedRealityProfile DefaultProfile
+ {
+ get
+ {
+#if UNITY_EDITOR
+ MixedRealityToolkitModuleType moduleType = MixedRealityToolkitFiles.GetModuleFromPackageFolder(PackageFolder);
+
+ if (moduleType != MixedRealityToolkitModuleType.None)
+ {
+ string folder = MixedRealityToolkitFiles.MapModulePath(moduleType);
+ if (!string.IsNullOrWhiteSpace(folder))
+ {
+ return AssetDatabase.LoadAssetAtPath(System.IO.Path.Combine(folder, DefaultProfilePath));
+ }
+ }
+ else if (EditorProjectUtilities.FindRelativeDirectory(PackageFolder, out string folder))
+ {
+ return AssetDatabase.LoadAssetAtPath(System.IO.Path.Combine(folder, DefaultProfilePath));
+ }
+
+ // If we get here, there was an issue finding the profile.
+ Debug.LogError("Unable to find or load the profile.");
+#endif
+ return null;
+ }
+ }
+
+ ///
+ /// Constructor
+ ///
+ /// The platforms on which the extension service is supported.
+ /// The relative path to the default profile asset.
+ /// The package folder to which the path is relative.
+ public MixedRealityExtensionServiceAttribute(
+ SupportedPlatforms runtimePlatforms,
+ string name = "",
+ string defaultProfilePath = "",
+ string packageFolder = "MixedRealityToolkit",
+ bool requiresProfile = false)
+ {
+ RuntimePlatforms = runtimePlatforms;
+ Name = name;
+ DefaultProfilePath = defaultProfilePath;
+ PackageFolder = packageFolder;
+ RequiresProfile = requiresProfile;
+ }
+
+ ///
+ /// Convenience function for retrieving the attribute given a certain class type.
+ ///
+ public static MixedRealityExtensionServiceAttribute Find(Type type)
+ {
+ return type.GetCustomAttributes(typeof(MixedRealityExtensionServiceAttribute), true).FirstOrDefault() as MixedRealityExtensionServiceAttribute;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityExtensionServiceAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityExtensionServiceAttribute.cs.meta
new file mode 100644
index 0000000..eedbf67
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityExtensionServiceAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9dc03bea6acf89949a69dd79afc411c8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceInspectorAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceInspectorAttribute.cs
new file mode 100644
index 0000000..c0c96cb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceInspectorAttribute.cs
@@ -0,0 +1,24 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attach to a class implementing IMixedRealityServiceInspector to generate a facade inspector.
+ ///
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)]
+ public class MixedRealityServiceInspectorAttribute : Attribute
+ {
+ public MixedRealityServiceInspectorAttribute(Type serviceType)
+ {
+ if (!typeof(IMixedRealityService).IsAssignableFrom(serviceType))
+ throw new Exception("Can't use this attribute with type " + serviceType + " - service must implement " + typeof(IMixedRealityService) + " interface.");
+
+ ServiceType = serviceType;
+ }
+
+ public Type ServiceType { get; private set; }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceInspectorAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceInspectorAttribute.cs.meta
new file mode 100644
index 0000000..da281be
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceInspectorAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e159fc0613d5db9479be3c2d92f449b9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceProfileAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceProfileAttribute.cs
new file mode 100644
index 0000000..c68ce3f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceProfileAttribute.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute that defines which service a profile is meant to be consumed by.
+ /// Only applies to profiles that are consumed by types implementing IMixedRealityService.
+ /// A service must implement all required types and no excluded types to be considered compatible with the profile.
+ ///
+ [AttributeUsage(AttributeTargets.Class, AllowMultiple = true, Inherited = true)]
+ public class MixedRealityServiceProfileAttribute : Attribute
+ {
+ public MixedRealityServiceProfileAttribute(Type requiredType, Type excludedType = null)
+ {
+ RequiredTypes = new Type[] { requiredType };
+ ExcludedTypes = excludedType != null ? new Type[] { excludedType } : new Type[0];
+ }
+
+ public MixedRealityServiceProfileAttribute(Type[] requiredTypes, Type[] excludedTypes = null)
+ {
+ RequiredTypes = requiredTypes;
+ ExcludedTypes = excludedTypes ?? (new Type[0]);
+ }
+
+ public Type[] RequiredTypes { get; private set; }
+ public Type[] ExcludedTypes { get; private set; }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceProfileAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceProfileAttribute.cs.meta
new file mode 100644
index 0000000..f07ca7b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/MixedRealityServiceProfileAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 87d3cb16f49f8b14397e2c89b430f774
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PhysicsLayerAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PhysicsLayerAttribute.cs
new file mode 100644
index 0000000..2702cf3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PhysicsLayerAttribute.cs
@@ -0,0 +1,17 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Physics
+{
+ ///
+ /// Attribute used to make an field render a dropdown generated from the current layers defined in the Tag Manager.
+ ///
+ [AttributeUsage(AttributeTargets.Field)]
+ public sealed class PhysicsLayerAttribute : PropertyAttribute
+ {
+ public PhysicsLayerAttribute() { }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PhysicsLayerAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PhysicsLayerAttribute.cs.meta
new file mode 100644
index 0000000..5bc971b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PhysicsLayerAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a46f813550c3448383a069c6988d64da
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PrefabAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PrefabAttribute.cs
new file mode 100644
index 0000000..3302709
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PrefabAttribute.cs
@@ -0,0 +1,14 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute used to ensure that a GameObject inspector slot only accepts prefabs.
+ ///
+ [AttributeUsage(AttributeTargets.Field)]
+ public sealed class PrefabAttribute : PropertyAttribute { }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PrefabAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PrefabAttribute.cs.meta
new file mode 100644
index 0000000..7ef74ac
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/PrefabAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8f0704a2ddbf46c7ac01b75061f8da6c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ReadOnlyAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ReadOnlyAttribute.cs
new file mode 100644
index 0000000..f6fd8d0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ReadOnlyAttribute.cs
@@ -0,0 +1,11 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ public class ReadOnlyAttribute : PropertyAttribute { }
+ public class BeginReadOnlyGroupAttribute : PropertyAttribute { }
+ public class EndReadOnlyGroupAttribute : PropertyAttribute { }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ReadOnlyAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ReadOnlyAttribute.cs.meta
new file mode 100644
index 0000000..dc613aa
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ReadOnlyAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 700a45d7f9cad034aa2c70acbe010829
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ScenePickAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ScenePickAttribute.cs
new file mode 100644
index 0000000..22ac278
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ScenePickAttribute.cs
@@ -0,0 +1,19 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute to mark up an int field to be drawn using the
+ /// ScenePickPropertyDrawer
+ /// This allows the UI to display a dropdown instead of a
+ /// numeric entry field.
+ ///
+ [AttributeUsage(AttributeTargets.Field, AllowMultiple = false)]
+ public class ScenePickAttribute : PropertyAttribute
+ {
+ // Nothing to see Here, This only acts as a marker to help the editor.
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ScenePickAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ScenePickAttribute.cs.meta
new file mode 100644
index 0000000..ada1d32
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/ScenePickAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: aada00f885183904f84f3e17ea5336a5
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SpeechKeywordAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SpeechKeywordAttribute.cs
new file mode 100644
index 0000000..ae13bda
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SpeechKeywordAttribute.cs
@@ -0,0 +1,14 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute used to display a dropdown of registered keywords from the speech profile.
+ ///
+ [AttributeUsage(AttributeTargets.Field)]
+ public sealed class SpeechKeywordAttribute : PropertyAttribute { }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SpeechKeywordAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SpeechKeywordAttribute.cs.meta
new file mode 100644
index 0000000..eee412d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SpeechKeywordAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f47425f809ae9f945ae63e6616771ba0
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SystemTypeAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SystemTypeAttribute.cs
new file mode 100644
index 0000000..7f9c9f8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SystemTypeAttribute.cs
@@ -0,0 +1,66 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using Microsoft.MixedReality.Toolkit;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Base class for class selection constraints that can be applied when selecting
+ /// a with the Unity inspector.
+ ///
+ public abstract class SystemTypeAttribute : PropertyAttribute
+ {
+ ///
+ /// Gets or sets grouping of selectable classes. Defaults to unless explicitly specified.
+ ///
+ public TypeGrouping Grouping { get; protected set; }
+
+ ///
+ /// Gets or sets whether abstract classes can be selected from drop-down.
+ /// Defaults to a value of false unless explicitly specified.
+ ///
+ public bool AllowAbstract { get; protected set; } = false;
+
+ ///
+ /// Constructor.
+ ///
+ /// Initializes a new instance of the class.
+ /// Gets or sets grouping of selectable classes. Defaults to unless explicitly specified.
+ protected SystemTypeAttribute(Type type, TypeGrouping grouping = TypeGrouping.ByNamespaceFlat)
+ {
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+ bool isValid = type.IsClass() || type.IsInterface() || type.IsValueType() && !type.IsEnum();
+#else
+ bool isValid = type.IsClass || type.IsInterface || type.IsValueType && !type.IsEnum;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+ if (!isValid)
+ {
+ Debug.Assert(isValid, $"Invalid Type {type} in attribute.");
+ }
+ Grouping = grouping;
+ }
+
+ ///
+ /// Determines whether the specified satisfies filter constraint.
+ ///
+ /// Type to test.
+ ///
+ /// A value indicating if the type specified by
+ /// satisfies this constraint and should thus be selectable.
+ ///
+ public virtual bool IsConstraintSatisfied(Type type)
+ {
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+ return AllowAbstract || !type.IsAbstract();
+#else
+ return AllowAbstract || !type.IsAbstract;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SystemTypeAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SystemTypeAttribute.cs.meta
new file mode 100644
index 0000000..6ab4e34
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/SystemTypeAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d78dba64176b4c168d5a5f3ff8865b97
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/TagPropertyAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/TagPropertyAttribute.cs
new file mode 100644
index 0000000..b3e7a13
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/TagPropertyAttribute.cs
@@ -0,0 +1,17 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// A PropertyAttribute for Unity tags (a string field).
+ ///
+ [AttributeUsage(AttributeTargets.Field, AllowMultiple = false)]
+ public class TagPropertyAttribute : PropertyAttribute
+ {
+ // Do nothing
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/TagPropertyAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/TagPropertyAttribute.cs.meta
new file mode 100644
index 0000000..6ed6b4e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/TagPropertyAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 27713c34a4c46f54da03155ceb0bdd6e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/Vector3RangeAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/Vector3RangeAttribute.cs
new file mode 100644
index 0000000..9a8cae8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/Vector3RangeAttribute.cs
@@ -0,0 +1,36 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute used to make a float or int variable in a script be restricted to a specific range.
+ ///
+ [AttributeUsage(AttributeTargets.Field)]
+ public sealed class Vector3RangeAttribute : PropertyAttribute
+ {
+ ///
+ /// Minimum value.
+ ///
+ public readonly float Min;
+
+ ///
+ /// Maximum value.
+ ///
+ public readonly float Max;
+
+ ///
+ /// Attribute used to make a float or int variable in a script be restricted to a specific range.
+ ///
+ /// The minimum allowed value.
+ /// The maximum allowed value.
+ public Vector3RangeAttribute(float min, float max)
+ {
+ Min = min;
+ Max = max;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/Vector3RangeAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/Vector3RangeAttribute.cs.meta
new file mode 100644
index 0000000..1b34f4a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Attributes/Vector3RangeAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: cbd8fa9213b04d238d65cc7be8d4b032
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions.meta
new file mode 100644
index 0000000..fb17e9e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 15ee9eb60ada4de58f7afe5d1bec0d15
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BaseMixedRealityProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BaseMixedRealityProfile.cs
new file mode 100644
index 0000000..d5ed275
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BaseMixedRealityProfile.cs
@@ -0,0 +1,22 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Base abstract class for all Mixed Reality profile configurations.
+ /// Extends ScriptableObject and used as a property container to initialize MRTK services.
+ ///
+ [Serializable]
+ public abstract class BaseMixedRealityProfile : ScriptableObject
+ {
+ [SerializeField]
+ [HideInInspector]
+ private bool isCustomProfile = true;
+
+ internal bool IsCustomProfile => isCustomProfile;
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BaseMixedRealityProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BaseMixedRealityProfile.cs.meta
new file mode 100644
index 0000000..730dd4c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BaseMixedRealityProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7bf26942d2da73b49a9ae41a021e6ea6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem.meta
new file mode 100644
index 0000000..1a5e0df
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 7316ed9148a54d82a67abfb3b08eea45
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/BoundaryType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/BoundaryType.cs
new file mode 100644
index 0000000..8f9b599
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/BoundaryType.cs
@@ -0,0 +1,20 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Boundary
+{
+ ///
+ /// Defines different types of boundaries that can be requested.
+ ///
+ public enum BoundaryType
+ {
+ ///
+ /// A rectangular area calculated as the largest rectangle within the tracked area, good for placing content near the user.
+ ///
+ PlayArea,
+ ///
+ /// The full tracked boundary, typically manually drawn by a user while setting up their device.
+ ///
+ TrackedArea
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/BoundaryType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/BoundaryType.cs.meta
new file mode 100644
index 0000000..25dcd60
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/BoundaryType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 14f00db1440205648911737e3720c79f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/Edge.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/Edge.cs
new file mode 100644
index 0000000..685ea8d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/Edge.cs
@@ -0,0 +1,45 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Boundary
+{
+ ///
+ /// The Edge structure defines the points of a line segment that are used to
+ /// construct a polygonal boundary.
+ ///
+ public struct Edge
+ {
+ ///
+ /// The first point of the edge line segment.
+ ///
+ public readonly Vector2 PointA;
+
+ ///
+ /// The second point of the edge line segment.
+ ///
+ public readonly Vector2 PointB;
+
+ ///
+ /// Initializes the Edge structure.
+ ///
+ /// The first point of the line segment.
+ /// The second point of the line segment.
+ public Edge(Vector2 pointA, Vector2 pointB)
+ {
+ PointA = pointA;
+ PointB = pointB;
+ }
+
+ ///
+ /// Initializes the Edge structure.
+ ///
+ /// The first point of the line segment.
+ /// The second point of the line segment.
+ public Edge(Vector3 pointA, Vector3 pointB) :
+ // Use the X and Z parameters as our edges are height agnostic.
+ this(new Vector2(pointA.x, pointA.z), new Vector2(pointB.x, pointB.z))
+ { }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/Edge.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/Edge.cs.meta
new file mode 100644
index 0000000..e5e0d09
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/Edge.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6181034d7a2744628b5f013c1a37ee51
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/InscribedRectangle.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/InscribedRectangle.cs
new file mode 100644
index 0000000..88d7c6d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/InscribedRectangle.cs
@@ -0,0 +1,563 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Boundary
+{
+ ///
+ /// The InscribedRectangle class defines the largest rectangle within an
+ /// arbitrary shape.
+ ///
+ public class InscribedRectangle
+ {
+ ///
+ /// Total number of starting points randomly generated within the boundary.
+ ///
+ private const int randomPointCount = 30;
+
+ ///
+ /// The total amount of height, in meters, we want to gain with each binary search
+ /// change before we decide that it's good enough.
+ ///
+ private const float minimumHeightGain = 0.01f;
+
+ ///
+ /// Angles to use for fitting the rectangle within the boundary.
+ ///
+ private static readonly float[] FitAngles = { 0, 15, 30, 45, 60, 75, 90, 105, 120, 135, 150, 165 };
+
+ ///
+ /// Aspect ratios used when fitting rectangles within the boundary.
+ ///
+ private static readonly float[] AspectRatios = {
+ 1.0f, 1.5f, 2.0f, 2.5f, 3.0f, 3.5f, 4.0f, 4.5f,
+ 5.0f, 5.5f, 6, 6.5f, 7, 7.5f, 8.0f, 8.5f, 9.0f,
+ 9.5f, 10.0f, 10.5f, 11.0f, 11.5f, 12.0f, 12.5f,
+ 13.0f, 13.5f, 14.0f, 14.5f, 15.0f};
+
+ ///
+ /// The center point of the inscribed rectangle.
+ ///
+ public Vector2 Center { get; private set; } = EdgeUtilities.InvalidPoint;
+
+ ///
+ /// The width of the inscribed rectangle.
+ ///
+ public float Width { get; private set; } = 0f;
+
+ ///
+ /// The height of the inscribed rectangle.
+ ///
+ public float Height { get; private set; } = 0f;
+
+ ///
+ /// The rotation angle, in degrees, of the inscribed rectangle.
+ ///
+ public float Angle { get; private set; } = 0f;
+
+ ///
+ /// Is the described rectangle valid?
+ ///
+ ///
+ /// A rectangle is considered valid if its center point is valid.
+ ///
+ public bool IsValid => EdgeUtilities.IsValidPoint(Center);
+
+ ///
+ /// Finds a large inscribed rectangle. Tries to be maximal but this is
+ /// best effort. The algorithm used was inspired by the blog post
+ /// https://d3plus.org/blog/behind-the-scenes/2014/07/08/largest-rect/
+ /// Random points within the polygon are chosen, and then 2 lines are
+ /// drawn through those points. The midpoints of those lines are
+ /// used as the center of various rectangles, using a binary search to
+ /// vary the size, until the largest fit-able rectangle is found.
+ /// This is then repeated for predefined angles (0-180 in steps of 15)
+ /// and aspect ratios (1 to 15 in steps of 0.5).
+ ///
+ /// The boundary geometry.
+ /// Random number generator seed.
+ ///
+ /// For the most reproducible results, use the same randomSeed value each time this method is called.
+ ///
+ public InscribedRectangle(Edge[] geometryEdges, int randomSeed)
+ {
+ if (geometryEdges == null || geometryEdges.Length == 0)
+ {
+ Debug.LogError("InscribedRectangle requires an array of Edges. You passed in a null or empty array.");
+ return;
+ }
+
+ // Clear previous rectangle
+ Center = EdgeUtilities.InvalidPoint;
+ Width = 0;
+ Height = 0;
+ Angle = 0;
+
+ float minX = EdgeUtilities.maxWidth;
+ float minY = EdgeUtilities.maxWidth;
+ float maxX = -EdgeUtilities.maxWidth;
+ float maxY = -EdgeUtilities.maxWidth;
+
+ // Find min x, min y, max x, max y
+ for (int i = 0; i < geometryEdges.Length; i++)
+ {
+ Edge edge = geometryEdges[i];
+
+ if ((edge.PointA.x < minX) || (edge.PointB.x < minX))
+ {
+ minX = Mathf.Min(edge.PointA.x, edge.PointB.x);
+ }
+
+ if ((edge.PointA.y < minY) || (edge.PointB.y < minY))
+ {
+ minY = Mathf.Min(edge.PointA.y, edge.PointB.y);
+ }
+
+ if ((edge.PointA.x > maxX) || (edge.PointB.x > maxX))
+ {
+ maxX = Mathf.Max(edge.PointA.x, edge.PointB.x);
+ }
+
+ if ((edge.PointA.y > maxY) || (edge.PointB.y > maxY))
+ {
+ maxY = Mathf.Max(edge.PointA.y, edge.PointB.y);
+ }
+
+ }
+
+ // Generate random points until we have randomPointCount starting points
+ Vector2[] startingPoints = new Vector2[randomPointCount];
+ {
+ System.Random random = new System.Random(randomSeed);
+ for (int i = 0; i < startingPoints.Length; i++)
+ {
+ Vector2 candidatePoint;
+
+ do
+ {
+ candidatePoint.x = ((float)random.NextDouble() * (maxX - minX)) + minX;
+ candidatePoint.y = ((float)random.NextDouble() * (maxY - minY)) + minY;
+ }
+ while (!EdgeUtilities.IsInsideBoundary(geometryEdges, candidatePoint));
+
+ startingPoints[i] = candidatePoint;
+ }
+ }
+
+ for (int angleIndex = 0; angleIndex < FitAngles.Length; angleIndex++)
+ {
+ for (int pointIndex = 0; pointIndex < startingPoints.Length; pointIndex++)
+ {
+ Vector2 topCollisionPoint;
+ Vector2 bottomCollisionPoint;
+ Vector2 leftCollisionPoint;
+ Vector2 rightCollisionPoint;
+
+ float angleRadians = MathUtilities.DegreesToRadians(FitAngles[angleIndex]);
+
+ // Find the collision point of a cross through the given point at the given angle.
+ // Note, we are ignoring the return value as we are checking each point's validity
+ // individually.
+ FindSurroundingCollisionPoints(
+ geometryEdges,
+ startingPoints[pointIndex],
+ angleRadians,
+ out topCollisionPoint,
+ out bottomCollisionPoint,
+ out leftCollisionPoint,
+ out rightCollisionPoint);
+
+ float newWidth;
+ float newHeight;
+
+ if (EdgeUtilities.IsValidPoint(topCollisionPoint) && EdgeUtilities.IsValidPoint(bottomCollisionPoint))
+ {
+ float aX = topCollisionPoint.x;
+ float aY = topCollisionPoint.y;
+ float bX = bottomCollisionPoint.x;
+ float bY = bottomCollisionPoint.y;
+
+ // Calculate the midpoint between the top and bottom collision points.
+ Vector2 verticalMidpoint = new Vector2((aX + bX) * 0.5f, (aY + bY) * 0.5f);
+ if (TryFixMaximumRectangle(
+ geometryEdges,
+ verticalMidpoint,
+ angleRadians,
+ Width * Height,
+ out newWidth,
+ out newHeight))
+ {
+ Center = verticalMidpoint;
+ Angle = FitAngles[angleIndex];
+ Width = newWidth;
+ Height = newHeight;
+ }
+ }
+
+ if (EdgeUtilities.IsValidPoint(leftCollisionPoint) && EdgeUtilities.IsValidPoint(rightCollisionPoint))
+ {
+ float aX = leftCollisionPoint.x;
+ float aY = leftCollisionPoint.y;
+ float bX = rightCollisionPoint.x;
+ float bY = rightCollisionPoint.y;
+
+ // Calculate the midpoint between the left and right collision points.
+ Vector2 horizontalMidpoint = new Vector2((aX + bX) * 0.5f, (aY + bY) * 0.5f);
+ if (TryFixMaximumRectangle(
+ geometryEdges,
+ horizontalMidpoint,
+ angleRadians,
+ Width * Height,
+ out newWidth,
+ out newHeight))
+ {
+ Center = horizontalMidpoint;
+ Angle = FitAngles[angleIndex];
+ Width = newWidth;
+ Height = newHeight;
+ }
+ }
+ }
+ }
+ }
+
+ ///
+ /// Find points at which there are collisions with the geometry around a given point.
+ ///
+ /// The boundary geometry.
+ /// The point around which collisions will be identified.
+ /// The angle, in radians, at which the collision points will be oriented.
+ /// Receives the coordinates of the upper collision point.
+ /// Receives the coordinates of the lower collision point.
+ /// Receives the coordinates of the left collision point.
+ /// Receives the coordinates of the right collision point.
+ ///
+ /// True if all of the required collision points are located, false otherwise.
+ /// If a point is unable to be found, the appropriate out parameter will be set to .
+ ///
+ private bool FindSurroundingCollisionPoints(
+ Edge[] geometryEdges,
+ Vector2 point,
+ float angleRadians,
+ out Vector2 topCollisionPoint,
+ out Vector2 bottomCollisionPoint,
+ out Vector2 leftCollisionPoint,
+ out Vector2 rightCollisionPoint)
+ {
+ // Initialize out parameters.
+ topCollisionPoint = EdgeUtilities.InvalidPoint;
+ bottomCollisionPoint = EdgeUtilities.InvalidPoint;
+ leftCollisionPoint = EdgeUtilities.InvalidPoint;
+ rightCollisionPoint = EdgeUtilities.InvalidPoint;
+
+ // Check to see if the point is inside the geometry.
+ if (!EdgeUtilities.IsInsideBoundary(geometryEdges, point))
+ {
+ return false;
+ }
+
+ // Define values that are outside of the maximum boundary size.
+ float largeValue = EdgeUtilities.maxWidth;
+ float smallValue = -largeValue;
+
+ // Find the top and bottom collision points by creating a large line segment that goes through the point to MAX and MIN values on Y
+ Vector2 topEndpoint = new Vector2(point.x, largeValue);
+ Vector2 bottomEndpoint = new Vector2(point.x, smallValue);
+ topEndpoint = RotatePoint(topEndpoint, point, angleRadians);
+ bottomEndpoint = RotatePoint(bottomEndpoint, point, angleRadians);
+ Edge verticalLine = new Edge(topEndpoint, bottomEndpoint);
+
+ // Find the left and right collision points by creating a large line segment that goes through the point to MAX and Min values on X
+ Vector2 rightEndpoint = new Vector2(largeValue, point.y);
+ Vector2 leftEndpoint = new Vector2(smallValue, point.y);
+ rightEndpoint = RotatePoint(rightEndpoint, point, angleRadians);
+ leftEndpoint = RotatePoint(leftEndpoint, point, angleRadians);
+ Edge horizontalLine = new Edge(rightEndpoint, leftEndpoint);
+
+ for (int i = 0; i < geometryEdges.Length; i++)
+ {
+ // Look for a vertical collision
+ Vector2 verticalIntersectionPoint = EdgeUtilities.GetIntersectionPoint(geometryEdges[i], verticalLine);
+ if (EdgeUtilities.IsValidPoint(verticalIntersectionPoint))
+ {
+ // Is the intersection above or below the point?
+ if (RotatePoint(verticalIntersectionPoint, point, -angleRadians).y > point.y)
+ {
+ // Update the top collision point
+ if (!EdgeUtilities.IsValidPoint(topCollisionPoint) ||
+ (Vector2.SqrMagnitude(point - verticalIntersectionPoint) < Vector2.SqrMagnitude(point - topCollisionPoint)))
+ {
+ topCollisionPoint = verticalIntersectionPoint;
+ }
+ }
+ else
+ {
+ // Update the bottom collision point
+ if (!EdgeUtilities.IsValidPoint(bottomCollisionPoint) ||
+ (Vector2.SqrMagnitude(point - verticalIntersectionPoint) < Vector2.SqrMagnitude(point - bottomCollisionPoint)))
+ {
+ bottomCollisionPoint = verticalIntersectionPoint;
+ }
+ }
+ }
+
+ // Look for a horizontal collision
+ Vector2 horizontalIntersection = EdgeUtilities.GetIntersectionPoint(geometryEdges[i], horizontalLine);
+ if (EdgeUtilities.IsValidPoint(horizontalIntersection))
+ {
+ // Is this intersection to the left or the right of the point?
+ if (RotatePoint(horizontalIntersection, point, -angleRadians).x < point.x)
+ {
+ // Update the left collision point
+ if (!EdgeUtilities.IsValidPoint(leftCollisionPoint) ||
+ (Vector2.SqrMagnitude(point - horizontalIntersection) < Vector2.SqrMagnitude(point - leftCollisionPoint)))
+ {
+ leftCollisionPoint = horizontalIntersection;
+ }
+ }
+ else
+ {
+ // Update the right collision point
+ if (!EdgeUtilities.IsValidPoint(rightCollisionPoint) ||
+ (Vector2.SqrMagnitude(point - horizontalIntersection) < Vector2.SqrMagnitude(point - rightCollisionPoint)))
+ {
+ rightCollisionPoint = horizontalIntersection;
+ }
+ }
+ }
+ }
+
+ // Each corner of the rectangle must intersect with the geometry.
+ if (!EdgeUtilities.IsValidPoint(topCollisionPoint) ||
+ !EdgeUtilities.IsValidPoint(bottomCollisionPoint) ||
+ !EdgeUtilities.IsValidPoint(leftCollisionPoint) ||
+ !EdgeUtilities.IsValidPoint(rightCollisionPoint))
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ ///
+ /// Determine of the provided point lies within the defined rectangle.
+ ///
+ /// The point to check
+ ///
+ /// True if the point is within the rectangle's bounds, false otherwise.
+ ///
+ /// The rectangle is not valid.
+ public bool IsInsideBoundary(Vector2 point)
+ {
+ if (!IsValid)
+ {
+ throw new InvalidOperationException("A point cannot be within an invalid rectangle.");
+ }
+
+ point -= Center;
+ point = RotatePoint(point, Vector2.zero, MathUtilities.DegreesToRadians(-Angle));
+
+ bool inWidth = Mathf.Abs(point.x) <= (Width * 0.5f);
+ bool inHeight = Mathf.Abs(point.y) <= (Height * 0.5f);
+
+ return (inWidth && inHeight);
+ }
+
+ ///
+ /// Rotate a two dimensional point about another point by the specified angle.
+ ///
+ /// The point to be rotated.
+ /// The point about which the rotation is to occur.
+ /// The angle for the rotation, in radians
+ ///
+ /// The coordinates of the rotated point.
+ ///
+ private Vector2 RotatePoint(Vector2 point, Vector2 origin, float angleRadians)
+ {
+ if (angleRadians.Equals(0f))
+ {
+ return point;
+ }
+
+ Vector2 rotated = point;
+
+ // Translate to origin of rotation
+ rotated.x -= origin.x;
+ rotated.y -= origin.y;
+
+ // Rotate the point
+ float sin = Mathf.Sin(angleRadians);
+ float cos = Mathf.Cos(angleRadians);
+ float x = rotated.x * cos - rotated.y * sin;
+ float y = rotated.x * sin + rotated.y * cos;
+
+ // Translate back and return
+ rotated.x = x + origin.x;
+ rotated.y = y + origin.y;
+
+ return rotated;
+ }
+
+ ///
+ /// Check to see if a rectangle centered at the specified point and oriented at
+ /// the specified angle will fit within the geometry.
+ ///
+ /// The boundary geometry.
+ /// The center point of the rectangle.
+ /// The orientation, in radians, of the rectangle.
+ /// The width of the rectangle.
+ /// The height of the rectangle.
+ private bool CheckRectangleFit(
+ Edge[] geometryEdges,
+ Vector2 centerPoint,
+ float angleRadians,
+ float width,
+ float height)
+ {
+ float halfWidth = width * 0.5f;
+ float halfHeight = height * 0.5f;
+
+ // Calculate the rectangle corners.
+ Vector2 topLeft = new Vector2(centerPoint.x - halfWidth, centerPoint.y + halfHeight);
+ Vector2 topRight = new Vector2(centerPoint.x + halfWidth, centerPoint.y + halfHeight);
+ Vector2 bottomLeft = new Vector2(centerPoint.x - halfWidth, centerPoint.y - halfHeight);
+ Vector2 bottomRight = new Vector2(centerPoint.x + halfWidth, centerPoint.y - halfHeight);
+
+ // Rotate the rectangle.
+ topLeft = RotatePoint(topLeft, centerPoint, angleRadians);
+ topRight = RotatePoint(topRight, centerPoint, angleRadians);
+ bottomLeft = RotatePoint(bottomLeft, centerPoint, angleRadians);
+ bottomRight = RotatePoint(bottomRight, centerPoint, angleRadians);
+
+ // Get the rectangle edges.
+ Edge topEdge = new Edge(topLeft, topRight);
+ Edge rightEdge = new Edge(topRight, bottomRight);
+ Edge bottomEdge = new Edge(bottomLeft, bottomRight);
+ Edge leftEdge = new Edge(topLeft, bottomLeft);
+
+ // Check for collisions with the boundary geometry. If any of our edges collide,
+ // the rectangle will not fit within the playspace.
+ for (int i = 0; i < geometryEdges.Length; i++)
+ {
+ if (EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], topEdge)) ||
+ EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], rightEdge)) ||
+ EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], bottomEdge)) ||
+ EdgeUtilities.IsValidPoint(EdgeUtilities.GetIntersectionPoint(geometryEdges[i], leftEdge)))
+ {
+ return false;
+ }
+ }
+
+ // No collisions found with the rectangle. Success!
+ return true;
+ }
+
+ ///
+ /// Attempt to fit the largest rectangle possible within the geometry.
+ ///
+ /// The boundary geometry.
+ /// The center point for the rectangle.
+ /// The rotation, in radians, of the rectangle.
+ /// The smallest allowed area.
+ /// Returns the width of the rectangle.
+ /// Returns the height of the rectangle.
+ ///
+ /// True if a rectangle with an area greater than or equal to minArea was able to be fit
+ /// within the geometry at centerPoint.
+ ///
+ private bool TryFixMaximumRectangle(
+ Edge[] geometryEdges,
+ Vector2 centerPoint,
+ float angleRadians,
+ float minArea,
+ out float width,
+ out float height)
+ {
+ width = 0.0f;
+ height = 0.0f;
+
+ Vector2 topCollisionPoint;
+ Vector2 bottomCollisionPoint;
+ Vector2 leftCollisionPoint;
+ Vector2 rightCollisionPoint;
+
+ // Find the collision points with the geometry
+ if (!FindSurroundingCollisionPoints(geometryEdges, centerPoint, angleRadians,
+ out topCollisionPoint, out bottomCollisionPoint, out leftCollisionPoint, out rightCollisionPoint))
+ {
+ return false;
+ }
+
+ // Start by calculating max width and height by ray-casting a cross from the point at the given angle
+ // and taking the shortest leg of each ray. Width is the longest.
+ float verticalMinDistanceToEdge = Mathf.Min(
+ Vector2.Distance(centerPoint, topCollisionPoint),
+ Vector2.Distance(centerPoint, bottomCollisionPoint));
+
+ float horizontalMinDistanceToEdge = Mathf.Min(
+ Vector2.Distance(centerPoint, leftCollisionPoint),
+ Vector2.Distance(centerPoint, rightCollisionPoint));
+
+ // Width is the largest of the possible dimensions
+ float maxWidth = Math.Max(verticalMinDistanceToEdge, horizontalMinDistanceToEdge) * 2.0f;
+ float maxHeight = Math.Min(verticalMinDistanceToEdge, horizontalMinDistanceToEdge) * 2.0f;
+
+ float aspectRatio = 0.0f;
+
+ // For each aspect ratio we do a binary search to find the maximum rectangle that fits,
+ // though once we start increasing our area by minimumHeightGain we call it good enough.
+ for (int i = 0; i < AspectRatios.Length; i++)
+ {
+ // The height is limited by the width. If a height would make our width exceed maxWidth, it can't be used
+ float searchHeightUpperBound = Mathf.Max(maxHeight, maxWidth / AspectRatios[i]);
+
+ // Set to the min height that will out perform our previous area at the given aspect ratio. This is 0 the first time.
+ // Derived from biggestAreaSoFar=height*(height*aspectRatio)
+ float searchHeightLowerBound = Mathf.Sqrt(Mathf.Max((width * height), minArea) / AspectRatios[i]);
+
+ // If the lowest value needed to outperform the previous best is greater than our max,
+ // this aspect ratio can't outperform what we've already calculated.
+ if ((searchHeightLowerBound > searchHeightUpperBound) ||
+ (searchHeightLowerBound * AspectRatios[i] > maxWidth))
+ {
+ continue;
+ }
+
+ float currentTestingHeight = Mathf.Max(searchHeightLowerBound, maxHeight * 0.5f);
+
+
+ // Perform the binary search until continuing to search will not give us a significant win.
+ do
+ {
+ if (CheckRectangleFit(geometryEdges,
+ centerPoint,
+ angleRadians,
+ AspectRatios[i] * currentTestingHeight,
+ currentTestingHeight))
+ {
+ // Binary search up-ward
+ // If the rectangle will fit, increase the lower bounds of our binary search
+ searchHeightLowerBound = currentTestingHeight;
+
+ width = currentTestingHeight * AspectRatios[i];
+ height = currentTestingHeight;
+ aspectRatio = AspectRatios[i];
+ currentTestingHeight = (searchHeightUpperBound + currentTestingHeight) * 0.5f;
+ }
+ else
+ {
+ // If the rectangle won't fit, update our upper bound and lower our binary search
+ searchHeightUpperBound = currentTestingHeight;
+ currentTestingHeight = (currentTestingHeight + searchHeightLowerBound) * 0.5f;
+ }
+ }
+ while ((searchHeightUpperBound - searchHeightLowerBound) > minimumHeightGain);
+ }
+
+ return (aspectRatio > 0.0f);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/InscribedRectangle.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/InscribedRectangle.cs.meta
new file mode 100644
index 0000000..dbd0270
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/InscribedRectangle.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 3ec7fd504604466b90bb50609ed1b35f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/MixedRealityBoundaryVisualizationProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/MixedRealityBoundaryVisualizationProfile.cs
new file mode 100644
index 0000000..ea60a21
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/MixedRealityBoundaryVisualizationProfile.cs
@@ -0,0 +1,201 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Physics;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Boundary
+{
+ ///
+ /// Configuration profile settings for setting up boundary visualizations.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Boundary Visualization Profile", fileName = "MixedRealityBoundaryVisualizationProfile", order = (int)CreateProfileMenuItemIndices.BoundaryVisualization)]
+ [MixedRealityServiceProfile(typeof(IMixedRealityBoundarySystem))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/boundary/boundary-system-getting-started")]
+ public class MixedRealityBoundaryVisualizationProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("The approximate height of the play space, in meters.")]
+ private float boundaryHeight = 3.0f;
+
+ ///
+ /// The developer defined height of the boundary, in meters.
+ ///
+ ///
+ /// The BoundaryHeight property is used to create a three dimensional volume for the play space.
+ ///
+ public float BoundaryHeight => boundaryHeight;
+
+ #region Floor settings
+
+ [SerializeField]
+ [Tooltip("Should the floor be displayed in the scene?")]
+ private bool showFloor = true;
+
+ ///
+ /// Should the boundary system display the floor?
+ ///
+ public bool ShowFloor => showFloor;
+
+ // todo: consider allowing optional custom prefab
+
+ [SerializeField]
+ [Tooltip("The material to use when displaying the floor.")]
+ private Material floorMaterial = null;
+
+ ///
+ /// The material to use for the floor GameObject when created by the boundary system.
+ ///
+ public Material FloorMaterial => floorMaterial;
+
+ [PhysicsLayer]
+ [SerializeField]
+ [Tooltip("The physics layer to assign to the generated floor.")]
+ private int floorPhysicsLayer = 0;
+
+ ///
+ /// The physics layer to assign to the generated floor.
+ ///
+ public int FloorPhysicsLayer => floorPhysicsLayer;
+
+ [SerializeField]
+ [Tooltip("The dimensions of the floor, in meters.")]
+ private Vector2 floorScale = new Vector2(10f, 10f);
+
+ ///
+ /// The size at which to display the rectangular floor plane GameObject.
+ ///
+ public Vector2 FloorScale => floorScale;
+
+ #endregion Floor settings
+
+ #region Play area settings
+
+ [SerializeField]
+ [Tooltip("Should the play area be displayed in the scene?")]
+ private bool showPlayArea = true;
+
+ ///
+ /// Should the boundary system display the play area?
+ ///
+ public bool ShowPlayArea => showPlayArea;
+
+ [SerializeField]
+ [Tooltip("The material to use when displaying the play area.")]
+ private Material playAreaMaterial = null;
+
+ ///
+ /// The material to use for the rectangular play area GameObject.
+ ///
+ public Material PlayAreaMaterial => playAreaMaterial;
+
+ [PhysicsLayer]
+ [SerializeField]
+ [Tooltip("The physics layer to assign to the generated play area.")]
+ private int playAreaPhysicsLayer = 2;
+
+ ///
+ /// The physics layer to assign to the generated play area.
+ ///
+ public int PlayAreaPhysicsLayer => playAreaPhysicsLayer;
+
+ #endregion Play area settings
+
+ #region Tracked area settings
+
+ [SerializeField]
+ [Tooltip("Should the tracked area be displayed in the scene?")]
+ private bool showTrackedArea = true;
+
+ ///
+ /// Should the boundary system display the tracked area?
+ ///
+ public bool ShowTrackedArea => showTrackedArea;
+
+ [SerializeField]
+ [Tooltip("The material to use when displaying the tracked area.")]
+ private Material trackedAreaMaterial = null;
+
+ ///
+ /// The material to use for the boundary geometry GameObject.
+ ///
+ public Material TrackedAreaMaterial => trackedAreaMaterial;
+
+ [PhysicsLayer]
+ [SerializeField]
+ [Tooltip("The physics layer to assign to the generated tracked area.")]
+ private int trackedAreaPhysicsLayer = 2;
+
+ ///
+ /// The physics layer to assign to the generated tracked area.
+ ///
+ public int TrackedAreaPhysicsLayer => trackedAreaPhysicsLayer;
+
+ #endregion Tracked area settings
+
+ #region Boundary wall settings
+
+ [SerializeField]
+ [Tooltip("Should the boundary walls be displayed in the scene?")]
+ private bool showBoundaryWalls = false;
+
+ ///
+ /// Should the boundary system display the boundary geometry walls?
+ ///
+ public bool ShowBoundaryWalls => showBoundaryWalls;
+
+ [SerializeField]
+ [Tooltip("The material to use when displaying the boundary walls.")]
+ private Material boundaryWallMaterial = null;
+
+ ///
+ /// The material to use for displaying the boundary geometry walls.
+ ///
+ public Material BoundaryWallMaterial => boundaryWallMaterial;
+
+ [PhysicsLayer]
+ [SerializeField]
+ [Tooltip("The physics layer to assign to the generated boundary walls.")]
+ private int boundaryWallsPhysicsLayer = 2;
+
+ ///
+ /// The physics layer to assign to the generated boundary walls.
+ ///
+ public int BoundaryWallsPhysicsLayer => boundaryWallsPhysicsLayer;
+
+ #endregion Boundary wall settings
+
+ #region Boundary ceiling settings
+
+ [SerializeField]
+ [Tooltip("Should the boundary ceiling be displayed in the scene?")]
+ private bool showBoundaryCeiling = false;
+
+ ///
+ /// Should the boundary system display the boundary ceiling?
+ ///
+ public bool ShowBoundaryCeiling => showBoundaryCeiling;
+
+ [SerializeField]
+ [Tooltip("The material to use when displaying the boundary ceiling.")]
+ private Material boundaryCeilingMaterial = null;
+
+ ///
+ /// The material to use for displaying the boundary ceiling.
+ ///
+ public Material BoundaryCeilingMaterial => boundaryCeilingMaterial;
+
+ [PhysicsLayer]
+ [SerializeField]
+ [Tooltip("The physics layer to assign to the generated boundary ceiling.")]
+ private int ceilingPhysicsLayer = 2;
+
+ ///
+ /// The physics layer to assign to the generated boundary ceiling.
+ ///
+ public int CeilingPhysicsLayer => ceilingPhysicsLayer;
+
+ #endregion Boundary ceiling settings
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/MixedRealityBoundaryVisualizationProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/MixedRealityBoundaryVisualizationProfile.cs.meta
new file mode 100644
index 0000000..3f33aa9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/BoundarySystem/MixedRealityBoundaryVisualizationProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9fd6338e77774badb73a2b1320b41caf
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem.meta
new file mode 100644
index 0000000..2dba6f9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 60807eb531085cf4c819c2179aa32336
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/BaseCameraSettingsProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/BaseCameraSettingsProfile.cs
new file mode 100644
index 0000000..edabe21
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/BaseCameraSettingsProfile.cs
@@ -0,0 +1,16 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.CameraSystem
+{
+ ///
+ /// Base class used to derive custom camera settings profiles.
+ ///
+ [Serializable]
+ public class BaseCameraSettingsProfile : BaseMixedRealityProfile
+ {
+ // This class is intentionally blank. It exists for future expansion of common functionality.
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/BaseCameraSettingsProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/BaseCameraSettingsProfile.cs.meta
new file mode 100644
index 0000000..a47a32e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/BaseCameraSettingsProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 5d76cd22dd005ac438d4c183c7b85a54
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/CameraDisplayType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/CameraDisplayType.cs
new file mode 100644
index 0000000..6f5d612
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/CameraDisplayType.cs
@@ -0,0 +1,24 @@
+// Copyright (c) Microsoft Corporation.
+// Copyright(c) 2019 Takahiro Miyaura
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.CameraSystem
+{
+ ///
+ /// The type of displays on which an application may run.
+ ///
+ public enum DisplayType
+ {
+ ///
+ /// The display is opaque. Devices on the digital reality (ex: VR) side of the Mixed Reality
+ /// spectrum generally have opaque displays.
+ ///
+ Opaque = 0,
+
+ ///
+ /// The display is transparent. Devices on the physical reality (ex: Microsoft HoloLens) side
+ /// of the Mixed Reality spectrum generally have transparent displays.
+ ///
+ Transparent
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/CameraDisplayType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/CameraDisplayType.cs.meta
new file mode 100644
index 0000000..0dfb631
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/CameraDisplayType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: eef635c67fbf2084bb6f1106e99ae1b3
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraProfile.cs
new file mode 100644
index 0000000..e5dbb66
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraProfile.cs
@@ -0,0 +1,140 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.CameraSystem;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+using UnityEngine.Serialization;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// This Scriptable Object tells you if your head mounted display (HMD)
+ /// is a transparent device or an occluded device.
+ /// Based on those values, you can customize your camera and quality settings.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Camera Profile", fileName = "MixedRealityCameraProfile", order = (int)CreateProfileMenuItemIndices.Camera)]
+ [MixedRealityServiceProfile(typeof(IMixedRealityCameraSystem))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/configuration/mixed-reality-configuration-guide#camera")]
+ public class MixedRealityCameraProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("Configuration objects describing the registered settings providers.")]
+ private MixedRealityCameraSettingsConfiguration[] settingsConfigurations = new MixedRealityCameraSettingsConfiguration[0];
+
+ ///
+ /// Configuration objects describing the registered settings providers.
+ ///
+ public MixedRealityCameraSettingsConfiguration[] SettingsConfigurations
+ {
+ get { return settingsConfigurations; }
+ internal set { settingsConfigurations = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Near clipping plane distance for an opaque display.")]
+ private float nearClipPlaneOpaqueDisplay = 0.1f;
+
+ ///
+ /// Near clipping plane distance for an opaque display.
+ ///
+ public float NearClipPlaneOpaqueDisplay => nearClipPlaneOpaqueDisplay;
+
+ [SerializeField]
+ [Tooltip("Far clipping plane distance for an opaque display.")]
+ private float farClipPlaneOpaqueDisplay = 1000f;
+
+ ///
+ /// Far clipping plane distance for an opaque display.
+ ///
+ public float FarClipPlaneOpaqueDisplay => farClipPlaneOpaqueDisplay;
+
+ [SerializeField]
+ [Tooltip("Flags describing how to clear the camera for an opaque display.")]
+ private CameraClearFlags cameraClearFlagsOpaqueDisplay = CameraClearFlags.Skybox;
+
+ ///
+ /// Flags describing how to clear the camera for an opaque display.
+ ///
+ public CameraClearFlags CameraClearFlagsOpaqueDisplay => cameraClearFlagsOpaqueDisplay;
+
+ [SerializeField]
+ [Tooltip("Background color for an opaque display.")]
+ private Color backgroundColorOpaqueDisplay = Color.black;
+
+ ///
+ /// Background color for an opaque display.
+ ///
+ public Color BackgroundColorOpaqueDisplay => backgroundColorOpaqueDisplay;
+
+ [SerializeField]
+ [Tooltip("Quality level for an opaque display.")]
+ private int opaqueQualityLevel = 0;
+
+ ///
+ /// Quality level for an opaque display.
+ ///
+ public int OpaqueQualityLevel => opaqueQualityLevel;
+
+ [SerializeField]
+ [Tooltip("Near clipping plane distance for a transparent display.")]
+ private float nearClipPlaneTransparentDisplay = 0.85f;
+
+ ///
+ /// Near clipping plane distance for a transparent display.
+ ///
+ public float NearClipPlaneTransparentDisplay => nearClipPlaneTransparentDisplay;
+
+ [SerializeField]
+ [Tooltip("Far clipping plane distance for a transparent display.")]
+ private float farClipPlaneTransparentDisplay = 50f;
+
+ ///
+ /// Far clipping plane distance for a transparent display.
+ ///
+ public float FarClipPlaneTransparentDisplay => farClipPlaneTransparentDisplay;
+
+ [SerializeField]
+ [Tooltip("Flags describing how to clear the camera for a transparent display.")]
+ private CameraClearFlags cameraClearFlagsTransparentDisplay = CameraClearFlags.SolidColor;
+
+ ///
+ /// Flags describing how to clear the camera for a transparent display.
+ ///
+ public CameraClearFlags CameraClearFlagsTransparentDisplay => cameraClearFlagsTransparentDisplay;
+
+ [SerializeField]
+ [Tooltip("Background color for a transparent display.")]
+ private Color backgroundColorTransparentDisplay = Color.clear;
+
+ ///
+ /// Background color for a transparent display.
+ ///
+ public Color BackgroundColorTransparentDisplay => backgroundColorTransparentDisplay;
+
+ [SerializeField]
+ [Tooltip("Quality level for a transparent display.")]
+ [FormerlySerializedAs("holoLensQualityLevel")]
+ private int transparentQualityLevel = 0;
+
+ ///
+ /// Quality level for a transparent display.
+ ///
+ public int TransparentQualityLevel => transparentQualityLevel;
+
+ #region Obsolete properties
+
+ ///
+ /// Quality level for a HoloLens device.
+ ///
+ ///
+ /// HoloLensQualityLevel is obsolete and will be removed in a future Mixed Reality Toolkit release. Please use TransparentQualityLevel.
+ ///
+ [Obsolete("HoloLensQualityLevel is obsolete and will be removed in a future Mixed Reality Toolkit release. Please use TransparentQualityLevel.")]
+ public int HoloLensQualityLevel => transparentQualityLevel;
+
+ #endregion Obsolete properties
+
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraProfile.cs.meta
new file mode 100644
index 0000000..885b30c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a4a1c93114e9437cb75d8b3ee4e0e1ba
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraSettingsConfiguration.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraSettingsConfiguration.cs
new file mode 100644
index 0000000..e7b2d21
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraSettingsConfiguration.cs
@@ -0,0 +1,79 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.CameraSystem
+{
+ ///
+ /// Defines the configuration for a camera settings provider.
+ ///
+ [Serializable]
+ public struct MixedRealityCameraSettingsConfiguration : IMixedRealityServiceConfiguration
+ {
+ [SerializeField]
+ [Tooltip("The concrete type of the camera settings provider.")]
+ [Implements(typeof(IMixedRealityCameraSettingsProvider), TypeGrouping.ByNamespaceFlat)]
+ private SystemType componentType;
+
+ ///
+ public SystemType ComponentType => componentType;
+
+ [SerializeField]
+ [Tooltip("The name of the camera settings provider.")]
+ private string componentName;
+
+ ///
+ public string ComponentName => componentName;
+
+ [SerializeField]
+ [Tooltip("The camera settings provider priority.")]
+ private uint priority;
+
+ ///
+ public uint Priority => priority;
+
+ [SerializeField]
+ [Tooltip("The platform(s) on which the camera settings provider is supported.")]
+ [EnumFlags]
+ private SupportedPlatforms runtimePlatform;
+
+ ///
+ public SupportedPlatforms RuntimePlatform => runtimePlatform;
+
+ [SerializeField]
+ private BaseCameraSettingsProfile settingsProfile;
+
+ ///
+ public BaseMixedRealityProfile Profile => settingsProfile;
+
+ ///
+ /// Camera settings specific configuration profile.
+ ///
+ public BaseCameraSettingsProfile SettingsProfile => settingsProfile;
+
+ ///
+ /// Constructor.
+ ///
+ /// The of the provider.
+ /// The friendly name of the provider.
+ /// The load priority of the provider.
+ /// The runtime platform(s) supported by the provider.
+ /// The configuration profile for the provider.
+ public MixedRealityCameraSettingsConfiguration(
+ SystemType componentType,
+ string componentName,
+ uint priority,
+ SupportedPlatforms runtimePlatform,
+ BaseCameraSettingsProfile configurationProfile)
+ {
+ this.componentType = componentType;
+ this.componentName = componentName;
+ this.priority = priority;
+ this.runtimePlatform = runtimePlatform;
+ this.settingsProfile = configurationProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraSettingsConfiguration.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraSettingsConfiguration.cs.meta
new file mode 100644
index 0000000..9475d93
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/CameraSystem/MixedRealityCameraSettingsConfiguration.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f69ed9082ab6c1d44ab5f12daa194ffb
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices.meta
new file mode 100644
index 0000000..775f596
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 5c3a2b2396814e36a6bf40dfbefe3b9a
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ArticulatedHandDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ArticulatedHandDefinition.cs
new file mode 100644
index 0000000..f57d130
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ArticulatedHandDefinition.cs
@@ -0,0 +1,461 @@
+// 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
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ArticulatedHandDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ArticulatedHandDefinition.cs.meta
new file mode 100644
index 0000000..c620a87
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ArticulatedHandDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4f8a39d60b289814cbc4714d023d2b54
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/BaseInputSourceDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/BaseInputSourceDefinition.cs
new file mode 100644
index 0000000..ed89062
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/BaseInputSourceDefinition.cs
@@ -0,0 +1,60 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System.Collections.Generic;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Defines the base interactions and data that an controller can provide.
+ ///
+ public abstract class BaseInputSourceDefinition : IMixedRealityInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public BaseInputSourceDefinition(Handedness handedness)
+ {
+ Handedness = handedness;
+ }
+
+ ///
+ /// The (ex: Left, Right, None) of this controller.
+ ///
+ public Handedness Handedness { get; }
+
+ ///
+ /// The collection of interactions supported by a left-handed instance of this controller.
+ ///
+ /// Optional. Override DefaultInteractions if both handed controllers have identical interactions.
+ protected virtual MixedRealityInputActionMapping[] DefaultLeftHandedMappings => DefaultMappings;
+
+ ///
+ /// The collection of interactions supported by a right-handed instance of this controller.
+ ///
+ /// Optional. Override DefaultInteractions if both handed controllers have identical interactions.
+ protected virtual MixedRealityInputActionMapping[] DefaultRightHandedMappings => DefaultMappings;
+
+ ///
+ /// The collection of interactions supported by this controller.
+ ///
+ /// Optional. Override the specifically-handed properties if each controller has different interactions.
+ protected virtual MixedRealityInputActionMapping[] DefaultMappings => null;
+
+ ///
+ public IReadOnlyList GetDefaultMappings(Handedness handedness)
+ {
+ switch (handedness)
+ {
+ case Handedness.Left:
+ return DefaultLeftHandedMappings;
+ case Handedness.Right:
+ return DefaultRightHandedMappings;
+ default:
+ return DefaultMappings;
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/BaseInputSourceDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/BaseInputSourceDefinition.cs.meta
new file mode 100644
index 0000000..077ecb8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/BaseInputSourceDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ee63e674ce15bf944b8b1bf2bcdd0f6d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ControllerMappingLibrary.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ControllerMappingLibrary.cs
new file mode 100644
index 0000000..4f0a3f6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ControllerMappingLibrary.cs
@@ -0,0 +1,324 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+#if UNITY_EDITOR
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System;
+using System.Collections.Generic;
+using System.IO;
+using UnityEditor;
+using UnityEngine;
+#endif
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Helper utility to manage all the required Axis configuration for platforms, where required
+ ///
+ public static class ControllerMappingLibrary
+ {
+ #region Constants
+
+ ///
+ /// Axis for movement along the up (gravity) vector.
+ ///
+ public const string UP_DOWN = "UpDown";
+
+ ///
+ /// Mouse: Position Horizontal Movement
+ /// HTC Vive Controller: Left Controller Trackpad (2) Horizontal Movement
+ /// Oculus Touch Controller: Axis2D.PrimaryThumbstick Horizontal Movement
+ /// Valve Knuckles Controller: Left Controller Trackpad Horizontal Movement
+ /// Windows Mixed Reality Motion Controller: Left Thumbstick Horizontal Movement
+ /// Xbox Controller: Left Thumbstick Horizontal Movement
+ ///
+ public const string AXIS_1 = "AXIS_1";
+
+ ///
+ /// Mouse: Position Vertical Movement
+ /// HTC Vive Controller: Left Controller Trackpad (2) Vertical Movement
+ /// Oculus Touch Controller: Axis2D.PrimaryThumbstick Vertical Movement
+ /// Valve Knuckles Controller: Left Controller Trackpad Vertical Movement
+ /// Windows Mixed Reality Motion Controller: Left Thumbstick Vertical Movement
+ /// Xbox Controller: Left Thumbstick Vertical Movement
+ ///
+ public const string AXIS_2 = "AXIS_2";
+
+ ///
+ /// Mouse: Scroll
+ /// Xbox Controller: Shared Trigger
+ ///
+ public const string AXIS_3 = "AXIS_3";
+
+ ///
+ /// HTC Vive Controller: Right Controller Trackpad (2) Horizontal Movement
+ /// Oculus Touch Controller: Axis2D.SecondaryThumbstick Horizontal Movement
+ /// Valve Knuckles Controller: Right Controller Trackpad Horizontal Movement
+ /// Windows Mixed Reality Motion Controller: Right Thumbstick Horizontal Movement
+ /// Xbox Controller: Right Thumbstick Vertical Movement
+ ///
+ public const string AXIS_4 = "AXIS_4";
+
+ ///
+ /// HTC Vive Controller: Right Controller Trackpad (2) Vertical Movement
+ /// Oculus Touch Controller: Axis2D.SecondaryThumbstick Vertical Movement
+ /// Valve Knuckles Controller: Right Controller Trackpad Vertical Movement
+ /// Windows Mixed Reality Motion Controller: Right Thumbstick Vertical Movement
+ /// Xbox Controller: Right Thumbstick Vertical Movement
+ ///
+ public const string AXIS_5 = "AXIS_5";
+
+ ///
+ /// None
+ ///
+ public const string AXIS_6 = "AXIS_6";
+
+ ///
+ /// Xbox Controller: D-Pad Horizontal
+ ///
+ public const string AXIS_7 = "AXIS_7";
+
+ ///
+ /// Xbox Controller: D-Pad Vertical
+ ///
+ public const string AXIS_8 = "AXIS_8";
+
+ ///
+ /// HTC Vive Controller: Left Controller Trigger (7) Squeeze
+ /// Oculus Touch Controller: Axis1D.PrimaryIndexTrigger Squeeze
+ /// Valve Knuckles Controller: Left Controller Trigger Squeeze
+ /// Windows Mixed Reality Motion Controller: Left Trigger Squeeze
+ ///
+ public const string AXIS_9 = "AXIS_9";
+
+ ///
+ /// HTC Vive Controller: Right Controller Trigger (7) Squeeze
+ /// Oculus Touch Controller: Axis1D.SecondaryIndexTrigger Movement Squeeze
+ /// Valve Knuckles Controller: Right Controller Trigger Squeeze
+ /// Windows Mixed Reality Motion Controller: Right Trigger Squeeze
+ ///
+ public const string AXIS_10 = "AXIS_10";
+
+ ///
+ /// HTC Vive Controller: Left Controller Grip Button (8) Squeeze
+ /// Oculus Touch Controller: Axis1D.PrimaryHandTrigger Squeeze
+ /// Valve Knuckles Controller: Left Controller Grip Average Squeeze
+ /// Windows Mixed Reality Motion Controller: Left Grip Squeeze
+ ///
+ public const string AXIS_11 = "AXIS_11";
+
+ ///
+ /// HTC Vive Controller: Right Controller Grip Button (8) Squeeze
+ /// Oculus Touch Controller: Axis1D.SecondaryHandTrigger Squeeze
+ /// Valve Knuckles Controller: Right Controller Grip Average Squeeze
+ /// Windows Mixed Reality Motion Controller: Right Grip Squeeze
+ ///
+ public const string AXIS_12 = "AXIS_12";
+
+ ///
+ /// Oculus Touch Controller: Axis1D.PrimaryIndexTrigger Near Touch
+ ///
+ public const string AXIS_13 = "AXIS_13";
+
+ ///
+ /// Oculus Touch Controller: Axis1D.SecondaryIndexTrigger Near Touch
+ ///
+ public const string AXIS_14 = "AXIS_14";
+
+ ///
+ /// Oculus Touch Controller: Touch.PrimaryThumbRest Near Touch
+ ///
+ public const string AXIS_15 = "AXIS_15";
+
+ ///
+ /// Oculus Touch Controller: Button.SecondaryThumbstick Near Touch
+ ///
+ public const string AXIS_16 = "AXIS_16";
+
+ ///
+ /// Windows Mixed Reality Motion Controller: Left Touchpad Horizontal Movement
+ ///
+ public const string AXIS_17 = "AXIS_17";
+
+ ///
+ /// Windows Mixed Reality Motion Controller: Left Touchpad Vertical Movement
+ ///
+ public const string AXIS_18 = "AXIS_18";
+
+ ///
+ /// Windows Mixed Reality Motion Controller: Right Touchpad Horizontal Movement
+ ///
+ public const string AXIS_19 = "AXIS_19";
+
+ ///
+ /// Windows Mixed Reality Motion Controller: Right Touchpad Vertical Movement
+ /// Valve Knuckles Controller: Left Controller Index Finger Cap Sensor
+ ///
+ public const string AXIS_20 = "AXIS_20";
+
+ ///
+ /// Valve Knuckles Controller: Right Controller Index Finger Cap Sensor
+ ///
+ public const string AXIS_21 = "AXIS_21";
+
+ ///
+ /// Valve Knuckles Controller: Left Controller Middle Finger Cap Sensor
+ ///
+ public const string AXIS_22 = "AXIS_22";
+
+ ///
+ /// Valve Knuckles Controller: Right Controller Middle Finger Cap Sensor
+ ///
+ public const string AXIS_23 = "AXIS_23";
+
+ ///
+ /// Valve Knuckles Controller: Left Controller Ring Finger Cap Sensor
+ ///
+ public const string AXIS_24 = "AXIS_24";
+
+ ///
+ /// Valve Knuckles Controller: Right Controller Ring Finger Cap Sensor
+ ///
+ public const string AXIS_25 = "AXIS_25";
+
+ ///
+ /// Valve Knuckles Controller: Left Controller Pinky Finger Cap Sensor
+ ///
+ public const string AXIS_26 = "AXIS_26";
+
+ ///
+ /// Valve Knuckles Controller: Right Controller Pinky Finger Cap Sensor
+ ///
+ public const string AXIS_27 = "AXIS_27";
+
+ public const string AXIS_28 = "AXIS_28";
+
+ #endregion Constants
+
+#if UNITY_EDITOR
+
+ #region InputAxisConfig
+
+ // Default value for the dead zone. This should match the default used by Unity for the pre-created Horizontal and Vertical axes.
+ public const float defaultDeadZone = 0.19f;
+
+ ///
+ /// Get the InputManagerAxis data needed to configure the Input Mappings for a controller
+ ///
+ public static InputManagerAxis[] UnityInputManagerAxes => new[]
+ {
+ new InputManagerAxis { Name = AXIS_1, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 1 },
+ new InputManagerAxis { Name = AXIS_2, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 2 },
+ new InputManagerAxis { Name = AXIS_3, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 3 },
+ new InputManagerAxis { Name = AXIS_4, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 4 },
+ new InputManagerAxis { Name = AXIS_5, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 5 },
+ new InputManagerAxis { Name = AXIS_6, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 6 },
+ new InputManagerAxis { Name = AXIS_7, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 7 },
+ new InputManagerAxis { Name = AXIS_8, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 8 },
+ new InputManagerAxis { Name = AXIS_9, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 9 },
+ new InputManagerAxis { Name = AXIS_10, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 10 },
+ new InputManagerAxis { Name = AXIS_11, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 11 },
+ new InputManagerAxis { Name = AXIS_12, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 12 },
+ new InputManagerAxis { Name = AXIS_13, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 13 },
+ new InputManagerAxis { Name = AXIS_14, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 14 },
+ new InputManagerAxis { Name = AXIS_15, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 15 },
+ new InputManagerAxis { Name = AXIS_16, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 16 },
+ new InputManagerAxis { Name = AXIS_17, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 17 },
+ new InputManagerAxis { Name = AXIS_18, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 18 },
+ new InputManagerAxis { Name = AXIS_19, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 19 },
+ new InputManagerAxis { Name = AXIS_20, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 20 },
+ new InputManagerAxis { Name = AXIS_21, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 21 },
+ new InputManagerAxis { Name = AXIS_22, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 22 },
+ new InputManagerAxis { Name = AXIS_23, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 23 },
+ new InputManagerAxis { Name = AXIS_24, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 24 },
+ new InputManagerAxis { Name = AXIS_25, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 25 },
+ new InputManagerAxis { Name = AXIS_26, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 26 },
+ new InputManagerAxis { Name = AXIS_27, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 27 },
+ new InputManagerAxis { Name = AXIS_28, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 28 },
+ new InputManagerAxis { Name = UP_DOWN, Gravity = 3, Dead = 0.001f, Sensitivity = 3, Snap = true, Invert = false, Type = InputManagerAxisType.KeyOrMouseButton, PositiveButton = "e", NegativeButton = "q" },
+ new InputManagerAxis { Name = UP_DOWN, Dead = defaultDeadZone, Sensitivity = 1, Invert = false, Type = InputManagerAxisType.JoystickAxis, Axis = 3 },
+ };
+
+ #endregion InputAxisConfig
+
+ private static Dictionary, Texture2D> cachedTextures = new Dictionary, Texture2D>();
+
+ public static Texture2D GetControllerTexture(Type controllerType, Handedness handedness)
+ {
+ return GetControllerTextureCached(controllerType, handedness, "");
+ }
+
+ public static Texture2D GetControllerTextureScaled(Type controllerType, Handedness handedness)
+ {
+ return GetControllerTextureCached(controllerType, handedness, "_scaled");
+ }
+
+ private static Texture2D GetControllerTextureCached(Type controllerType, Handedness handedness, string suffix)
+ {
+ var key = new Tuple(controllerType, handedness, suffix);
+ if (cachedTextures.TryGetValue(key, out Texture2D texture))
+ {
+ return texture;
+ }
+
+ texture = GetControllerTextureInternal(controllerType, handedness, suffix);
+ cachedTextures.Add(key, texture);
+ return texture;
+ }
+
+ private static Texture2D GetControllerTextureInternal(Type controllerType, Handedness handedness, string suffix)
+ {
+ if (controllerType != null)
+ {
+ var attr = MixedRealityControllerAttribute.Find(controllerType);
+ if (attr != null)
+ {
+ if (attr.TexturePath.Length > 0)
+ {
+ Texture2D texture = GetControllerTextureInternal(attr.TexturePath, handedness, suffix);
+ if (texture != null)
+ {
+ return texture;
+ }
+ }
+ }
+ }
+
+ return GetControllerTextureInternal("Textures/Generic_controller", Handedness.None, suffix);
+ }
+
+ private static Texture2D GetControllerTextureInternal(string relativeTexturePath, Handedness handedness, string suffix)
+ {
+ string handednessSuffix = string.Empty;
+ if (handedness == Handedness.Left)
+ {
+ handednessSuffix = "_left";
+ }
+ else if (handedness == Handedness.Right)
+ {
+ handednessSuffix = "_right";
+ }
+
+ string themeSuffix = EditorGUIUtility.isProSkin ? "_white" : "_black";
+
+ string textureName = $"{Path.GetFileName(relativeTexturePath)}{handednessSuffix}{themeSuffix}{suffix}";
+ string[] textureGuids = AssetDatabase.FindAssets(textureName);
+ string texturePath = string.Empty;
+ foreach (string guid in textureGuids)
+ {
+ string tempPath = AssetDatabase.GUIDToAssetPath(guid);
+ // Ensure the path we're looking at contains the exact file name we're looking for
+ if (tempPath.Contains(textureName + ".png"))
+ {
+ texturePath = tempPath;
+ break;
+ }
+ }
+
+ return (Texture2D)AssetDatabase.LoadAssetAtPath(texturePath, typeof(Texture2D));
+ }
+
+#endif // UNITY_EDITOR
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ControllerMappingLibrary.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ControllerMappingLibrary.cs.meta
new file mode 100644
index 0000000..1baf506
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ControllerMappingLibrary.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 63448e049fca40c7bfc14ec46b6b21e2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/DeviceInputType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/DeviceInputType.cs
new file mode 100644
index 0000000..d86de01
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/DeviceInputType.cs
@@ -0,0 +1,98 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ // TODO - Expand input list for additional controller types and have a filter defined by the controller
+ ///
+ /// The InputType defines the types of input exposed by a controller.
+ /// Denoting the available buttons / interactions that a controller supports.
+ ///
+ public enum DeviceInputType
+ {
+ None = 0,
+ Gaze,
+ Voice,
+ ///
+ /// 6-DoF pointer with position and rotation.
+ ///
+ SpatialPointer,
+ ///
+ /// 3-DoF pointer with only position.
+ ///
+ PointerPosition,
+ ///
+ /// 3-DoF pointer with only rotation.
+ ///
+ PointerRotation,
+ PointerClick,
+ ButtonPress,
+ ButtonTouch,
+ ButtonNearTouch,
+ Trigger,
+ TriggerTouch,
+ TriggerNearTouch,
+ // TriggerPress, in some cases, maps to the grab/grasp gesture.
+ TriggerPress,
+ ///
+ /// 6-DoF grip with position and rotation.
+ ///
+ SpatialGrip,
+ ///
+ /// 3-DoF grip with only position.
+ ///
+ GripPosition,
+ ///
+ /// 3-DoF grip with only rotation.
+ ///
+ GripRotation,
+ ThumbStick,
+ ThumbStickPress,
+ ThumbStickTouch,
+ ThumbStickNearTouch,
+ Touchpad,
+ TouchpadTouch,
+ TouchpadNearTouch,
+ TouchpadPress,
+ ///
+ /// Select, in some cases, maps to the pinch/airtap gesture.
+ ///
+ Select,
+ Start,
+ Menu,
+ Hand,
+ Thumb,
+ ThumbTouch,
+ ThumbNearTouch,
+ ThumbPress,
+ IndexFinger,
+ IndexFingerTouch,
+ IndexFingerNearTouch,
+ IndexFingerPress,
+ MiddleFinger,
+ MiddleFingerTouch,
+ MiddleFingerNearTouch,
+ MiddleFingerPress,
+ RingFinger,
+ RingFingerTouch,
+ RingFingerNearTouch,
+ RingFingerPress,
+ PinkyFinger,
+ PinkyFingerTouch,
+ PinkyFingerNearTouch,
+ PinkyFingerPress,
+ DirectionalPad,
+ Scroll,
+ PrimaryButtonPress,
+ PrimaryButtonTouch,
+ PrimaryButtonNearTouch,
+ SecondaryButtonPress,
+ SecondaryButtonTouch,
+ SecondaryButtonNearTouch,
+ Grip,
+ GripTouch,
+ GripNearTouch,
+ // GripPress, in some cases, maps to the grab/grasp gesture.
+ GripPress,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/DeviceInputType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/DeviceInputType.cs.meta
new file mode 100644
index 0000000..57c77b9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/DeviceInputType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ef1c18787847430e8163d5884eb12480
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GenericOpenVRControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GenericOpenVRControllerDefinition.cs
new file mode 100644
index 0000000..430a316
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GenericOpenVRControllerDefinition.cs
@@ -0,0 +1,126 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class GenericOpenVRControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public GenericOpenVRControllerDefinition(Handedness handedness) : base(handedness)
+ { }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultLeftHandedMappings => new[]
+ {
+ // Controller Pose
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ // HTC Vive Controller - Left Controller Trigger (7) Squeeze
+ // Oculus Touch Controller - Axis1D.PrimaryIndexTrigger Squeeze
+ // Valve Knuckles Controller - Left Controller Trigger Squeeze
+ // Windows Mixed Reality Controller - Left Trigger Squeeze
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ // HTC Vive Controller - Left Controller Trigger (7)
+ // Oculus Touch Controller - Axis1D.PrimaryIndexTrigger
+ // Valve Knuckles Controller - Left Controller Trigger
+ // Windows Mixed Reality Controller - Left Trigger Press (Select)
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ // HTC Vive Controller - Left Controller Trigger (7)
+ // Oculus Touch Controller - Axis1D.PrimaryIndexTrigger
+ // Valve Knuckles Controller - Left Controller Trigger
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ // HTC Vive Controller - Left Controller Grip Button (8)
+ // Oculus Touch Controller - Axis1D.PrimaryHandTrigger
+ // Valve Knuckles Controller - Left Controller Grip Average
+ // Windows Mixed Reality Controller - Left Grip Button Press
+ new MixedRealityInputActionMapping("Grip Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ // HTC Vive Controller - Left Controller Trackpad (2)
+ // Oculus Touch Controller - Axis2D.PrimaryThumbstick
+ // Valve Knuckles Controller - Left Controller Trackpad
+ // Windows Mixed Reality Controller - Left Thumbstick Position
+ new MixedRealityInputActionMapping("Trackpad-Thumbstick Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ // HTC Vive Controller - Left Controller Trackpad (2)
+ // Oculus Touch Controller - Button.PrimaryThumbstick
+ // Valve Knuckles Controller - Left Controller Trackpad
+ // Windows Mixed Reality Controller - Left Touchpad Touch
+ new MixedRealityInputActionMapping("Trackpad-Thumbstick Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ // HTC Vive Controller - Left Controller Trackpad (2)
+ // Oculus Touch Controller - Button.PrimaryThumbstick
+ // Valve Knuckles Controller - Left Controller Trackpad
+ // Windows Mixed Reality Controller - Left Thumbstick Press
+ new MixedRealityInputActionMapping("Trackpad-Thumbstick Press", AxisType.Digital, DeviceInputType.TouchpadPress),
+ // HTC Vive Controller - Left Controller Menu Button (1)
+ // Oculus Touch Controller - Button.Three Press
+ // Valve Knuckles Controller - Left Controller Inner Face Button
+ // Windows Mixed Reality Controller - Left Menu Button
+ new MixedRealityInputActionMapping("Unity Button Id 2", AxisType.Digital, DeviceInputType.ButtonPress),
+ // Oculus Touch Controller - Button.Four Press
+ // Valve Knuckles Controller - Left Controller Outer Face Button
+ new MixedRealityInputActionMapping("Unity Button Id 3", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("WMR Touchpad Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ new MixedRealityInputActionMapping("WMR Touchpad Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ new MixedRealityInputActionMapping("Spatial Grip", AxisType.SixDof, DeviceInputType.SpatialGrip),
+ };
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultRightHandedMappings => new[]
+ {
+ // Controller Pose
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ // HTC Vive Controller - Right Controller Trigger (7) Squeeze
+ // Oculus Touch Controller - Axis1D.SecondaryIndexTrigger Squeeze
+ // Valve Knuckles Controller - Right Controller Trigger Squeeze
+ // Windows Mixed Reality Controller - Right Trigger Squeeze
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ // HTC Vive Controller - Right Controller Trigger (7)
+ // Oculus Touch Controller - Axis1D.SecondaryIndexTrigger
+ // Valve Knuckles Controller - Right Controller Trigger
+ // Windows Mixed Reality Controller - Right Trigger Press (Select)
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ // HTC Vive Controller - Right Controller Trigger (7)
+ // Oculus Touch Controller - Axis1D.SecondaryIndexTrigger
+ // Valve Knuckles Controller - Right Controller Trigger
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ // HTC Vive Controller - Right Controller Grip Button (8)
+ // Oculus Touch Controller - Axis1D.SecondaryHandTrigger
+ // Valve Knuckles Controller - Right Controller Grip Average
+ // Windows Mixed Reality Controller - Right Grip Button Press
+ new MixedRealityInputActionMapping("Grip Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ // HTC Vive Controller - Right Controller Trackpad (2)
+ // Oculus Touch Controller - Axis2D.PrimaryThumbstick
+ // Valve Knuckles Controller - Right Controller Trackpad
+ // Windows Mixed Reality Controller - Right Thumbstick Position
+ new MixedRealityInputActionMapping("Trackpad-Thumbstick Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ // HTC Vive Controller - Right Controller Trackpad (2)
+ // Oculus Touch Controller - Button.SecondaryThumbstick
+ // Valve Knuckles Controller - Right Controller Trackpad
+ // Windows Mixed Reality Controller - Right Touchpad Touch
+ new MixedRealityInputActionMapping("Trackpad-Thumbstick Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ // HTC Vive Controller - Right Controller Trackpad (2)
+ // Oculus Touch Controller - Button.SecondaryThumbstick
+ // Valve Knuckles Controller - Right Controller Trackpad
+ // Windows Mixed Reality Controller - Right Thumbstick Press
+ new MixedRealityInputActionMapping("Trackpad-Thumbstick Press", AxisType.Digital, DeviceInputType.TouchpadPress),
+ // HTC Vive Controller - Right Controller Menu Button (1)
+ // Oculus Remote - Button.One Press
+ // Oculus Touch Controller - Button.One Press
+ // Valve Knuckles Controller - Right Controller Inner Face Button
+ // Windows Mixed Reality Controller - Right Menu Button
+ new MixedRealityInputActionMapping("Unity Button Id 0", AxisType.Digital, DeviceInputType.ButtonPress),
+ // Oculus Remote - Button.Two Press
+ // Oculus Touch Controller - Button.Two Press
+ // Valve Knuckles Controller - Right Controller Outer Face Button
+ new MixedRealityInputActionMapping("Unity Button Id 1", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("WMR Touchpad Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ new MixedRealityInputActionMapping("WMR Touchpad Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ new MixedRealityInputActionMapping("Spatial Grip", AxisType.SixDof, DeviceInputType.SpatialGrip),
+ };
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GenericOpenVRControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GenericOpenVRControllerDefinition.cs.meta
new file mode 100644
index 0000000..5880565
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GenericOpenVRControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8a95ef72794d5af4083cbc7616bb2c35
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GestureInputType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GestureInputType.cs
new file mode 100644
index 0000000..893e6fa
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GestureInputType.cs
@@ -0,0 +1,17 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// The GestureInputType defines the types of gestures exposed by a controller.
+ ///
+ public enum GestureInputType
+ {
+ None = 0,
+ Hold,
+ Navigation,
+ Manipulation,
+ Select
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GestureInputType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GestureInputType.cs.meta
new file mode 100644
index 0000000..46e45f0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/GestureInputType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 41cdb027c55749febc76488d3a4a381a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/HPMotionControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/HPMotionControllerDefinition.cs
new file mode 100644
index 0000000..5ddea05
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/HPMotionControllerDefinition.cs
@@ -0,0 +1,59 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ public class HPMotionControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public HPMotionControllerDefinition(Handedness handedness) : base(handedness)
+ {
+ if ((handedness != Handedness.Left) &&
+ (handedness != Handedness.Right))
+ {
+ throw new System.ArgumentException($"Unsupported Handedness ({handedness}). The OculusTouchControllerDefinition supports Left and Right.");
+ }
+ }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultLeftHandedMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Spatial Grip", AxisType.SixDof, DeviceInputType.SpatialGrip),
+ new MixedRealityInputActionMapping("Grip Position", AxisType.SingleAxis, DeviceInputType.Grip),
+ new MixedRealityInputActionMapping("Grip Touch", AxisType.Digital, DeviceInputType.GripTouch),
+ new MixedRealityInputActionMapping("Grip Press", AxisType.SingleAxis, DeviceInputType.GripPress),
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ new MixedRealityInputActionMapping("Button.X Press", AxisType.Digital, DeviceInputType.PrimaryButtonPress),
+ new MixedRealityInputActionMapping("Button.Y Press", AxisType.Digital, DeviceInputType.SecondaryButtonPress),
+ new MixedRealityInputActionMapping("Menu Press", AxisType.Digital, DeviceInputType.Menu),
+ new MixedRealityInputActionMapping("Thumbstick Position", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Thumbstick Press", AxisType.Digital, DeviceInputType.ThumbStickPress),
+ };
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultRightHandedMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Spatial Grip", AxisType.SixDof, DeviceInputType.SpatialGrip),
+ new MixedRealityInputActionMapping("Grip Position", AxisType.SingleAxis, DeviceInputType.Grip),
+ new MixedRealityInputActionMapping("Grip Touch", AxisType.Digital, DeviceInputType.GripTouch),
+ new MixedRealityInputActionMapping("Grip Press", AxisType.SingleAxis, DeviceInputType.GripPress),
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ new MixedRealityInputActionMapping("Button.A Press", AxisType.Digital, DeviceInputType.PrimaryButtonPress),
+ new MixedRealityInputActionMapping("Button.B Press", AxisType.Digital, DeviceInputType.SecondaryButtonPress),
+ new MixedRealityInputActionMapping("Menu Press", AxisType.Digital, DeviceInputType.Menu),
+ new MixedRealityInputActionMapping("Thumbstick Position", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Thumbstick Press", AxisType.Digital, DeviceInputType.ThumbStickPress),
+ };
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/HPMotionControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/HPMotionControllerDefinition.cs.meta
new file mode 100644
index 0000000..9336887
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/HPMotionControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e99886907799b674dacbc8699917f8c5
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/Headset.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/Headset.cs
new file mode 100644
index 0000000..51febf9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/Headset.cs
@@ -0,0 +1,54 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ // TODO - currently not used, consider removing maybe?
+ ///
+ /// The headset definition defines the headset as defined by the SDK / Unity.
+ ///
+ public struct Headset
+ {
+ ///
+ /// The ID assigned to the Headset
+ ///
+ public string Id { get; set; }
+
+ ///
+ /// The designated hand that the controller is managing, as defined by the SDK / Unity.
+ ///
+ public SDKType HeadsetSDKType { get; set; }
+
+ ///
+ /// Indicates whether or not the headset is currently providing position data.
+ ///
+ public bool IsPositionAvailable { get; set; }
+
+ ///
+ /// Outputs the current position of the headset, as defined by the SDK / Unity.
+ ///
+ public Vector3 Position { get; set; }
+
+ ///
+ /// Indicates whether or not the headset is currently providing rotation data.
+ ///
+ public bool IsRotationAvailable { get; set; }
+
+ ///
+ /// Outputs the current rotation of the headset, as defined by the SDK / Unity.
+ ///
+ public Quaternion Rotation { get; set; }
+
+ ///
+ /// Outputs the current state of the headset, whether it is tracked or not. As defined by the SDK / Unity.
+ ///
+ public TrackingState TrackingState { get; set; }
+
+ ///
+ /// Indicates whether or not the headset display is opaque. As defined by the SDK / Unity.
+ ///
+ public bool IsOpaque { get; set; }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/Headset.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/Headset.cs.meta
new file mode 100644
index 0000000..833754d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/Headset.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: aeb86b84fa9f48bb89e0c82508cc4147
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/InputSourceType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/InputSourceType.cs
new file mode 100644
index 0000000..097117a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/InputSourceType.cs
@@ -0,0 +1,18 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// The InputSourceType defines the types of input sources.
+ ///
+ public enum InputSourceType
+ {
+ Other = 0,
+ Hand,
+ Controller,
+ Voice,
+ Head,
+ Eyes
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/InputSourceType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/InputSourceType.cs.meta
new file mode 100644
index 0000000..79022d3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/InputSourceType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ffd06547248f42243a0549fa6f484b0b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerConfigurationFlags.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerConfigurationFlags.cs
new file mode 100644
index 0000000..18dd6b8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerConfigurationFlags.cs
@@ -0,0 +1,18 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Flags used by MixedRealityControllerAttribute.
+ ///
+ [System.Flags]
+ public enum MixedRealityControllerConfigurationFlags : byte
+ {
+ ///
+ /// Controllers with custom interaction mappings can have their mappings be added / removed to the
+ /// controller mapping profile in the property inspector.
+ ///
+ UseCustomInteractionMappings = 1 << 0,
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerConfigurationFlags.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerConfigurationFlags.cs.meta
new file mode 100644
index 0000000..b22a589
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerConfigurationFlags.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 16b062a49cc40054d8a97c1c7653b17b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMapping.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMapping.cs
new file mode 100644
index 0000000..43dce68
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMapping.cs
@@ -0,0 +1,252 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Runtime.CompilerServices;
+using UnityEngine;
+
+[assembly: InternalsVisibleTo("Microsoft.MixedReality.Toolkit.Editor.Inspectors")]
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Used to define a controller or other input device's physical buttons, and other attributes.
+ ///
+ [Serializable]
+ public struct MixedRealityControllerMapping
+ {
+ ///
+ /// Constructor.
+ ///
+ /// Controller Type to instantiate at runtime.
+ /// The designated hand that the device is managing.
+ public MixedRealityControllerMapping(Type controllerType, Handedness handedness = Handedness.None) : this()
+ {
+ this.controllerType = new SystemType(controllerType);
+ this.handedness = handedness;
+ interactions = null;
+ }
+
+ ///
+ /// Description of the Device.
+ ///
+ public string Description
+ {
+ get
+ {
+ string controllerName = "Unknown";
+ if (controllerType.Type != null)
+ {
+ var attr = MixedRealityControllerAttribute.Find(controllerType);
+ if (attr != null)
+ {
+ controllerName = attr.SupportedControllerType.ToString().ToProperCase();
+ }
+ }
+
+ string handednessText = string.Empty;
+ switch (handedness)
+ {
+ case Handedness.Left:
+ case Handedness.Right:
+ handednessText = $"{handedness} Hand ";
+ // Avoid multiple occurrences of "Hand":
+ controllerName = controllerName.Replace("Hand", "").Trim();
+ break;
+ }
+
+ return $"{controllerName} {handednessText}Controller";
+ }
+ }
+
+ [SerializeField]
+ [Tooltip("Controller type to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealityController), TypeGrouping.ByNamespaceFlat)]
+ private SystemType controllerType;
+
+ ///
+ /// Controller Type to instantiate at runtime.
+ ///
+ public SystemType ControllerType => controllerType;
+
+ public SupportedControllerType SupportedControllerType
+ {
+ get
+ {
+ if (controllerType.Type != null)
+ {
+ var attr = MixedRealityControllerAttribute.Find(controllerType);
+ if (attr != null)
+ {
+ return attr.SupportedControllerType;
+ }
+ }
+ return 0;
+ }
+ }
+
+ [SerializeField]
+ [Tooltip("The designated hand that the device is managing.")]
+ private Handedness handedness;
+
+ ///
+ /// The designated hand that the device is managing.
+ ///
+ public Handedness Handedness => handedness;
+
+ ///
+ /// Is this controller mapping using custom interactions?
+ ///
+ public bool HasCustomInteractionMappings
+ {
+ get
+ {
+ if (controllerType.Type != null)
+ {
+ var attr = MixedRealityControllerAttribute.Find(controllerType);
+ if (attr != null)
+ {
+ return attr.Flags.HasFlag(MixedRealityControllerConfigurationFlags.UseCustomInteractionMappings);
+ }
+ }
+ return false;
+ }
+ }
+
+ [SerializeField]
+ [Tooltip("Details the list of available buttons / interactions available from the device.")]
+ private MixedRealityInteractionMapping[] interactions;
+
+ ///
+ /// Details the list of available buttons / interactions available from the device.
+ ///
+ public MixedRealityInteractionMapping[] Interactions => interactions;
+
+ ///
+ /// Sets the default interaction mapping based on the current controller type.
+ ///
+ internal void SetDefaultInteractionMapping(bool overwrite = false)
+ {
+ if (interactions == null || interactions.Length == 0 || overwrite)
+ {
+ MixedRealityInteractionMapping[] defaultMappings = GetDefaultInteractionMappings();
+
+ if (defaultMappings != null)
+ {
+ interactions = defaultMappings;
+ }
+ }
+ }
+
+ internal bool UpdateInteractionSettingsFromDefault()
+ {
+ if (interactions == null || interactions.Length == 0) { return false; }
+
+ MixedRealityInteractionMapping[] newDefaultInteractions = GetDefaultInteractionMappings();
+
+ if (newDefaultInteractions == null)
+ {
+ return false;
+ }
+
+ if (interactions.Length != newDefaultInteractions.Length)
+ {
+ interactions = CreateNewMatchedMapping(interactions, newDefaultInteractions);
+ return true;
+ }
+
+ bool updatedMappings = false;
+
+ for (int i = 0; i < newDefaultInteractions.Length; i++)
+ {
+ MixedRealityInteractionMapping currentMapping = interactions[i];
+ MixedRealityInteractionMapping currentDefaultMapping = newDefaultInteractions[i];
+
+ if (!Equals(currentMapping, currentDefaultMapping))
+ {
+ interactions[i] = new MixedRealityInteractionMapping(currentDefaultMapping)
+ {
+ MixedRealityInputAction = currentMapping.MixedRealityInputAction
+ };
+
+ updatedMappings = true;
+ }
+ }
+
+ return updatedMappings;
+ }
+
+ private MixedRealityInteractionMapping[] CreateNewMatchedMapping(MixedRealityInteractionMapping[] interactions, MixedRealityInteractionMapping[] newDefaultInteractions)
+ {
+ MixedRealityInteractionMapping[] newDefaultMapping = new MixedRealityInteractionMapping[newDefaultInteractions.Length];
+
+ for (int i = 0; i < newDefaultInteractions.Length; i++)
+ {
+ for (int j = 0; j < interactions.Length; j++)
+ {
+ if (Equals(interactions[j], newDefaultInteractions[i]))
+ {
+ newDefaultMapping[i] = new MixedRealityInteractionMapping(newDefaultInteractions[i])
+ {
+ MixedRealityInputAction = interactions[j].MixedRealityInputAction
+ };
+ break;
+ }
+ }
+
+ if (newDefaultMapping[i] == null)
+ {
+ newDefaultMapping[i] = new MixedRealityInteractionMapping(newDefaultInteractions[i]);
+ }
+ }
+
+ return newDefaultMapping;
+ }
+
+ private bool Equals(MixedRealityInteractionMapping a, MixedRealityInteractionMapping b)
+ {
+ return !(a.Description != b.Description ||
+ a.AxisType != b.AxisType ||
+ a.InputType != b.InputType ||
+ a.KeyCode != b.KeyCode ||
+ a.AxisCodeX != b.AxisCodeX ||
+ a.AxisCodeY != b.AxisCodeY ||
+ a.InvertXAxis != b.InvertXAxis ||
+ a.InvertYAxis != b.InvertYAxis);
+ }
+
+ private MixedRealityInteractionMapping[] GetDefaultInteractionMappings()
+ {
+ if (Activator.CreateInstance(controllerType, TrackingState.NotTracked, handedness, null, null) is BaseController detectedController)
+ {
+ switch (handedness)
+ {
+ case Handedness.Left:
+ return detectedController.DefaultLeftHandedInteractions ?? detectedController.DefaultInteractions;
+ case Handedness.Right:
+ return detectedController.DefaultRightHandedInteractions ?? detectedController.DefaultInteractions;
+ default:
+ return detectedController.DefaultInteractions;
+ }
+ }
+
+ return null;
+ }
+
+ ///
+ /// Synchronizes the input actions of the same physical controller of a different concrete type.
+ ///
+ internal void SynchronizeInputActions(MixedRealityInteractionMapping[] otherControllerMapping)
+ {
+ if (otherControllerMapping.Length != interactions.Length)
+ {
+ throw new ArgumentException($"otherControllerMapping length {otherControllerMapping.Length} does not match this length {interactions.Length}.");
+ }
+
+ for (int i = 0; i < interactions.Length; i++)
+ {
+ interactions[i].MixedRealityInputAction = otherControllerMapping[i].MixedRealityInputAction;
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMapping.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMapping.cs.meta
new file mode 100644
index 0000000..7d487e5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMapping.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 02eb7c6d2cea47968dbe9662d1ca935d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMappingProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMappingProfile.cs
new file mode 100644
index 0000000..3dc127a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMappingProfile.cs
@@ -0,0 +1,230 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Reflection;
+#if UNITY_EDITOR
+using UnityEditor;
+#endif // UNITY_EDITOR
+using UnityEngine;
+using UnityEngine.Serialization;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// New controller types can be registered by adding the MixedRealityControllerAttribute to
+ /// the controller class.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Controller Mapping Profile", fileName = "MixedRealityControllerMappingProfile", order = (int)CreateProfileMenuItemIndices.ControllerMapping)]
+ public class MixedRealityControllerMappingProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("The list of controller mappings your application can use.")]
+ [FormerlySerializedAs("mixedRealityControllerMappingProfiles")]
+ private MixedRealityControllerMapping[] mixedRealityControllerMappings = Array.Empty();
+
+ ///
+ /// The list of controller mappings your application can use.
+ ///
+ public MixedRealityControllerMapping[] MixedRealityControllerMappings => mixedRealityControllerMappings;
+
+ [Obsolete("MixedRealityControllerMappingProfiles is obsolete. Please use MixedRealityControllerMappings.")]
+ public MixedRealityControllerMapping[] MixedRealityControllerMappingProfiles => mixedRealityControllerMappings;
+
+#if UNITY_EDITOR
+ [MenuItem("Mixed Reality/Toolkit/Utilities/Update/Controller Mapping Profiles")]
+ private static void UpdateAllControllerMappingProfiles()
+ {
+ string[] guids = AssetDatabase.FindAssets("t:MixedRealityControllerMappingProfile");
+ string[] assetPaths = new string[guids.Length];
+ for (int i = 0; i < guids.Length; i++)
+ {
+ string guid = guids[i];
+ assetPaths[i] = AssetDatabase.GUIDToAssetPath(guid);
+
+ MixedRealityControllerMappingProfile asset = AssetDatabase.LoadAssetAtPath(assetPaths[i], typeof(MixedRealityControllerMappingProfile)) as MixedRealityControllerMappingProfile;
+
+ List updatedMappings = new List();
+
+ foreach (MixedRealityControllerMapping mapping in asset.MixedRealityControllerMappings)
+ {
+ if (mapping.ControllerType.Type == null)
+ {
+ continue;
+ }
+
+ if (!mapping.HasCustomInteractionMappings)
+ {
+ mapping.UpdateInteractionSettingsFromDefault();
+ }
+
+ updatedMappings.Add(mapping);
+ }
+
+ asset.mixedRealityControllerMappings = updatedMappings.ToArray();
+ }
+ AssetDatabase.ForceReserializeAssets(assetPaths);
+ }
+
+ private static Type[] controllerMappingTypes;
+
+ public static Type[] ControllerMappingTypes { get { CollectControllerTypes(); return controllerMappingTypes; } }
+
+ public static Type[] CustomControllerMappingTypes { get => (from type in ControllerMappingTypes where UsesCustomInteractionMapping(type) select type).ToArray(); }
+
+ private static void CollectControllerTypes()
+ {
+ if (controllerMappingTypes == null)
+ {
+ List controllerTypes = new List();
+ foreach (Assembly assembly in AppDomain.CurrentDomain.GetAssemblies())
+ {
+ IEnumerable types = null;
+ try
+ {
+ types = assembly.ExportedTypes;
+ }
+ catch (NotSupportedException)
+ {
+ // assembly.ExportedTypes may not be supported.
+ }
+ catch (ReflectionTypeLoadException e)
+ {
+ // Not all assemblies may load correctly, but even upon encountering error
+ // some subset may have loaded in.
+ if (e.Types != null)
+ {
+ List loadedTypes = new List();
+ foreach (Type type in e.Types)
+ {
+ // According to API docs, this array may contain null values
+ // so they must be filtered out here.
+ if (type != null)
+ {
+ loadedTypes.Add(type);
+ }
+ }
+ types = loadedTypes;
+ }
+ }
+
+ if (types != null)
+ {
+ foreach (Type type in types)
+ {
+ if (type.IsSubclassOf(typeof(BaseController)) &&
+ MixedRealityControllerAttribute.Find(type) != null)
+ {
+ controllerTypes.Add(type);
+ }
+ }
+ }
+ }
+
+ controllerMappingTypes = controllerTypes.ToArray();
+ }
+ }
+
+ public void Awake()
+ {
+ AddMappings();
+ SortMappings();
+ }
+
+ private void AddMappings()
+ {
+ foreach (var controllerType in ControllerMappingTypes)
+ {
+ // Don't auto-add custom mappings when migrating, these can be removed by the user in the inspector.
+ if (UsesCustomInteractionMapping(controllerType))
+ {
+ continue;
+ }
+
+ foreach (Handedness handedness in GetSupportedHandedness(controllerType))
+ {
+ // Try to find index of mapping in asset.
+ int idx = Array.FindIndex(MixedRealityControllerMappings, 0, MixedRealityControllerMappings.Length,
+ profile => profile.ControllerType.Type == controllerType && profile.Handedness == handedness);
+
+ if (idx < 0)
+ {
+ var newMapping = new MixedRealityControllerMapping(controllerType, handedness);
+ newMapping.SetDefaultInteractionMapping(overwrite: false);
+
+ // Re-use existing mapping with the same supported controller type.
+ foreach (var otherMapping in mixedRealityControllerMappings)
+ {
+ if (otherMapping.SupportedControllerType == newMapping.SupportedControllerType &&
+ otherMapping.Handedness == newMapping.Handedness)
+ {
+ try
+ {
+ newMapping.SynchronizeInputActions(otherMapping.Interactions);
+ }
+ catch (ArgumentException e)
+ {
+ Debug.LogError($"Controller mappings between {newMapping.Description} and {otherMapping.Description} do not match. Error message: {e.Message}");
+ }
+ break;
+ }
+ }
+
+ idx = mixedRealityControllerMappings.Length;
+ Array.Resize(ref mixedRealityControllerMappings, idx + 1);
+ mixedRealityControllerMappings[idx] = newMapping;
+ }
+ else
+ {
+ mixedRealityControllerMappings[idx].UpdateInteractionSettingsFromDefault();
+ }
+ }
+ }
+ }
+
+ private void SortMappings()
+ {
+ Array.Sort(mixedRealityControllerMappings, (profile1, profile2) =>
+ {
+ bool isOptional1 = (profile1.ControllerType.Type == null || profile1.HasCustomInteractionMappings);
+ bool isOptional2 = (profile2.ControllerType.Type == null || profile2.HasCustomInteractionMappings);
+ if (!isOptional1 && !isOptional2)
+ {
+ int idx1 = Array.FindIndex(ControllerMappingTypes, type => type == profile1.ControllerType.Type);
+ int idx2 = Array.FindIndex(ControllerMappingTypes, type => type == profile2.ControllerType.Type);
+
+ if (idx1 == idx2)
+ {
+ idx1 = (int)profile1.Handedness;
+ idx2 = (int)profile2.Handedness;
+ }
+
+ return Math.Sign(idx1 - idx2);
+ }
+
+ if (isOptional1 && isOptional2)
+ {
+ return 0;
+ }
+
+ return isOptional1 ? 1 : -1; // Put custom mappings at the end. These can be added / removed in the inspector.
+ });
+ }
+
+ private static bool UsesCustomInteractionMapping(Type controllerType)
+ {
+ var attribute = MixedRealityControllerAttribute.Find(controllerType);
+ return attribute != null && attribute.Flags.HasFlag(MixedRealityControllerConfigurationFlags.UseCustomInteractionMappings);
+ }
+
+ private static Handedness[] GetSupportedHandedness(Type controllerType)
+ {
+ var attribute = MixedRealityControllerAttribute.Find(controllerType);
+ return attribute != null ? attribute.SupportedHandedness : Array.Empty();
+ }
+#endif // UNITY_EDITOR
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMappingProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMappingProfile.cs.meta
new file mode 100644
index 0000000..b98023a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerMappingProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1a1cf5fd47e548e6a23bdd3ddcc00cf6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerModelHelpers.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerModelHelpers.cs
new file mode 100644
index 0000000..62c539c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerModelHelpers.cs
@@ -0,0 +1,62 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Provides helpers for setting up the controller model with a visualization script.
+ ///
+ public static class MixedRealityControllerModelHelpers
+ {
+ private static MixedRealityControllerVisualizationProfile visualizationProfile = null;
+
+ ///
+ /// Tries to read the controller visualization profile to apply a visualization script to the passed-in controller model.
+ ///
+ /// Automatically disables DestroyOnSourceLost to encourage controller model creators to manage their life-cycle themselves.
+ /// The GameObject to modify.
+ /// The type of controller this model represents.
+ /// The handedness of this controller.
+ /// True if a visualization script could be loaded and applied.
+ public static bool TryAddVisualizationScript(GameObject controllerModel, Type controllerType, Handedness handedness)
+ {
+ if (controllerModel != null)
+ {
+ if (visualizationProfile == null && CoreServices.InputSystem?.InputSystemProfile != null)
+ {
+ visualizationProfile = CoreServices.InputSystem.InputSystemProfile.ControllerVisualizationProfile;
+ }
+
+ if (visualizationProfile != null)
+ {
+ var visualizationType = visualizationProfile.GetControllerVisualizationTypeOverride(controllerType, handedness);
+ if (visualizationType != null)
+ {
+ // Set the platform controller model to not be destroyed when the source is lost. It'll be disabled instead,
+ // and re-enabled when the same controller is re-detected.
+ if (controllerModel.EnsureComponent(visualizationType.Type) is IMixedRealityControllerPoseSynchronizer visualizer)
+ {
+ visualizer.DestroyOnSourceLost = false;
+ }
+
+ return true;
+ }
+ else
+ {
+ Debug.LogError("Controller visualization type not defined for controller visualization profile");
+ }
+ }
+ else
+ {
+ Debug.LogError("Failed to obtain a controller visualization profile");
+ }
+ }
+
+ return false;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerModelHelpers.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerModelHelpers.cs.meta
new file mode 100644
index 0000000..6682308
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerModelHelpers.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4cb80de25855ae343a2b66dc2d2c197c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationProfile.cs
new file mode 100644
index 0000000..c8c9359
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationProfile.cs
@@ -0,0 +1,254 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+using UnityEngine.Serialization;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Profile that determines relevant overrides and properties for controller visualization
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Controller Visualization Profile", fileName = "MixedRealityControllerVisualizationProfile", order = (int)CreateProfileMenuItemIndices.ControllerVisualization)]
+ [MixedRealityServiceProfile(typeof(IMixedRealityControllerVisualizer))]
+ public class MixedRealityControllerVisualizationProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("Enable and configure the controller rendering of the Motion Controllers on Startup.")]
+ private bool renderMotionControllers = false;
+
+ ///
+ /// Enable and configure the controller rendering of the Motion Controllers on Startup.
+ ///
+ public bool RenderMotionControllers
+ {
+ get => renderMotionControllers;
+ private set => renderMotionControllers = value;
+ }
+
+ [SerializeField]
+ [Implements(typeof(IMixedRealityControllerVisualizer), TypeGrouping.ByNamespaceFlat)]
+ [Tooltip("The default controller visualization type. This value is used as a fallback if no controller definition exists with a custom visualization type.")]
+ private SystemType defaultControllerVisualizationType;
+
+ ///
+ /// The default controller visualization type. This value is used as a fallback if no controller definition exists with a custom visualization type.
+ ///
+ public SystemType DefaultControllerVisualizationType
+ {
+ get => defaultControllerVisualizationType;
+ private set => defaultControllerVisualizationType = value;
+ }
+
+ [SerializeField]
+ [Tooltip("Check to obtain controller models from the platform SDK. If left unchecked, the global models will be used. Note: this value is overridden by controller definitions.")]
+ [FormerlySerializedAs("useDefaultModels")]
+ private bool usePlatformModels = false;
+
+ ///
+ /// Check to obtain controller models from the platform SDK. If left unchecked, the global models will be used. Note: this value is overridden by controller definitions.
+ ///
+ public bool UsePlatformModels
+ {
+ get => usePlatformModels;
+ private set => usePlatformModels = value;
+ }
+
+ ///
+ /// Check to obtain controller models from the platform SDK. If left unchecked, the global models will be used. Note: this value is overridden by controller definitions.
+ ///
+ [Obsolete("Use UsePlatformModels instead.")]
+ public bool UseDefaultModels => usePlatformModels;
+
+ [SerializeField]
+ [Tooltip("The default controller model material when loading platform SDK controller models. This value is used as a fallback if no controller definition exists with a custom material type.")]
+ [FormerlySerializedAs("defaultControllerModelMaterial")]
+ private Material platformModelMaterial;
+
+ ///
+ /// The default controller model material when loading platform SDK controller models. This value is used as a fallback if no controller definition exists with a custom material type.
+ ///
+ public Material PlatformModelMaterial
+ {
+ get => platformModelMaterial;
+ private set => platformModelMaterial = value;
+ }
+
+ ///
+ /// The default controller model material when loading platform SDK controller models. This value is used as a fallback if no controller definition exists with a custom material type.
+ ///
+ [Obsolete("Use PlatformModelMaterial instead.")]
+ public Material DefaultControllerModelMaterial => platformModelMaterial;
+
+ [SerializeField]
+ [Tooltip("Override Left Controller Model.")]
+ private GameObject globalLeftControllerModel;
+
+ ///
+ /// The Default controller model when there is no specific controller model for the Left hand or when no hand is specified (Handedness = none)
+ ///
+ ///
+ /// If the default model for the left hand controller can not be found, the controller will fall back and use this for visualization.
+ ///
+ public GameObject GlobalLeftHandModel
+ {
+ get => globalLeftControllerModel;
+ private set => globalLeftControllerModel = value;
+ }
+
+ [SerializeField]
+ [Tooltip("Override Right Controller Model.\nNote: If the default model is not found, the fallback is the global right hand model.")]
+ private GameObject globalRightControllerModel;
+
+ ///
+ /// The Default controller model when there is no specific controller model for the Right hand.
+ ///
+ ///
+ /// If the default model for the right hand controller can not be found, the controller will fall back and use this for visualization.
+ ///
+ public GameObject GlobalRightHandModel
+ {
+ get => globalRightControllerModel;
+ private set => globalRightControllerModel = value;
+ }
+
+ [SerializeField]
+ [Tooltip("Override Left Hand Visualizer.")]
+ private GameObject globalLeftHandVisualizer;
+
+ ///
+ /// The Default controller model when there is no specific controller model for the Left hand or when no hand is specified (Handedness = none)
+ ///
+ ///
+ /// If the default model for the left hand controller can not be found, the controller will fall back and use this for visualization.
+ ///
+ public GameObject GlobalLeftHandVisualizer
+ {
+ get => globalLeftHandVisualizer;
+ private set => globalLeftHandVisualizer = value;
+ }
+
+ [SerializeField]
+ [Tooltip("Override Right Controller Model.\nNote: If the default model is not found, the fallback is the global right hand model.")]
+ private GameObject globalRightHandVisualizer;
+
+ ///
+ /// The Default hand model when there is no specific hand model for the Right hand.
+ ///
+ ///
+ /// If the default model for the right hand can not be found, the hand will fall back and use this for visualization.
+ ///
+ public GameObject GlobalRightHandVisualizer
+ {
+ get => globalRightHandVisualizer;
+ private set => globalRightHandVisualizer = value;
+ }
+
+ [SerializeField]
+ private MixedRealityControllerVisualizationSetting[] controllerVisualizationSettings = Array.Empty();
+
+ ///
+ /// The current list of controller visualization settings.
+ ///
+ public MixedRealityControllerVisualizationSetting[] ControllerVisualizationSettings => controllerVisualizationSettings;
+
+ private MixedRealityControllerVisualizationSetting? GetControllerVisualizationDefinition(Type controllerType, Handedness hand)
+ {
+ for (int i = 0; i < controllerVisualizationSettings.Length; i++)
+ {
+ if (SettingContainsParameters(controllerVisualizationSettings[i], controllerType, hand))
+ {
+ return controllerVisualizationSettings[i];
+ }
+ }
+ return null;
+ }
+
+ ///
+ /// Gets the override model for a specific controller type and hand
+ ///
+ /// The type of controller to query for
+ /// The specific hand assigned to the controller
+ public GameObject GetControllerModelOverride(Type controllerType, Handedness hand)
+ {
+ MixedRealityControllerVisualizationSetting? setting = GetControllerVisualizationDefinition(controllerType, hand);
+ return setting.HasValue ? setting.Value.OverrideControllerModel : null;
+ }
+
+ ///
+ /// Gets the override type for a specific controller type and hand.
+ /// If the requested controller type is not defined, DefaultControllerVisualizationType is returned.
+ ///
+ /// The type of controller to query for
+ /// The specific hand assigned to the controller
+ public SystemType GetControllerVisualizationTypeOverride(Type controllerType, Handedness hand)
+ {
+ MixedRealityControllerVisualizationSetting? setting = GetControllerVisualizationDefinition(controllerType, hand);
+ return setting.HasValue ? setting.Value.ControllerVisualizationType : DefaultControllerVisualizationType;
+ }
+
+ ///
+ /// Gets the UsePlatformModels value defined for the specified controller definition.
+ /// If the requested controller type is not defined, the default UsePlatformModels is returned.
+ ///
+ /// The type of controller to query for
+ /// The specific hand assigned to the controller
+ ///
+ /// GetUseDefaultModelsOverride is obsolete and will be removed in a future Mixed Reality Toolkit release. Please use GetUsePlatformModelsOverride.
+ ///
+ [Obsolete("GetUseDefaultModelsOverride is obsolete and will be removed in a future Mixed Reality Toolkit release. Please use GetUsePlatformModelsOverride.")]
+ public bool GetUseDefaultModelsOverride(Type controllerType, Handedness hand)
+ {
+ return GetUsePlatformModelsOverride(controllerType, hand);
+ }
+
+ ///
+ /// Gets the UsePlatformModels value defined for the specified controller definition.
+ /// If the requested controller type is not defined, the default UsePlatformModels is returned.
+ ///
+ /// The type of controller to query for
+ /// The specific hand assigned to the controller
+ public bool GetUsePlatformModelsOverride(Type controllerType, Handedness hand)
+ {
+ MixedRealityControllerVisualizationSetting? setting = GetControllerVisualizationDefinition(controllerType, hand);
+ return setting.HasValue ? setting.Value.UsePlatformModels : usePlatformModels;
+ }
+
+ ///
+ /// Gets the DefaultModelMaterial value defined for the specified controller definition.
+ /// If the requested controller type is not defined, the global platformModelMaterial is returned.
+ ///
+ /// The type of controller to query for
+ /// The specific hand assigned to the controller
+ ///
+ /// GetDefaultControllerModelMaterialOverride is obsolete and will be removed in a future Mixed Reality Toolkit release. Please use GetPlatformModelMaterialOverride.
+ ///
+ [Obsolete("GetDefaultControllerModelMaterialOverride is obsolete and will be removed in a future Mixed Reality Toolkit release. Please use GetPlatformModelMaterial.")]
+ public Material GetDefaultControllerModelMaterialOverride(Type controllerType, Handedness hand)
+ {
+ return GetPlatformModelMaterialOverride(controllerType, hand);
+ }
+
+ ///
+ /// Gets the PlatformModelMaterial value defined for the specified controller definition.
+ /// If the requested controller type is not defined, the global platformModelMaterial is returned.
+ ///
+ /// The type of controller to query for
+ /// The specific hand assigned to the controller
+ public Material GetPlatformModelMaterialOverride(Type controllerType, Handedness hand)
+ {
+
+ MixedRealityControllerVisualizationSetting? setting = GetControllerVisualizationDefinition(controllerType, hand);
+ return setting.HasValue ? setting.Value.PlatformModelMaterial : platformModelMaterial;
+ }
+
+ private bool SettingContainsParameters(MixedRealityControllerVisualizationSetting setting, Type controllerType, Handedness hand)
+ {
+ return setting.ControllerType != null &&
+ setting.ControllerType.Type == controllerType &&
+ setting.Handedness.IsMaskSet(hand) && setting.Handedness != Handedness.None;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationProfile.cs.meta
new file mode 100644
index 0000000..e06d0c5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 514da61389c049c0bdaf31b7f6970d33
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationSetting.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationSetting.cs
new file mode 100644
index 0000000..f7bbba0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationSetting.cs
@@ -0,0 +1,105 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+using UnityEngine.Serialization;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Used to define a controller's visualization settings.
+ ///
+ [Serializable]
+ public struct MixedRealityControllerVisualizationSetting
+ {
+ ///
+ /// Constructor.
+ ///
+ /// Description of the Device.
+ /// Controller Type to instantiate at runtime.
+ /// The designated hand that the device is managing.
+ /// The controller model prefab to be rendered.
+ [Obsolete("This constructor doesn't need to be called directly.")]
+ public MixedRealityControllerVisualizationSetting(string description, Type controllerType, Handedness handedness = Handedness.None, GameObject overrideModel = null) : this()
+ {
+ this.description = description;
+ this.controllerType = new SystemType(controllerType);
+ this.handedness = handedness;
+ this.overrideModel = overrideModel;
+ usePlatformModels = false;
+ platformModelMaterial = null;
+ }
+
+ [SerializeField]
+ private string description;
+
+ ///
+ /// Description of the Device.
+ ///
+ public string Description => description;
+
+ [SerializeField]
+ [Tooltip("Controller type to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealityController), TypeGrouping.ByNamespaceFlat)]
+ private SystemType controllerType;
+
+ ///
+ /// Controller Type to instantiate at runtime.
+ ///
+ public SystemType ControllerType => controllerType;
+
+ [SerializeField]
+ [Tooltip("The designated hand that the device is managing.")]
+ private Handedness handedness;
+
+ ///
+ /// The designated hand that the device is managing.
+ ///
+ public Handedness Handedness => handedness;
+
+ [SerializeField]
+ [Tooltip("Check to obtain controller models from the platform SDK. If left unchecked, the global models will be used.")]
+ [FormerlySerializedAs("useDefaultModel")]
+ private bool usePlatformModels;
+
+ ///
+ /// Check to obtain controller models from the platform SDK. If left unchecked, the global models will be used.
+ ///
+ public bool UsePlatformModels => usePlatformModels;
+
+ [SerializeField]
+ [Tooltip("The default controller model material when loading platform SDK controller models.")]
+ [FormerlySerializedAs("defaultModelMaterial")]
+ private Material platformModelMaterial;
+
+ ///
+ /// The default controller model material when loading platform SDK controller models. This value is used as a fallback if no controller definition exists with a custom material type.
+ ///
+ public Material PlatformModelMaterial => platformModelMaterial;
+
+ [SerializeField]
+ [Tooltip("An override model to display for this specific controller.")]
+ private GameObject overrideModel;
+
+ ///
+ /// The controller model prefab to be rendered.
+ ///
+ public GameObject OverrideControllerModel => overrideModel;
+
+ [SerializeField]
+ [Tooltip("The concrete Controller Visualizer component to use on the rendered controller model.")]
+ [Implements(typeof(IMixedRealityControllerVisualizer), TypeGrouping.ByNamespaceFlat)]
+ private SystemType controllerVisualizationType;
+
+ ///
+ /// The concrete Controller Visualizer component to use on the rendered controller model
+ ///
+ public SystemType ControllerVisualizationType
+ {
+ get => controllerVisualizationType;
+ private set => controllerVisualizationType = value;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationSetting.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationSetting.cs.meta
new file mode 100644
index 0000000..5195bb8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityControllerVisualizationSetting.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 47b950ed46534082a4eacaa769c11eb9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityEyeTrackingProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityEyeTrackingProfile.cs
new file mode 100644
index 0000000..f6ba966
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityEyeTrackingProfile.cs
@@ -0,0 +1,23 @@
+// 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 Eye Tracking Profile", fileName = "MixedRealityEyeTrackingProfile", order = (int)CreateProfileMenuItemIndices.EyeTracking)]
+ [MixedRealityServiceProfile(requiredTypes: new Type[] { typeof(IMixedRealityEyeGazeDataProvider), typeof(IMixedRealityEyeSaccadeProvider) })]
+ public class MixedRealityEyeTrackingProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("Use smoothed eye tracking signal.")]
+ private bool smoothEyeTracking = false;
+
+ ///
+ /// Use smoothed eye tracking signal.
+ ///
+ public bool SmoothEyeTracking => smoothEyeTracking;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityEyeTrackingProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityEyeTrackingProfile.cs.meta
new file mode 100644
index 0000000..4412cf7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityEyeTrackingProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e534d3d8fbada634785d567d8fe8562c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityHandTrackingProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityHandTrackingProfile.cs
new file mode 100644
index 0000000..bdb0e77
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityHandTrackingProfile.cs
@@ -0,0 +1,159 @@
+// 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
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityHandTrackingProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityHandTrackingProfile.cs.meta
new file mode 100644
index 0000000..5229b4f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityHandTrackingProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8275efdbe76bdff49a97a8e82fba118d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInputActionMapping.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInputActionMapping.cs
new file mode 100644
index 0000000..f9c4594
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInputActionMapping.cs
@@ -0,0 +1,82 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Maps the capabilities of controllers, defining the physical inputs of a controller.
+ ///
+ ///
+ /// One definition should exist for each physical device input, such as buttons, triggers, joysticks, dpads, etc.
+ ///
+ [Serializable]
+ public class MixedRealityInputActionMapping
+ {
+ ///
+ /// The constructor for a new MixedRealityInputActionMapping definition
+ ///
+ /// The description of the interaction mapping.
+ /// The logical MixedRealityInputAction that this input performs
+ public MixedRealityInputActionMapping(string description, AxisType axisType, DeviceInputType inputType) :
+ this(description, axisType, inputType, MixedRealityInputAction.None)
+ {
+ }
+
+ ///
+ /// The constructor for a new MixedRealityInputActionMapping definition
+ ///
+ /// The description of the interaction mapping.
+ /// The logical MixedRealityInputAction that this input performs
+ public MixedRealityInputActionMapping(string description, AxisType axisType, DeviceInputType inputType, MixedRealityInputAction inputAction)
+ {
+ this.description = description;
+ this.axisType = axisType;
+ this.inputType = inputType;
+ this.inputAction = inputAction;
+ }
+
+ [SerializeField]
+ [Tooltip("The description of the interaction mapping.")]
+ private string description;
+
+ ///
+ /// The description of the input action mapping.
+ ///
+ public string Description => description;
+
+ [SerializeField]
+ [Tooltip("The axis type of the button, e.g. Analogue, Digital, etc.")]
+ private AxisType axisType;
+
+ ///
+ /// The axis type of the button, e.g. Analog, Digital, etc.
+ ///
+ public AxisType AxisType => axisType;
+
+ [SerializeField]
+ [Tooltip("The primary action of the input as defined by the controller SDK.")]
+ private DeviceInputType inputType;
+
+ ///
+ /// The primary action of the input as defined by the controller SDK.
+ ///
+ public DeviceInputType InputType => inputType;
+
+ [SerializeField]
+ [Tooltip("Action to be raised to the Input Manager when the input data has changed.")]
+ private MixedRealityInputAction inputAction;
+
+ ///
+ /// Action to be raised when the input data has changed.
+ ///
+ public MixedRealityInputAction InputAction
+ {
+ get { return inputAction; }
+ internal set { inputAction = value; }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInputActionMapping.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInputActionMapping.cs.meta
new file mode 100644
index 0000000..a5354e4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInputActionMapping.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6003b5d53a7137d468aa55dcab8430a3
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMapping.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMapping.cs
new file mode 100644
index 0000000..5b7b33f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMapping.cs
@@ -0,0 +1,473 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Maps the capabilities of controllers, linking the physical inputs of a controller to a logical construct in a runtime project.
+ ///
+ ///
+ /// One definition should exist for each physical device input, such as buttons, triggers, joysticks, dpads, and more.
+ ///
+ [Serializable]
+ public class MixedRealityInteractionMapping
+ {
+ ///
+ /// The constructor for a new Interaction Mapping definition
+ ///
+ /// Identity for mapping
+ /// The description of the interaction mapping.
+ /// The axis that the mapping operates on, also denotes the data type for the mapping
+ /// The physical input device / control
+ /// Optional horizontal or single axis value to get axis data from Unity's old input system.
+ /// Optional vertical axis value to get axis data from Unity's old input system.
+ /// Optional horizontal axis invert option.
+ /// Optional vertical axis invert option.
+ public MixedRealityInteractionMapping(uint id, string description, AxisType axisType, DeviceInputType inputType, string axisCodeX = "", string axisCodeY = "", bool invertXAxis = false, bool invertYAxis = false)
+ : this(id, description, axisType, inputType, MixedRealityInputAction.None, KeyCode.None, axisCodeX, axisCodeY, invertXAxis, invertYAxis) { }
+
+ ///
+ /// The constructor for a new Interaction Mapping definition
+ ///
+ /// Identity for mapping
+ /// The description of the interaction mapping.
+ /// The axis that the mapping operates on, also denotes the data type for the mapping
+ /// The physical input device / control
+ /// Optional KeyCode value to get input from Unity's old input system
+ public MixedRealityInteractionMapping(uint id, string description, AxisType axisType, DeviceInputType inputType, KeyCode keyCode)
+ : this(id, description, axisType, inputType, MixedRealityInputAction.None, keyCode) { }
+
+ ///
+ /// The constructor for a new Interaction Mapping definition
+ ///
+ /// Identity for mapping
+ /// The description of the interaction mapping.
+ /// The axis that the mapping operates on, also denotes the data type for the mapping
+ /// The physical input device / control
+ /// The logical MixedRealityInputAction that this input performs
+ /// Optional KeyCode value to get input from Unity's old input system
+ /// Optional horizontal or single axis value to get axis data from Unity's old input system.
+ /// Optional vertical axis value to get axis data from Unity's old input system.
+ /// Optional horizontal axis invert option.
+ /// Optional vertical axis invert option.
+ public MixedRealityInteractionMapping(uint id, string description, AxisType axisType, DeviceInputType inputType, MixedRealityInputAction inputAction, KeyCode keyCode = KeyCode.None, string axisCodeX = "", string axisCodeY = "", bool invertXAxis = false, bool invertYAxis = false)
+ {
+ this.id = id;
+ this.description = description;
+ this.axisType = axisType;
+ this.inputType = inputType;
+ this.inputAction = inputAction;
+ this.keyCode = keyCode;
+ this.axisCodeX = axisCodeX;
+ this.axisCodeY = axisCodeY;
+ this.invertXAxis = invertXAxis;
+ this.invertYAxis = invertYAxis;
+
+ rawData = null;
+ boolData = false;
+ floatData = 0f;
+ vector2Data = Vector2.zero;
+ positionData = Vector3.zero;
+ rotationData = Quaternion.identity;
+ poseData = MixedRealityPose.ZeroIdentity;
+ changed = false;
+ }
+
+ public MixedRealityInteractionMapping(MixedRealityInteractionMapping mixedRealityInteractionMapping)
+ : this(mixedRealityInteractionMapping.id,
+ mixedRealityInteractionMapping.description,
+ mixedRealityInteractionMapping.axisType,
+ mixedRealityInteractionMapping.inputType,
+ mixedRealityInteractionMapping.inputAction,
+ mixedRealityInteractionMapping.keyCode,
+ mixedRealityInteractionMapping.axisCodeX,
+ mixedRealityInteractionMapping.axisCodeY,
+ mixedRealityInteractionMapping.invertXAxis,
+ mixedRealityInteractionMapping.invertYAxis) { }
+
+ public MixedRealityInteractionMapping(MixedRealityInteractionMapping mixedRealityInteractionMapping, MixedRealityInteractionMappingLegacyInput legacyInput)
+ : this(mixedRealityInteractionMapping.id,
+ mixedRealityInteractionMapping.description,
+ mixedRealityInteractionMapping.axisType,
+ mixedRealityInteractionMapping.inputType,
+ mixedRealityInteractionMapping.inputAction,
+ legacyInput.KeyCode,
+ legacyInput.AxisCodeX,
+ legacyInput.AxisCodeY,
+ legacyInput.InvertXAxis,
+ legacyInput.InvertYAxis) { }
+
+ public MixedRealityInteractionMapping(uint id, MixedRealityInputActionMapping mixedRealityInputActionMapping)
+ : this(id,
+ mixedRealityInputActionMapping.Description,
+ mixedRealityInputActionMapping.AxisType,
+ mixedRealityInputActionMapping.InputType,
+ mixedRealityInputActionMapping.InputAction) { }
+
+ public MixedRealityInteractionMapping(uint id, MixedRealityInputActionMapping mixedRealityInputActionMapping, MixedRealityInteractionMappingLegacyInput legacyInput)
+ : this(id,
+ mixedRealityInputActionMapping.Description,
+ mixedRealityInputActionMapping.AxisType,
+ mixedRealityInputActionMapping.InputType,
+ mixedRealityInputActionMapping.InputAction,
+ legacyInput.KeyCode,
+ legacyInput.AxisCodeX ?? string.Empty, // defaults to null in the struct, but Unity serializes as empty string
+ legacyInput.AxisCodeY ?? string.Empty, // defaults to null in the struct, but Unity serializes as empty string
+ legacyInput.InvertXAxis,
+ legacyInput.InvertYAxis) { }
+
+ #region Interaction Properties
+
+ [SerializeField]
+ [Tooltip("The Id assigned to the Interaction.")]
+ private uint id;
+
+ ///
+ /// The Id assigned to the Interaction.
+ ///
+ public uint Id => id;
+
+ [SerializeField]
+ [Tooltip("The description of the interaction mapping.")]
+ private string description;
+
+ ///
+ /// The description of the interaction mapping.
+ ///
+ public string Description => description;
+
+ [SerializeField]
+ [Tooltip("The axis type of the button, e.g. Analogue, Digital, etc.")]
+ private AxisType axisType;
+
+ ///
+ /// The axis type of the button, e.g. Analogue, Digital, etc.
+ ///
+ public AxisType AxisType => axisType;
+
+ [SerializeField]
+ [Tooltip("The primary action of the input as defined by the controller SDK.")]
+ private DeviceInputType inputType;
+
+ ///
+ /// The primary action of the input as defined by the controller SDK.
+ ///
+ public DeviceInputType InputType => inputType;
+
+ [SerializeField]
+ [Tooltip("Action to be raised to the Input Manager when the input data has changed.")]
+ private MixedRealityInputAction inputAction;
+
+ ///
+ /// Action to be raised to the Input Manager when the input data has changed.
+ ///
+ public MixedRealityInputAction MixedRealityInputAction
+ {
+ get { return inputAction; }
+ internal set { inputAction = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Optional KeyCode value to get input from Unity's old input system.")]
+ private KeyCode keyCode;
+
+ ///
+ /// Optional KeyCode value to get input from Unity's old input system.
+ ///
+ public KeyCode KeyCode => keyCode;
+
+ [SerializeField]
+ [Tooltip("Optional horizontal or single axis value to get axis data from Unity's old input system.")]
+ private string axisCodeX;
+
+ ///
+ /// Optional horizontal or single axis value to get axis data from Unity's old input system.
+ ///
+ public string AxisCodeX => axisCodeX;
+
+ [SerializeField]
+ [Tooltip("Optional vertical axis value to get axis data from Unity's old input system.")]
+ private string axisCodeY;
+
+ ///
+ /// Optional vertical axis value to get axis data from Unity's old input system.
+ ///
+ public string AxisCodeY => axisCodeY;
+
+ [SerializeField]
+ [Tooltip("Should the X axis be inverted?")]
+ private bool invertXAxis = false;
+
+ ///
+ /// Should the X axis be inverted?
+ ///
+ ///
+ /// Only valid for and inputs.
+ ///
+ public bool InvertXAxis
+ {
+ get { return invertXAxis; }
+ set
+ {
+ if (AxisType != AxisType.SingleAxis && AxisType != AxisType.DualAxis)
+ {
+ Debug.LogWarning("Inverted X axis only valid for Single or Dual Axis inputs.");
+ return;
+ }
+
+ invertXAxis = value;
+ }
+ }
+
+ [SerializeField]
+ [Tooltip("Should the Y axis be inverted?")]
+ private bool invertYAxis = false;
+
+ ///
+ /// Should the Y axis be inverted?
+ ///
+ ///
+ /// Only valid for inputs.
+ ///
+ public bool InvertYAxis
+ {
+ get { return invertYAxis; }
+ set
+ {
+ if (AxisType != AxisType.DualAxis)
+ {
+ Debug.LogWarning("Inverted Y axis only valid for Dual Axis inputs.");
+ return;
+ }
+
+ invertYAxis = value;
+ }
+ }
+
+ private bool changed;
+
+ ///
+ /// Has the value changed since the last reading.
+ ///
+ public bool Changed
+ {
+ get
+ {
+ bool returnValue = changed;
+
+ if (changed)
+ {
+ changed = false;
+ }
+
+ return returnValue;
+ }
+ private set
+ {
+ changed = value;
+ }
+ }
+
+ #endregion Interaction Properties
+
+ #region Definition Data Items
+
+ private object rawData;
+
+ private bool boolData;
+
+ private float floatData;
+
+ private Vector2 vector2Data;
+
+ private Vector3 positionData;
+
+ private Quaternion rotationData;
+
+ private MixedRealityPose poseData;
+
+ #endregion Definition Data Items
+
+ #region Data Properties
+
+ ///
+ /// The Raw (object) data value.
+ ///
+ /// Only supported for a Raw mapping axis type
+ public object RawData
+ {
+ get
+ {
+ return rawData;
+ }
+
+ set
+ {
+ if (AxisType != AxisType.Raw)
+ {
+ Debug.LogError($"SetRawValue is only valid for AxisType.Raw InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+
+ Changed = rawData != value;
+ rawData = value;
+ }
+ }
+
+ ///
+ /// The Bool data value.
+ ///
+ /// Only supported for a Digital mapping axis type
+ public bool BoolData
+ {
+ get
+ {
+ return boolData;
+ }
+
+ set
+ {
+ if (AxisType != AxisType.Digital && AxisType != AxisType.SingleAxis && AxisType != AxisType.DualAxis)
+ {
+ Debug.LogError($"SetBoolValue is only valid for AxisType.Digital, AxisType.SingleAxis, or AxisType.DualAxis InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+
+ Changed = boolData != value;
+ boolData = value;
+ }
+ }
+
+ ///
+ /// The Float data value.
+ ///
+ /// Only supported for a SingleAxis mapping axis type
+ public float FloatData
+ {
+ get
+ {
+ return floatData;
+ }
+
+ set
+ {
+ if (AxisType != AxisType.SingleAxis)
+ {
+ Debug.LogError($"SetFloatValue is only valid for AxisType.SingleAxis InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+
+ if (invertXAxis)
+ {
+ Changed = !floatData.Equals(value * -1f);
+ floatData = value * -1f;
+ }
+ else
+ {
+ Changed = !floatData.Equals(value);
+ floatData = value;
+ }
+ }
+ }
+
+ ///
+ /// The Vector2 data value.
+ ///
+ /// Only supported for a DualAxis mapping axis type
+ public Vector2 Vector2Data
+ {
+ get
+ {
+ return vector2Data;
+ }
+
+ set
+ {
+ if (AxisType != AxisType.DualAxis)
+ {
+ Debug.LogError($"SetVector2Value is only valid for AxisType.DualAxis InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+
+ Vector2 newValue = value * new Vector2(invertXAxis ? -1f : 1f, invertYAxis ? -1f : 1f);
+ Changed = vector2Data != newValue;
+ vector2Data = newValue;
+ }
+ }
+
+ ///
+ /// The ThreeDoF Vector3 Position data value.
+ ///
+ /// Only supported for a ThreeDoF mapping axis type
+ public Vector3 PositionData
+ {
+ get
+ {
+ return positionData;
+ }
+
+ set
+ {
+ if (AxisType != AxisType.ThreeDofPosition)
+ {
+ {
+ Debug.LogError($"SetPositionValue is only valid for AxisType.ThreeDoFPosition InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+ }
+
+ Changed = positionData != value;
+ positionData = value;
+ }
+ }
+
+ ///
+ /// The ThreeDoF Quaternion Rotation data value.
+ ///
+ /// Only supported for a ThreeDoF mapping axis type
+ public Quaternion RotationData
+ {
+ get
+ {
+ return rotationData;
+ }
+
+ set
+ {
+ if (AxisType != AxisType.ThreeDofRotation)
+ {
+ Debug.LogError($"SetRotationValue is only valid for AxisType.ThreeDoFRotation InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+
+ Changed = rotationData != value;
+ rotationData = value;
+ }
+ }
+
+ ///
+ /// The Pose data value.
+ ///
+ /// Only supported for a SixDoF mapping axis type
+ public MixedRealityPose PoseData
+ {
+ get
+ {
+ return poseData;
+ }
+ set
+ {
+ if (AxisType != AxisType.SixDof)
+ {
+ Debug.LogError($"SetPoseValue is only valid for AxisType.SixDoF InteractionMappings\nPlease check the {InputType} mapping for the current controller");
+ }
+
+ Changed = poseData != value;
+
+ poseData = value;
+ positionData = poseData.Position;
+ rotationData = poseData.Rotation;
+ }
+ }
+
+ #endregion Data Properties
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMapping.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMapping.cs.meta
new file mode 100644
index 0000000..2766a16
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMapping.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 0e42e1b320254879911ebf94fc274939
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMappingLegacyInput.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMappingLegacyInput.cs
new file mode 100644
index 0000000..f6c1e64
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMappingLegacyInput.cs
@@ -0,0 +1,47 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Represents the subset of data held by a that represents Unity's legacy input system.
+ ///
+ public struct MixedRealityInteractionMappingLegacyInput
+ {
+ ///
+ /// Optional KeyCode value to get input from Unity's old input system.
+ ///
+ public KeyCode KeyCode { get; }
+
+ ///
+ /// Optional horizontal or single axis value to get axis data from Unity's old input system.
+ ///
+ public string AxisCodeX { get; }
+
+ ///
+ /// Optional vertical axis value to get axis data from Unity's old input system.
+ ///
+ public string AxisCodeY { get; }
+
+ ///
+ /// Should the X axis be inverted?
+ ///
+ public bool InvertXAxis { get; }
+
+ ///
+ /// Should the Y axis be inverted?
+ ///
+ public bool InvertYAxis { get; }
+
+ public MixedRealityInteractionMappingLegacyInput(KeyCode keyCode = KeyCode.None, string axisCodeX = "", string axisCodeY = "", bool invertXAxis = false, bool invertYAxis = false)
+ {
+ KeyCode = keyCode;
+ AxisCodeX = axisCodeX;
+ AxisCodeY = axisCodeY;
+ InvertXAxis = invertXAxis;
+ InvertYAxis = invertYAxis;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMappingLegacyInput.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMappingLegacyInput.cs.meta
new file mode 100644
index 0000000..2f1e217
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MixedRealityInteractionMappingLegacyInput.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 864201fd5b2bf9541a37abad6cbe2f89
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MouseControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MouseControllerDefinition.cs
new file mode 100644
index 0000000..e6c3412
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MouseControllerDefinition.cs
@@ -0,0 +1,34 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Defines the base interactions and data that an controller can provide.
+ ///
+ public class MouseControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ public MouseControllerDefinition() : base(Handedness.None)
+ { }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Mouse Position", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Mouse Delta Position", AxisType.DualAxis, DeviceInputType.PointerPosition),
+ new MixedRealityInputActionMapping("Mouse Scroll Position", AxisType.DualAxis, DeviceInputType.Scroll),
+ new MixedRealityInputActionMapping("Left Mouse Button", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Right Mouse Button", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Mouse Button 2", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Mouse Button 3", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Mouse Button 4", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Mouse Button 5", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Mouse Button 6", AxisType.Digital, DeviceInputType.ButtonPress),
+ };
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MouseControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MouseControllerDefinition.cs.meta
new file mode 100644
index 0000000..48b8f87
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/MouseControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1b8150a673821864e9c89cc56e6b6a6a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusRemoteControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusRemoteControllerDefinition.cs
new file mode 100644
index 0000000..757c31a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusRemoteControllerDefinition.cs
@@ -0,0 +1,27 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class OculusRemoteControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ public OculusRemoteControllerDefinition() : base(Handedness.None)
+ { }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("D-Pad Position", AxisType.DualAxis, DeviceInputType.DirectionalPad),
+ new MixedRealityInputActionMapping("Button.One", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Button.Two", AxisType.Digital, DeviceInputType.ButtonPress),
+ };
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusRemoteControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusRemoteControllerDefinition.cs.meta
new file mode 100644
index 0000000..a863df4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusRemoteControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 0cf2f1a42159a32488de6d0666949ee6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusTouchControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusTouchControllerDefinition.cs
new file mode 100644
index 0000000..2de6eab
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusTouchControllerDefinition.cs
@@ -0,0 +1,69 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class OculusTouchControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public OculusTouchControllerDefinition(Handedness handedness) : base(handedness)
+ {
+ if ((handedness != Handedness.Left) &&
+ (handedness != Handedness.Right))
+ {
+ throw new System.ArgumentException($"Unsupported Handedness ({handedness}). The OculusTouchControllerDefinition supports Left and Right.");
+ }
+ }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultLeftHandedMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Axis1D.PrimaryIndexTrigger", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Axis1D.PrimaryIndexTrigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Axis1D.PrimaryIndexTrigger Near Touch", AxisType.Digital, DeviceInputType.TriggerNearTouch),
+ new MixedRealityInputActionMapping("Axis1D.PrimaryIndexTrigger Press", AxisType.Digital, DeviceInputType.TriggerPress),
+ new MixedRealityInputActionMapping("Axis1D.PrimaryHandTrigger Press", AxisType.SingleAxis, DeviceInputType.GripPress),
+ new MixedRealityInputActionMapping("Axis2D.PrimaryThumbstick", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Button.PrimaryThumbstick Touch", AxisType.Digital, DeviceInputType.ThumbStickTouch),
+ new MixedRealityInputActionMapping("Button.PrimaryThumbstick Near Touch", AxisType.Digital, DeviceInputType.ThumbNearTouch),
+ new MixedRealityInputActionMapping("Button.PrimaryThumbstick Press", AxisType.Digital, DeviceInputType.ThumbStickPress),
+ new MixedRealityInputActionMapping("Button.Three Press", AxisType.Digital, DeviceInputType.PrimaryButtonPress),
+ new MixedRealityInputActionMapping("Button.Four Press", AxisType.Digital, DeviceInputType.SecondaryButtonPress),
+ new MixedRealityInputActionMapping("Button.Start Press", AxisType.Digital, DeviceInputType.Menu),
+ new MixedRealityInputActionMapping("Button.Three Touch", AxisType.Digital, DeviceInputType.PrimaryButtonTouch),
+ new MixedRealityInputActionMapping("Button.Four Touch", AxisType.Digital, DeviceInputType.SecondaryButtonTouch),
+ new MixedRealityInputActionMapping("Touch.PrimaryThumbRest Touch", AxisType.Digital, DeviceInputType.ThumbTouch),
+ new MixedRealityInputActionMapping("Touch.PrimaryThumbRest Near Touch", AxisType.Digital, DeviceInputType.ThumbNearTouch),
+ };
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultRightHandedMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Axis1D.SecondaryIndexTrigger", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Axis1D.SecondaryIndexTrigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Axis1D.SecondaryIndexTrigger Near Touch", AxisType.Digital, DeviceInputType.TriggerNearTouch),
+ new MixedRealityInputActionMapping("Axis1D.SecondaryIndexTrigger Press", AxisType.Digital, DeviceInputType.TriggerPress),
+ new MixedRealityInputActionMapping("Axis1D.SecondaryHandTrigger Press", AxisType.SingleAxis, DeviceInputType.GripPress),
+ new MixedRealityInputActionMapping("Axis2D.SecondaryThumbstick", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Button.SecondaryThumbstick Touch", AxisType.Digital, DeviceInputType.ThumbStickTouch),
+ new MixedRealityInputActionMapping("Button.SecondaryThumbstick Near Touch", AxisType.Digital, DeviceInputType.ThumbNearTouch),
+ new MixedRealityInputActionMapping("Button.SecondaryThumbstick Press", AxisType.Digital, DeviceInputType.ThumbStickPress),
+ new MixedRealityInputActionMapping("Button.One Press", AxisType.Digital, DeviceInputType.PrimaryButtonPress),
+ new MixedRealityInputActionMapping("Button.Two Press", AxisType.Digital, DeviceInputType.SecondaryButtonPress),
+ new MixedRealityInputActionMapping("Button.One Touch", AxisType.Digital, DeviceInputType.PrimaryButtonTouch),
+ new MixedRealityInputActionMapping("Button.Two Touch", AxisType.Digital, DeviceInputType.SecondaryButtonTouch),
+ new MixedRealityInputActionMapping("Touch.SecondaryThumbRest Touch", AxisType.Digital, DeviceInputType.ThumbTouch),
+ new MixedRealityInputActionMapping("Touch.SecondaryThumbRest Near Touch", AxisType.Digital, DeviceInputType.ThumbNearTouch),
+ };
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusTouchControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusTouchControllerDefinition.cs.meta
new file mode 100644
index 0000000..8b3475a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/OculusTouchControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9b88f64d248ddae419217f60f27f9328
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SDKType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SDKType.cs
new file mode 100644
index 0000000..be181f8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SDKType.cs
@@ -0,0 +1,33 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// The SDKType lists the XR SDKs that are supported by the Mixed Reality Toolkit.
+ /// Initially, this lists proposed SDKs, not all may be implemented at this time (please see ReleaseNotes for more details)
+ ///
+ public enum SDKType
+ {
+ ///
+ /// No specified type or Standalone / non-XR type
+ ///
+ None = 0,
+ ///
+ /// Undefined SDK.
+ ///
+ Other,
+ ///
+ /// The Windows 10 Mixed reality SDK provided by the Universal Windows Platform (UWP), for Immersive MR headsets and HoloLens.
+ ///
+ WindowsMR,
+ ///
+ /// The OpenVR platform provided by Unity (does not support the downloadable SteamVR SDK).
+ ///
+ OpenVR,
+ ///
+ /// The OpenXR platform. SDK to be determined once released.
+ ///
+ OpenXR
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SDKType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SDKType.cs.meta
new file mode 100644
index 0000000..cd6006e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SDKType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 958811dd1ef749ffa71f909abe458eb2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SimpleHandDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SimpleHandDefinition.cs
new file mode 100644
index 0000000..9cbc38b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SimpleHandDefinition.cs
@@ -0,0 +1,27 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class SimpleHandDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public SimpleHandDefinition(Handedness handedness) : base(handedness)
+ { }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Select", AxisType.Digital, DeviceInputType.Select),
+ new MixedRealityInputActionMapping("Grip Pose", AxisType.SixDof, DeviceInputType.SpatialGrip),
+ };
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SimpleHandDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SimpleHandDefinition.cs.meta
new file mode 100644
index 0000000..c57bed2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SimpleHandDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 63bc40c4664e08f498eee5aa4fa43b11
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SupportedControllerType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SupportedControllerType.cs
new file mode 100644
index 0000000..8bddeb8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SupportedControllerType.cs
@@ -0,0 +1,50 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ // todo: remove this.... it requires customization to add new device types
+
+ ///
+ /// The SDKType lists the XR SDKs that are supported by the Mixed Reality Toolkit.
+ /// Initially, this lists proposed SDKs, not all may be implemented at this time (please see ReleaseNotes for more details)
+ ///
+ [Flags]
+ public enum SupportedControllerType
+ {
+ GenericOpenVR = 1 << 0,
+ ViveWand = 1 << 1,
+ ViveKnuckles = 1 << 2,
+ OculusTouch = 1 << 3,
+ OculusRemote = 1 << 4,
+ WindowsMixedReality = 1 << 5,
+ GenericUnity = 1 << 6,
+ Xbox = 1 << 7,
+ TouchScreen = 1 << 8,
+ Mouse = 1 << 9,
+ ArticulatedHand = 1 << 10,
+ GGVHand = 1 << 11,
+ HPMotionController = 1 << 12
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class SupportedControllerTypeExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this SupportedControllerType a, SupportedControllerType b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SupportedControllerType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SupportedControllerType.cs.meta
new file mode 100644
index 0000000..07b18af
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/SupportedControllerType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4d7b06c23db4419b90fe7d8192856e91
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TouchScreenDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TouchScreenDefinition.cs
new file mode 100644
index 0000000..2d7a461
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TouchScreenDefinition.cs
@@ -0,0 +1,27 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class TouchScreenDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ public TouchScreenDefinition() : base(Handedness.None)
+ { }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Touch Pointer Delta", AxisType.DualAxis, DeviceInputType.PointerPosition),
+ new MixedRealityInputActionMapping("Touch Pointer Position", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Touch Press", AxisType.Digital, DeviceInputType.PointerClick),
+ };
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TouchScreenDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TouchScreenDefinition.cs.meta
new file mode 100644
index 0000000..8fff1fa
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TouchScreenDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 830dc9478b0633a4a8cd5d1d4a6f2756
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TrackingState.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TrackingState.cs
new file mode 100644
index 0000000..71fe0c0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TrackingState.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// The Tracking State defines how a device is currently being tracked.
+ /// This enables developers to be able to handle non-tracked situations and react accordingly.
+ ///
+ ///
+ /// Tracking is being defined as receiving sensor (positional and/or rotational) data from the device.
+ ///
+ public enum TrackingState
+ {
+ ///
+ /// The device does not support tracking (ex: a traditional game controller).
+ ///
+ NotApplicable = 0,
+ ///
+ /// The device is not tracked.
+ ///
+ NotTracked,
+ ///
+ /// The device is tracked (positionally and/or rotationally).
+ ///
+ ///
+ /// Some devices provide additional details regarding the accuracy of the tracking.
+ ///
+ Tracked
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TrackingState.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TrackingState.cs.meta
new file mode 100644
index 0000000..b14e1a5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/TrackingState.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1866a0c9bb60461ba5c3d76f75951794
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveKnucklesControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveKnucklesControllerDefinition.cs
new file mode 100644
index 0000000..1b83d04
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveKnucklesControllerDefinition.cs
@@ -0,0 +1,45 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class ViveKnucklesControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public ViveKnucklesControllerDefinition(Handedness handedness) : base(handedness)
+ {
+ if ((handedness != Handedness.Left) &&
+ (handedness != Handedness.Right))
+ {
+ throw new System.ArgumentException($"Unsupported Handedness ({handedness}). The ViveKnucklesControllerDefinition supports Left and Right.");
+ }
+ }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Grip Average", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trackpad Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ new MixedRealityInputActionMapping("Trackpad Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ new MixedRealityInputActionMapping("Trackpad Press", AxisType.Digital, DeviceInputType.TouchpadPress),
+ new MixedRealityInputActionMapping("Inner Face Button", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Outer Face Button", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Index Finger Cap Sensor", AxisType.SingleAxis, DeviceInputType.IndexFinger),
+ new MixedRealityInputActionMapping("Middle Finger Cap Sensor", AxisType.SingleAxis, DeviceInputType.MiddleFinger),
+ new MixedRealityInputActionMapping("Ring Finger Cap Sensor", AxisType.SingleAxis, DeviceInputType.RingFinger),
+ new MixedRealityInputActionMapping("Pinky Finger Cap Sensor", AxisType.SingleAxis, DeviceInputType.PinkyFinger),
+ };
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveKnucklesControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveKnucklesControllerDefinition.cs.meta
new file mode 100644
index 0000000..29513cc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveKnucklesControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: cd20a4c08ffc3264b96fa3a9e3a2a40c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveWandControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveWandControllerDefinition.cs
new file mode 100644
index 0000000..cd08907
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveWandControllerDefinition.cs
@@ -0,0 +1,40 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ ///
+ ///
+ public class ViveWandControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition instance represents.
+ public ViveWandControllerDefinition(Handedness handedness) : base(handedness)
+ {
+ if ((handedness != Handedness.Left) &&
+ (handedness != Handedness.Right))
+ {
+ throw new System.ArgumentException($"Unsupported Handedness ({handedness}). The ViveWandControllerDefinition supports Left and Right.");
+ }
+ }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Grip Press", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trackpad Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ new MixedRealityInputActionMapping("Trackpad Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ new MixedRealityInputActionMapping("Trackpad Press", AxisType.Digital, DeviceInputType.TouchpadPress),
+ new MixedRealityInputActionMapping("Menu Button", AxisType.Digital, DeviceInputType.ButtonPress),
+ };
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveWandControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveWandControllerDefinition.cs.meta
new file mode 100644
index 0000000..da1d699
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/ViveWandControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: cd2391a723b4eff4ea76d454c103e664
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/WindowsMixedRealityControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/WindowsMixedRealityControllerDefinition.cs
new file mode 100644
index 0000000..777e1bd
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/WindowsMixedRealityControllerDefinition.cs
@@ -0,0 +1,43 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Defines the interactions and data that a Windows Mixed Reality motion controller can provide.
+ ///
+ public class WindowsMixedRealityControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The handedness that this definition represents.
+ public WindowsMixedRealityControllerDefinition(Handedness handedness) : base(handedness)
+ {
+ if ((handedness != Handedness.Left) &&
+ (handedness != Handedness.Right))
+ {
+ throw new System.ArgumentException($"Unsupported Handedness ({handedness}). The ViveWandControllerDefinition supports Left and Right.");
+ }
+ }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Spatial Pointer", AxisType.SixDof, DeviceInputType.SpatialPointer),
+ new MixedRealityInputActionMapping("Spatial Grip", AxisType.SixDof, DeviceInputType.SpatialGrip),
+ new MixedRealityInputActionMapping("Grip Press", AxisType.SingleAxis, DeviceInputType.GripPress),
+ new MixedRealityInputActionMapping("Trigger Position", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Trigger Touch", AxisType.Digital, DeviceInputType.TriggerTouch),
+ new MixedRealityInputActionMapping("Trigger Press (Select)", AxisType.Digital, DeviceInputType.Select),
+ new MixedRealityInputActionMapping("Touchpad Position", AxisType.DualAxis, DeviceInputType.Touchpad),
+ new MixedRealityInputActionMapping("Touchpad Touch", AxisType.Digital, DeviceInputType.TouchpadTouch),
+ new MixedRealityInputActionMapping("Touchpad Press", AxisType.Digital, DeviceInputType.TouchpadPress),
+ new MixedRealityInputActionMapping("Menu Press", AxisType.Digital, DeviceInputType.Menu),
+ new MixedRealityInputActionMapping("Thumbstick Position", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Thumbstick Press", AxisType.Digital, DeviceInputType.ThumbStickPress),
+ };
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/WindowsMixedRealityControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/WindowsMixedRealityControllerDefinition.cs.meta
new file mode 100644
index 0000000..b78a8fe
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/WindowsMixedRealityControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2ecca80a54b0b59498e6c17e93b7719f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/XboxControllerDefinition.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/XboxControllerDefinition.cs
new file mode 100644
index 0000000..7e02bc5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/XboxControllerDefinition.cs
@@ -0,0 +1,40 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Defines the base interactions and data that an controller can provide.
+ ///
+ public class XboxControllerDefinition : BaseInputSourceDefinition
+ {
+ ///
+ /// Constructor.
+ ///
+ public XboxControllerDefinition() : base(Handedness.None)
+ { }
+
+ ///
+ protected override MixedRealityInputActionMapping[] DefaultMappings => new[]
+ {
+ new MixedRealityInputActionMapping("Left Thumbstick", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Left Thumbstick Click", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Right Thumbstick", AxisType.DualAxis, DeviceInputType.ThumbStick),
+ new MixedRealityInputActionMapping("Right Thumbstick Click", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("D-Pad", AxisType.DualAxis, DeviceInputType.DirectionalPad),
+ new MixedRealityInputActionMapping("Shared Trigger", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Left Trigger", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("Right Trigger", AxisType.SingleAxis, DeviceInputType.Trigger),
+ new MixedRealityInputActionMapping("View", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Menu", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Left Bumper", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Right Bumper", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("A", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("B", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("X", AxisType.Digital, DeviceInputType.ButtonPress),
+ new MixedRealityInputActionMapping("Y", AxisType.Digital, DeviceInputType.ButtonPress),
+ };
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/XboxControllerDefinition.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/XboxControllerDefinition.cs.meta
new file mode 100644
index 0000000..f6c7d90
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Devices/XboxControllerDefinition.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 37c5e65bd36b61942ae89bf53c24ff91
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics.meta
new file mode 100644
index 0000000..384f91c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: e2cc991907254864d8d1847b3b5b2854
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics/MixedRealityDiagnosticsProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics/MixedRealityDiagnosticsProfile.cs
new file mode 100644
index 0000000..e79c3fc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics/MixedRealityDiagnosticsProfile.cs
@@ -0,0 +1,122 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+using UnityEngine.Serialization;
+
+namespace Microsoft.MixedReality.Toolkit.Diagnostics
+{
+ ///
+ /// Configuration profile settings for setting up diagnostics.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Diagnostics Profile", fileName = "MixedRealityDiagnosticsProfile", order = (int)CreateProfileMenuItemIndices.Diagnostics)]
+ [MixedRealityServiceProfile(typeof(IMixedRealityDiagnosticsSystem))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/diagnostics/diagnostics-system-getting-started")]
+ public class MixedRealityDiagnosticsProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [FormerlySerializedAs("visible")]
+ [Tooltip("Display all enabled diagnostics")]
+ private bool showDiagnostics = true;
+
+ ///
+ /// Show or hide diagnostic visualizations.
+ ///
+ public bool ShowDiagnostics => showDiagnostics;
+
+ [SerializeField]
+ [Tooltip("Display profiler")]
+ private bool showProfiler = true;
+
+ ///
+ /// Show or hide the profiler UI.
+ ///
+ public bool ShowProfiler => showProfiler;
+
+ [SerializeField]
+ [Tooltip("Display the frame info (per frame stats).")]
+ private bool showFrameInfo = true;
+
+ ///
+ /// Show or hide the frame info (per frame stats).
+ ///
+ public bool ShowFrameInfo => showFrameInfo;
+
+ [SerializeField]
+ [Tooltip("Display the memory stats (used, peak, and limit).")]
+ private bool showMemoryStats = true;
+
+ ///
+ /// Show or hide the memory stats (used, peak, and limit).
+ ///
+ public bool ShowMemoryStats => showMemoryStats;
+
+ [SerializeField]
+ [FormerlySerializedAs("frameRateDuration")]
+ [Tooltip("The amount of time, in seconds, to collect frames for frame rate calculation.")]
+ [Range(0, 5)]
+ private float frameSampleRate = 0.1f;
+
+ ///
+ /// The amount of time, in seconds, to collect frames for frame rate calculation.
+ ///
+ public float FrameSampleRate => frameSampleRate;
+
+ [SerializeField]
+ [Tooltip("What part of the view port to anchor the window to.")]
+ private TextAnchor windowAnchor = TextAnchor.LowerCenter;
+
+ ///
+ /// What part of the view port to anchor the window to.
+ ///
+ public TextAnchor WindowAnchor => windowAnchor;
+
+ [SerializeField]
+ [Tooltip("The offset from the view port center applied based on the window anchor selection.")]
+ private Vector2 windowOffset = new Vector2(0.1f, 0.1f);
+
+ ///
+ /// The offset from the view port center applied based on the window anchor selection.
+ ///
+ public Vector2 WindowOffset => windowOffset;
+
+ [SerializeField]
+ [Tooltip("Use to scale the window size up or down, can simulate a zooming effect.")]
+ private float windowScale = 1.0f;
+
+ ///
+ /// Use to scale the window size up or down, can simulate a zooming effect.
+ ///
+ public float WindowScale => windowScale;
+
+ [SerializeField]
+ [Tooltip("How quickly to interpolate the window towards its target position and rotation.")]
+ private float windowFollowSpeed = 5.0f;
+
+ ///
+ /// How quickly to interpolate the window towards its target position and rotation.
+ ///
+ public float WindowFollowSpeed => windowFollowSpeed;
+
+ [SerializeField]
+ [Tooltip("A material that the diagnostics system can use to render objects with instanced color support.")]
+ private Material defaultInstancedMaterial = null;
+
+ ///
+ /// A material that the diagnostics system can use to render objects with instanced color support.
+ /// A asset reference is required here to make sure the shader permutation is pulled into player builds.
+ ///
+ public Material DefaultInstancedMaterial => defaultInstancedMaterial;
+
+ [SerializeField]
+ [Tooltip("If the diagnostics profiler should be visible while a mixed reality capture is happening on HoloLens.")]
+ private bool showProfilerDuringMRC = false;
+
+ ///
+ /// If the diagnostics profiler should be visible while a mixed reality capture is happening on HoloLens.
+ ///
+ /// This is not usually recommended, as MRC can have an effect on an app's frame rate.
+ public bool ShowProfilerDuringMRC => showProfilerDuringMRC;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics/MixedRealityDiagnosticsProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics/MixedRealityDiagnosticsProfile.cs.meta
new file mode 100644
index 0000000..459ac42
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Diagnostics/MixedRealityDiagnosticsProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8c2d00f2d26cc124caed106ffbfe3f06
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem.meta
new file mode 100644
index 0000000..e3d51b4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 1f295446ce9f4343b200ce4d0855d3a6
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/AnimatedCursorData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/AnimatedCursorData.cs
new file mode 100644
index 0000000..ff62234
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/AnimatedCursorData.cs
@@ -0,0 +1,51 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ [Serializable]
+ public class AnimatedCursorStateData : AnimatedCursorData { }
+
+ [Serializable]
+ public class AnimatedCursorContextData : AnimatedCursorData { }
+
+ ///
+ /// Data struct for cursor state information for the Animated Cursor, which leverages the Unity animation system.
+ /// This defines a modification to an Unity animation parameter, based on cursor state.
+ ///
+ [Serializable]
+ public class AnimatedCursorData
+ {
+
+ [SerializeField]
+ [Tooltip("The name of this specific cursor state.")]
+ protected string name;
+
+ ///
+ /// The name of this specific cursor state.
+ ///
+ public string Name => name;
+
+ [SerializeField]
+ [Tooltip("The Cursor State for this specific animation.")]
+ protected T cursorState;
+
+ ///
+ /// The Cursor State for this specific animation.
+ ///
+ public T CursorState => cursorState;
+
+ [SerializeField]
+ [Tooltip("Animator parameter definition for this cursor state.")]
+ protected AnimatorParameter parameter;
+
+ ///
+ /// Animator parameter definition for this cursor state.
+ ///
+ public AnimatorParameter Parameter => parameter;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/AnimatedCursorData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/AnimatedCursorData.cs.meta
new file mode 100644
index 0000000..4a4d274
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/AnimatedCursorData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b32a2d63c851467ca803851b4bfc2b24
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorContextEnum.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorContextEnum.cs
new file mode 100644
index 0000000..392b443
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorContextEnum.cs
@@ -0,0 +1,21 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Enum for current cursor context
+ ///
+ public enum CursorContextEnum
+ {
+ None = -1,
+ MoveEastWest,
+ MoveNorthSouth,
+ MoveNorthwestSoutheast,
+ MoveNortheastSouthwest,
+ MoveCross,
+ RotateEastWest,
+ RotateNorthSouth,
+ Contextual
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorContextEnum.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorContextEnum.cs.meta
new file mode 100644
index 0000000..9d11185
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorContextEnum.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d3d0769d312b177439f3415aad3486fa
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorStateEnum.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorStateEnum.cs
new file mode 100644
index 0000000..e4f1fbe
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorStateEnum.cs
@@ -0,0 +1,46 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Enum for current cursor state
+ ///
+ public enum CursorStateEnum
+ {
+ ///
+ /// Useful for releasing external override.
+ /// See CursorStateEnum.Contextual
+ ///
+ None = -1,
+ ///
+ /// Not IsHandDetected OR HasTeleportIntent
+ ///
+ Observe,
+ ///
+ /// Not IsHandDetected AND not IsPointerDown AND TargetedObject exists OR HasTeleportIntent AND Teleport Surface IsValid
+ ///
+ ObserveHover,
+ ///
+ /// IsHandDetected AND not IsPointerDown AND TargetedObject is NULL
+ ///
+ Interact,
+ ///
+ /// IsHandDetected AND not IsPointerDown AND TargetedObject exists
+ ///
+ InteractHover,
+ ///
+ /// IsHandDetected AND IsPointerDown
+ ///
+ Select,
+ ///
+ /// Available for use by classes that extend Cursor.
+ /// No logic for setting Release state exists in the base Cursor class.
+ ///
+ Release,
+ ///
+ /// Allows for external override
+ ///
+ Contextual
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorStateEnum.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorStateEnum.cs.meta
new file mode 100644
index 0000000..f884a80
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/CursorStateEnum.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 25bc355c777d4df790f9455a93d6333a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionEventPair.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionEventPair.cs
new file mode 100644
index 0000000..cef640c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionEventPair.cs
@@ -0,0 +1,43 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+using UnityEngine.Events;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Data class that maps s to s wired up in the inspector.
+ ///
+ [Serializable]
+ public struct InputActionEventPair
+ {
+ ///
+ /// Constructor.
+ ///
+ public InputActionEventPair(MixedRealityInputAction inputAction, UnityEvent unityEvent)
+ {
+ this.inputAction = inputAction;
+ this.unityEvent = unityEvent;
+ }
+
+ [SerializeField]
+ [Tooltip("The MixedRealityInputAction to listen for to invoke the UnityEvent.")]
+ private MixedRealityInputAction inputAction;
+
+ ///
+ /// The to listen for to invoke the .
+ ///
+ public MixedRealityInputAction InputAction => inputAction;
+
+ [SerializeField]
+ [Tooltip("The UnityEvent to invoke when MixedRealityInputAction is raised.")]
+ private UnityEvent unityEvent;
+
+ ///
+ /// The to invoke when is raised.
+ ///
+ public UnityEvent UnityEvent => unityEvent;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionEventPair.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionEventPair.cs.meta
new file mode 100644
index 0000000..9e774cc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionEventPair.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d97d1c3ab07b4ed4bbfc253887cfcf2e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDigital.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDigital.cs
new file mode 100644
index 0000000..49300be
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDigital.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Generic Input Action Rule for raising actions based on specific criteria.
+ ///
+ [Serializable]
+ public struct InputActionRuleDigital : IInputActionRule
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Base Action that the rule will listen to.
+ /// The Action to raise if the criteria is met.
+ /// The criteria to check against for determining if the action should be raised.
+ public InputActionRuleDigital(MixedRealityInputAction baseAction, MixedRealityInputAction ruleAction, bool criteria)
+ {
+ this.baseAction = baseAction;
+ this.ruleAction = ruleAction;
+ this.criteria = criteria;
+ }
+
+ [SerializeField]
+ [Tooltip("The Base Action that the rule will listen to.")]
+ private MixedRealityInputAction baseAction;
+
+ ///
+ public MixedRealityInputAction BaseAction => baseAction;
+
+ [SerializeField]
+ [Tooltip("The Action to raise if the criteria is met.")]
+ private MixedRealityInputAction ruleAction;
+
+ ///
+ public MixedRealityInputAction RuleAction => ruleAction;
+
+ [SerializeField]
+ [Tooltip("The criteria to check against for determining if the action should be raised.")]
+ private bool criteria;
+
+ ///
+ public bool Criteria => criteria;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDigital.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDigital.cs.meta
new file mode 100644
index 0000000..5e0de30
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDigital.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 314f71e5252e4aabbf166d5840a5728b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDualAxis.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDualAxis.cs
new file mode 100644
index 0000000..d2cef96
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDualAxis.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Generic Input Action Rule for raising actions based on specific criteria.
+ ///
+ [Serializable]
+ public struct InputActionRuleDualAxis : IInputActionRule
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Base Action that the rule will listen to.
+ /// The Action to raise if the criteria is met.
+ /// The criteria to check against for determining if the action should be raised.
+ public InputActionRuleDualAxis(MixedRealityInputAction baseAction, MixedRealityInputAction ruleAction, Vector2 criteria)
+ {
+ this.baseAction = baseAction;
+ this.ruleAction = ruleAction;
+ this.criteria = criteria;
+ }
+
+ [SerializeField]
+ [Tooltip("The Base Action that the rule will listen to.")]
+ private MixedRealityInputAction baseAction;
+
+ ///
+ public MixedRealityInputAction BaseAction => baseAction;
+
+ [SerializeField]
+ [Tooltip("The Action to raise if the criteria is met.")]
+ private MixedRealityInputAction ruleAction;
+
+ ///
+ public MixedRealityInputAction RuleAction => ruleAction;
+
+ [SerializeField]
+ [Tooltip("The criteria to check against for determining if the action should be raised.")]
+ private Vector2 criteria;
+
+ ///
+ public Vector2 Criteria => criteria;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDualAxis.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDualAxis.cs.meta
new file mode 100644
index 0000000..fcaffca
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleDualAxis.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 257fe64c9bf54235b76d1505266122b2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRulePoseAxis.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRulePoseAxis.cs
new file mode 100644
index 0000000..46fac9c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRulePoseAxis.cs
@@ -0,0 +1,50 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Generic Input Action Rule for raising actions based on specific criteria.
+ ///
+ [Serializable]
+ public struct InputActionRulePoseAxis : IInputActionRule
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Base Action that the rule will listen to.
+ /// The Action to raise if the criteria is met.
+ /// The criteria to check against for determining if the action should be raised.
+ public InputActionRulePoseAxis(MixedRealityInputAction baseAction, MixedRealityInputAction ruleAction, MixedRealityPose criteria)
+ {
+ this.baseAction = baseAction;
+ this.ruleAction = ruleAction;
+ this.criteria = criteria;
+ }
+
+ [SerializeField]
+ [Tooltip("The Base Action that the rule will listen to.")]
+ private MixedRealityInputAction baseAction;
+
+ ///
+ public MixedRealityInputAction BaseAction => baseAction;
+
+ [SerializeField]
+ [Tooltip("The Action to raise if the criteria is met.")]
+ private MixedRealityInputAction ruleAction;
+
+ ///
+ public MixedRealityInputAction RuleAction => ruleAction;
+
+ [SerializeField]
+ [Tooltip("The criteria to check against for determining if the action should be raised.")]
+ private MixedRealityPose criteria;
+
+ ///
+ public MixedRealityPose Criteria => criteria;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRulePoseAxis.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRulePoseAxis.cs.meta
new file mode 100644
index 0000000..4f0cbda
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRulePoseAxis.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 0c13b12f17d249ceb2757ea5837f9ed7
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleQuaternionAxis.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleQuaternionAxis.cs
new file mode 100644
index 0000000..0cad75d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleQuaternionAxis.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Generic Input Action Rule for raising actions based on specific criteria.
+ ///
+ [Serializable]
+ public struct InputActionRuleQuaternionAxis : IInputActionRule
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Base Action that the rule will listen to.
+ /// The Action to raise if the criteria is met.
+ /// The criteria to check against for determining if the action should be raised.
+ public InputActionRuleQuaternionAxis(MixedRealityInputAction baseAction, MixedRealityInputAction ruleAction, Quaternion criteria)
+ {
+ this.baseAction = baseAction;
+ this.ruleAction = ruleAction;
+ this.criteria = criteria;
+ }
+
+ [SerializeField]
+ [Tooltip("The Base Action that the rule will listen to.")]
+ private MixedRealityInputAction baseAction;
+
+ ///
+ public MixedRealityInputAction BaseAction => baseAction;
+
+ [SerializeField]
+ [Tooltip("The Action to raise if the criteria is met.")]
+ private MixedRealityInputAction ruleAction;
+
+ ///
+ public MixedRealityInputAction RuleAction => ruleAction;
+
+ [SerializeField]
+ [Tooltip("The criteria to check against for determining if the action should be raised.")]
+ private Quaternion criteria;
+
+ ///
+ public Quaternion Criteria => criteria;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleQuaternionAxis.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleQuaternionAxis.cs.meta
new file mode 100644
index 0000000..6e24339
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleQuaternionAxis.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a3a544a8ca494fa797c3a7334f98298c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleSingleAxis.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleSingleAxis.cs
new file mode 100644
index 0000000..1f04b69
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleSingleAxis.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Generic Input Action Rule for raising actions based on specific criteria.
+ ///
+ [Serializable]
+ public struct InputActionRuleSingleAxis : IInputActionRule
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Base Action that the rule will listen to.
+ /// The Action to raise if the criteria is met.
+ /// The criteria to check against for determining if the action should be raised.
+ public InputActionRuleSingleAxis(MixedRealityInputAction baseAction, MixedRealityInputAction ruleAction, float criteria)
+ {
+ this.baseAction = baseAction;
+ this.ruleAction = ruleAction;
+ this.criteria = criteria;
+ }
+
+ [SerializeField]
+ [Tooltip("The Base Action that the rule will listen to.")]
+ private MixedRealityInputAction baseAction;
+
+ ///
+ public MixedRealityInputAction BaseAction => baseAction;
+
+ [SerializeField]
+ [Tooltip("The Action to raise if the criteria is met.")]
+ private MixedRealityInputAction ruleAction;
+
+ ///
+ public MixedRealityInputAction RuleAction => ruleAction;
+
+ [SerializeField]
+ [Tooltip("The criteria to check against for determining if the action should be raised.")]
+ private float criteria;
+
+ ///
+ public float Criteria => criteria;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleSingleAxis.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleSingleAxis.cs.meta
new file mode 100644
index 0000000..3364d73
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleSingleAxis.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b95f48a8b6244cc79828e1e956ffdc67
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleVectorAxis.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleVectorAxis.cs
new file mode 100644
index 0000000..b485ac2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleVectorAxis.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Generic Input Action Rule for raising actions based on specific criteria.
+ ///
+ [Serializable]
+ public struct InputActionRuleVectorAxis : IInputActionRule
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Base Action that the rule will listen to.
+ /// The Action to raise if the criteria is met.
+ /// The criteria to check against for determining if the action should be raised.
+ public InputActionRuleVectorAxis(MixedRealityInputAction baseAction, MixedRealityInputAction ruleAction, Vector3 criteria)
+ {
+ this.baseAction = baseAction;
+ this.ruleAction = ruleAction;
+ this.criteria = criteria;
+ }
+
+ [SerializeField]
+ [Tooltip("The Base Action that the rule will listen to.")]
+ private MixedRealityInputAction baseAction;
+
+ ///
+ public MixedRealityInputAction BaseAction => baseAction;
+
+ [SerializeField]
+ [Tooltip("The Action to raise if the criteria is met.")]
+ private MixedRealityInputAction ruleAction;
+
+ ///
+ public MixedRealityInputAction RuleAction => ruleAction;
+
+ [SerializeField]
+ [Tooltip("The criteria to check against for determining if the action should be raised.")]
+ private Vector3 criteria;
+
+ ///
+ public Vector3 Criteria => criteria;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleVectorAxis.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleVectorAxis.cs.meta
new file mode 100644
index 0000000..2109805
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/InputActionRuleVectorAxis.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4fbc1b935f564f83aac269fb6c600920
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/KeywordAndResponse.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/KeywordAndResponse.cs
new file mode 100644
index 0000000..8e76b25
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/KeywordAndResponse.cs
@@ -0,0 +1,46 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+using UnityEngine.Events;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Keyword/UnityEvent pair that ties voice input to UnityEvents wired up in the inspector.
+ ///
+ [Serializable]
+ public struct KeywordAndResponse
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The keyword to listen for.
+ /// The handler to be invoked when the keyword is recognized.
+ public KeywordAndResponse(string keyword, UnityEvent response)
+ {
+ this.keyword = keyword;
+ this.response = response;
+ }
+
+ [SerializeField]
+ [Tooltip("The keyword to listen for.")]
+ [SpeechKeyword]
+ private string keyword;
+
+ ///
+ /// The keyword to listen for.
+ ///
+ public string Keyword => keyword;
+
+ [SerializeField]
+ [Tooltip("The handler to be invoked when the keyword is recognized.")]
+ private UnityEvent response;
+
+ ///
+ /// The handler to be invoked when the keyword is recognized.
+ ///
+ public UnityEvent Response => response;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/KeywordAndResponse.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/KeywordAndResponse.cs.meta
new file mode 100644
index 0000000..cb16cdc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/KeywordAndResponse.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 25ecffaec23b402193f8b6c3fcc37721
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGestureMapping.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGestureMapping.cs
new file mode 100644
index 0000000..dacc864
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGestureMapping.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Data structure for mapping gestures to s that can be raised by the Input System.
+ ///
+ [Serializable]
+ public struct MixedRealityGestureMapping
+ {
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityGestureMapping(string description, GestureInputType gestureType, MixedRealityInputAction action)
+ {
+ this.description = description;
+ this.gestureType = gestureType;
+ this.action = action;
+ }
+
+ [SerializeField]
+ private string description;
+
+ ///
+ /// Simple, human readable description of the gesture.
+ ///
+ public string Description => description;
+
+ [SerializeField]
+ private GestureInputType gestureType;
+
+ ///
+ /// Type of Gesture.
+ ///
+ public GestureInputType GestureType => gestureType;
+
+ [SerializeField]
+ private MixedRealityInputAction action;
+
+ ///
+ /// Action for the associated gesture.
+ ///
+ public MixedRealityInputAction Action => action;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGestureMapping.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGestureMapping.cs.meta
new file mode 100644
index 0000000..c2cfcd3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGestureMapping.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a90ce54668fe43a4ae6cfd9283c078b9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGesturesProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGesturesProfile.cs
new file mode 100644
index 0000000..14dac0e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGesturesProfile.cs
@@ -0,0 +1,71 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Windows.Input;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Configuration profile settings for setting up and consuming Input Actions.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Gestures Profile", fileName = "MixedRealityGesturesProfile", order = (int)CreateProfileMenuItemIndices.Gestures)]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/gestures")]
+ public class MixedRealityGesturesProfile : BaseMixedRealityProfile
+ {
+ [EnumFlags]
+ [SerializeField]
+ [Tooltip("The recognizable Manipulation Gestures.")]
+ private WindowsGestureSettings manipulationGestures = 0;
+
+ ///
+ /// The recognizable Manipulation Gestures.
+ ///
+ public WindowsGestureSettings ManipulationGestures => manipulationGestures;
+
+ [EnumFlags]
+ [SerializeField]
+ [Tooltip("The recognizable Navigation Gestures.")]
+ private WindowsGestureSettings navigationGestures = 0;
+
+ ///
+ /// The recognizable Navigation Gestures.
+ ///
+ public WindowsGestureSettings NavigationGestures => navigationGestures;
+
+ [SerializeField]
+ [Tooltip("Should the Navigation use Rails on start?\nNote: This can be changed at runtime to switch between the two Navigation settings.")]
+ private bool useRailsNavigation = false;
+
+ public bool UseRailsNavigation => useRailsNavigation;
+
+ [EnumFlags]
+ [SerializeField]
+ [Tooltip("The recognizable Rails Navigation Gestures.")]
+ private WindowsGestureSettings railsNavigationGestures = 0;
+
+ ///
+ /// The recognizable Navigation Gestures.
+ ///
+ public WindowsGestureSettings RailsNavigationGestures => railsNavigationGestures;
+
+ [SerializeField]
+ private AutoStartBehavior windowsGestureAutoStart = AutoStartBehavior.AutoStart;
+
+ public AutoStartBehavior WindowsGestureAutoStart => windowsGestureAutoStart;
+
+ [SerializeField]
+ private MixedRealityGestureMapping[] gestures =
+ {
+ new MixedRealityGestureMapping("Hold", GestureInputType.Hold, MixedRealityInputAction.None),
+ new MixedRealityGestureMapping("Navigation", GestureInputType.Navigation, MixedRealityInputAction.None),
+ new MixedRealityGestureMapping("Manipulation", GestureInputType.Manipulation, MixedRealityInputAction.None),
+ };
+
+ ///
+ /// The currently configured gestures for the application.
+ ///
+ public MixedRealityGestureMapping[] Gestures => gestures;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGesturesProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGesturesProfile.cs.meta
new file mode 100644
index 0000000..e97cdbe
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityGesturesProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7554812f08ec49e694a8d9d4ee235a9c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputAction.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputAction.cs
new file mode 100644
index 0000000..6af6331
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputAction.cs
@@ -0,0 +1,96 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Collections;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// An Input Action for mapping an action to an Input Sources Button, Joystick, Sensor, etc.
+ ///
+ [Serializable]
+ public struct MixedRealityInputAction : IEqualityComparer
+ {
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityInputAction(uint id, string description, AxisType axisConstraint = AxisType.None)
+ {
+ this.id = id;
+ this.description = description;
+ this.axisConstraint = axisConstraint;
+ }
+
+ public static MixedRealityInputAction None { get; } = new MixedRealityInputAction(0, "None");
+
+ ///
+ /// The Unique Id of this Input Action.
+ ///
+ public uint Id => id;
+
+ [SerializeField]
+ private uint id;
+
+ ///
+ /// A short description of the Input Action.
+ ///
+ public string Description => description;
+
+ [SerializeField]
+ private string description;
+
+ ///
+ /// The Axis constraint for the Input Action
+ ///
+ public AxisType AxisConstraint => axisConstraint;
+
+ [SerializeField]
+ private AxisType axisConstraint;
+
+ public static bool operator ==(MixedRealityInputAction left, MixedRealityInputAction right)
+ {
+ return left.Equals(right);
+ }
+
+ public static bool operator !=(MixedRealityInputAction left, MixedRealityInputAction right)
+ {
+ return !left.Equals(right);
+ }
+
+ #region IEqualityComparer Implementation
+
+ bool IEqualityComparer.Equals(object left, object right)
+ {
+ if (ReferenceEquals(null, left) || ReferenceEquals(null, right)) { return false; }
+ if (!(left is MixedRealityInputAction) || !(right is MixedRealityInputAction)) { return false; }
+ return ((MixedRealityInputAction)left).Equals((MixedRealityInputAction)right);
+ }
+
+ public bool Equals(MixedRealityInputAction other)
+ {
+ return Id == other.Id &&
+ AxisConstraint == other.AxisConstraint;
+ }
+
+ public override bool Equals(object obj)
+ {
+ if (ReferenceEquals(null, obj)) { return false; }
+ return obj is MixedRealityInputAction action && Equals(action);
+ }
+
+ int IEqualityComparer.GetHashCode(object obj)
+ {
+ return obj is MixedRealityInputAction action ? action.GetHashCode() : 0;
+ }
+
+ public override int GetHashCode()
+ {
+ return $"{Id}.{AxisConstraint}".GetHashCode();
+ }
+
+ #endregion IEqualityComparer Implementation
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputAction.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputAction.cs.meta
new file mode 100644
index 0000000..2463d97
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputAction.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f3bb489e64434852a26a446c4efc841d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionRulesProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionRulesProfile.cs
new file mode 100644
index 0000000..35cbe12
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionRulesProfile.cs
@@ -0,0 +1,60 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Input Action Rules Profile", fileName = "MixedRealityInputActionRulesProfile", order = (int)CreateProfileMenuItemIndices.InputActionRules)]
+ public class MixedRealityInputActionRulesProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ private InputActionRuleDigital[] inputActionRulesDigital = null;
+
+ ///
+ /// All the Input Action Rules for based s
+ ///
+ public InputActionRuleDigital[] InputActionRulesDigital => inputActionRulesDigital;
+
+ [SerializeField]
+ private InputActionRuleSingleAxis[] inputActionRulesSingleAxis = null;
+
+ ///
+ /// All the Input Action Rules for based s
+ ///
+ public InputActionRuleSingleAxis[] InputActionRulesSingleAxis => inputActionRulesSingleAxis;
+
+ [SerializeField]
+ private InputActionRuleDualAxis[] inputActionRulesDualAxis = null;
+
+ ///
+ /// All the Input Action Rules for Vector2 based s
+ ///
+ public InputActionRuleDualAxis[] InputActionRulesDualAxis => inputActionRulesDualAxis;
+
+ [SerializeField]
+ private InputActionRuleVectorAxis[] inputActionRulesVectorAxis = null;
+
+ ///
+ /// All the Input Action Rules for Vector3 based s
+ ///
+ public InputActionRuleVectorAxis[] InputActionRulesVectorAxis => inputActionRulesVectorAxis;
+
+ [SerializeField]
+ private InputActionRuleQuaternionAxis[] inputActionRulesQuaternionAxis = null;
+
+ ///
+ /// All the Input Action Rules for Quaternion based s
+ ///
+ public InputActionRuleQuaternionAxis[] InputActionRulesQuaternionAxis => inputActionRulesQuaternionAxis;
+
+ [SerializeField]
+ private InputActionRulePoseAxis[] inputActionRulesPoseAxis = null;
+
+ ///
+ /// All the Input Action Rules for based s
+ ///
+ public InputActionRulePoseAxis[] InputActionRulesPoseAxis => inputActionRulesPoseAxis;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionRulesProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionRulesProfile.cs.meta
new file mode 100644
index 0000000..7dddf90
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionRulesProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ee54661ca8af487c9db40e57d479fa48
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionsProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionsProfile.cs
new file mode 100644
index 0000000..0d02744
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionsProfile.cs
@@ -0,0 +1,94 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System.Collections.Generic;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Configuration profile settings for setting up and consuming Input Actions.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Input Actions Profile", fileName = "MixedRealityInputActionsProfile", order = (int)CreateProfileMenuItemIndices.InputActions)]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/input-actions")]
+ public class MixedRealityInputActionsProfile : BaseMixedRealityProfile
+ {
+ private readonly string[] defaultInputActions =
+ {
+ "Select",
+ "Menu",
+ "Grip",
+ "Pointer",
+ "Walk",
+ "Look",
+ "Interact",
+ "Pickup",
+ "Inventory",
+ "ConversationSelect"
+ }; // Examples only, to be refined later.
+
+ private readonly AxisType[] defaultInputActionsAxis =
+ {
+ AxisType.Digital,
+ AxisType.Digital,
+ AxisType.SixDof,
+ AxisType.SixDof,
+ AxisType.DualAxis,
+ AxisType.DualAxis,
+ AxisType.DualAxis,
+ AxisType.Digital,
+ AxisType.DualAxis,
+ AxisType.DualAxis
+ }; // Examples only, to be refined later
+
+ [SerializeField]
+ [Tooltip("The list of actions users can do in your application.")]
+ private MixedRealityInputAction[] inputActions =
+ {
+ // 0 is reserved for "None"
+ new MixedRealityInputAction(1, "Select"),
+ new MixedRealityInputAction(2, "Menu"),
+ new MixedRealityInputAction(3, "Grip")
+ }; // Examples only, to be refined later
+
+ ///
+ /// The list of actions users can do in your application.
+ ///
+ /// Input Actions are device agnostic and can be paired with any number of device inputs across all platforms.
+ public MixedRealityInputAction[] InputActions => inputActions;
+
+ ///
+ /// Reset the current InputActions definitions to the Mixed Reality Toolkit defaults
+ /// If existing mappings exist, they will be preserved and pushed to the end of the array
+ ///
+ /// Default MRTK Actions plus any custom actions (if already configured)
+ public MixedRealityInputAction[] LoadMixedRealityToolKitDefaults()
+ {
+ var defaultActions = new List();
+ bool exists = false;
+
+ for (uint i = 0; i < defaultInputActions.Length; i++)
+ {
+ defaultActions.Add(new MixedRealityInputAction(i, defaultInputActions[i], defaultInputActionsAxis[i]));
+ }
+
+ for (int i = 0; i < inputActions.Length; i++)
+ {
+ if (defaultActions.Contains(inputActions[i]))
+ {
+ exists = true;
+ }
+
+ if (!exists)
+ {
+ defaultActions.Add(inputActions[i]);
+ }
+
+ exists = false;
+ }
+
+ return inputActions = defaultActions.ToArray();
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionsProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionsProfile.cs.meta
new file mode 100644
index 0000000..fc87ed8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputActionsProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d1a15d870c8b4e52acc4643bd258ed6e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputSystemProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputSystemProfile.cs
new file mode 100644
index 0000000..c7e6995
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputSystemProfile.cs
@@ -0,0 +1,231 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System.Collections.Generic;
+using System.Globalization;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Configuration profile settings for setting up controller pointers.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Input System Profile", fileName = "MixedRealityInputSystemProfile", order = (int)CreateProfileMenuItemIndices.Input)]
+ [MixedRealityServiceProfile(typeof(IMixedRealityInputSystem))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/overview")]
+ public class MixedRealityInputSystemProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ private MixedRealityInputDataProviderConfiguration[] dataProviderConfigurations = System.Array.Empty();
+
+ ///
+ /// List of input data provider configurations to initialize and manage by the Input System registrar
+ ///
+ public MixedRealityInputDataProviderConfiguration[] DataProviderConfigurations
+ {
+ get { return dataProviderConfigurations; }
+ internal set { dataProviderConfigurations = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("The focus provider service concrete type to use when raycasting.")]
+ [Implements(typeof(IMixedRealityFocusProvider), TypeGrouping.ByNamespaceFlat)]
+ private SystemType focusProviderType;
+
+ ///
+ /// The focus provider service concrete type to use when raycasting.
+ ///
+ public SystemType FocusProviderType
+ {
+ get { return focusProviderType; }
+ internal set { focusProviderType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("The raycast provider service concrete type to use when raycasting.")]
+ [Implements(typeof(IMixedRealityRaycastProvider), TypeGrouping.ByNamespaceFlat)]
+ private SystemType raycastProviderType;
+
+ ///
+ /// The raycast provider service concrete type to use when raycasting.
+ ///
+ public SystemType RaycastProviderType
+ {
+ get { return raycastProviderType; }
+ internal set { raycastProviderType = value; }
+ }
+
+ [SerializeField]
+ [Range(1, 2048)]
+ [Tooltip("Maximum number of colliders that can be detected in a SphereOverlap scene query.")]
+ private int focusQueryBufferSize = 128;
+
+ ///
+ /// Maximum number of colliders that can be detected in a SphereOverlap scene query.
+ ///
+ public int FocusQueryBufferSize => focusQueryBufferSize;
+
+ [SerializeField]
+ [Tooltip("Whether or not MRTK should try to raycast against Unity UI.")]
+ private bool shouldUseGraphicsRaycast = true;
+
+ ///
+ /// Whether or not MRTK should try to raycast against Unity UI.
+ ///
+ public bool ShouldUseGraphicsRaycast => shouldUseGraphicsRaycast;
+
+ [SerializeField]
+ [Tooltip("In case of a compound collider, does the individual collider receive focus")]
+ private bool focusIndividualCompoundCollider = false;
+
+ ///
+ /// In case of a compound collider, does the individual collider receive focus
+ ///
+ public bool FocusIndividualCompoundCollider
+ {
+ get { return focusIndividualCompoundCollider; }
+ set { focusIndividualCompoundCollider = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Input System Action Mapping profile for wiring up Controller input to Actions.")]
+ private MixedRealityInputActionsProfile inputActionsProfile;
+
+ ///
+ /// Input System Action Mapping profile for wiring up Controller input to Actions.
+ ///
+ public MixedRealityInputActionsProfile InputActionsProfile
+ {
+ get { return inputActionsProfile; }
+ internal set { inputActionsProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Input Action Rules Profile for raising actions based on specific criteria.")]
+ private MixedRealityInputActionRulesProfile inputActionRulesProfile;
+
+ ///
+ /// Input Action Rules Profile for raising actions based on specific criteria.
+ ///
+ public MixedRealityInputActionRulesProfile InputActionRulesProfile
+ {
+ get { return inputActionRulesProfile; }
+ internal set { inputActionRulesProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Pointer Configuration options")]
+ private MixedRealityPointerProfile pointerProfile;
+
+ ///
+ /// Pointer configuration options
+ ///
+ public MixedRealityPointerProfile PointerProfile
+ {
+ get { return pointerProfile; }
+ internal set { pointerProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Gesture Mapping Profile for recognizing gestures across all platforms.")]
+ private MixedRealityGesturesProfile gesturesProfile;
+
+ ///
+ /// Gesture Mapping Profile for recognizing gestures across all platforms.
+ ///
+ public MixedRealityGesturesProfile GesturesProfile
+ {
+ get { return gesturesProfile; }
+ internal set { gesturesProfile = value; }
+ }
+
+ ///
+ /// The list of cultures where speech recognition is supported
+ ///
+ private List supportedVoiceCultures = new List
+ {
+ new CultureInfo("en-US"),
+ new CultureInfo("en-CA"),
+ new CultureInfo("fr-CA"),
+ new CultureInfo("en-GB"),
+ new CultureInfo("en-AU"),
+ new CultureInfo("de-DE"),
+ new CultureInfo("fr-FR"),
+ new CultureInfo("zh-CN"),
+ new CultureInfo("ja-JP"),
+ new CultureInfo("es-ES"),
+ new CultureInfo("it-IT")
+ };
+
+ ///
+ /// Returns whether speech is supported for the current language or not
+ ///
+ public bool IsSpeechSupported => supportedVoiceCultures.Contains(CultureInfo.CurrentUICulture);
+
+ [SerializeField]
+ [Tooltip("Speech Command profile for wiring up Voice Input to Actions.")]
+ private MixedRealitySpeechCommandsProfile speechCommandsProfile;
+
+ ///
+ /// Speech commands profile for configured speech commands, for use by the speech recognition system
+ ///
+ public MixedRealitySpeechCommandsProfile SpeechCommandsProfile
+ {
+ get { return speechCommandsProfile; }
+ internal set { speechCommandsProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Enable and configure the devices for your application.")]
+ private bool enableControllerMapping = false;
+
+ ///
+ /// Enable and configure the devices for your application.
+ ///
+ public bool IsControllerMappingEnabled
+ {
+ get { return controllerMappingProfile != null && enableControllerMapping; }
+ internal set { enableControllerMapping = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Device profile for wiring up physical inputs to Actions.")]
+ private MixedRealityControllerMappingProfile controllerMappingProfile;
+
+ ///
+ /// Active profile for controller mapping configuration
+ ///
+ public MixedRealityControllerMappingProfile ControllerMappingProfile
+ {
+ get { return controllerMappingProfile; }
+ internal set { controllerMappingProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Device profile for rendering spatial controllers.")]
+ private MixedRealityControllerVisualizationProfile controllerVisualizationProfile;
+
+ ///
+ /// Device profile for rendering spatial controllers.
+ ///
+ public MixedRealityControllerVisualizationProfile ControllerVisualizationProfile
+ {
+ get { return controllerVisualizationProfile; }
+ internal set { controllerVisualizationProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Profile for configuring Hands tracking.")]
+ private MixedRealityHandTrackingProfile handTrackingProfile;
+
+ ///
+ /// Active profile for hands tracking
+ ///
+ public MixedRealityHandTrackingProfile HandTrackingProfile
+ {
+ get { return handTrackingProfile; }
+ private set { handTrackingProfile = value; }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputSystemProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputSystemProfile.cs.meta
new file mode 100644
index 0000000..15bce68
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityInputSystemProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b71cb900fa9dec5488df2deb180db58f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityPointerProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityPointerProfile.cs
new file mode 100644
index 0000000..c2c55b1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityPointerProfile.cs
@@ -0,0 +1,174 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Linq;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Configuration profile settings for setting up controller pointers.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Pointer Profile", fileName = "MixedRealityInputPointerProfile", order = (int)CreateProfileMenuItemIndices.Pointer)]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/pointers")]
+ public class MixedRealityPointerProfile : BaseMixedRealityProfile, ISerializationCallbackReceiver
+ {
+ [SerializeField]
+ [Tooltip("Maximum distance at which all pointers can collide with a GameObject, unless it has an override extent.")]
+ private float pointingExtent = 10f;
+
+ ///
+ /// Maximum distance at which all pointers can collide with a GameObject, unless it has an override extent.
+ ///
+ public float PointingExtent => pointingExtent;
+
+ [SerializeField]
+ [Tooltip("The default LayerMasks, in prioritized order, that are used to determine pointer's target. These layer masks are used if " +
+ "the pointer doesn't specify its own override")]
+ private LayerMask[] pointingRaycastLayerMasks = { UnityEngine.Physics.DefaultRaycastLayers };
+
+ ///
+ /// The default layerMasks, in prioritized order, that are used to determine the target when raycasting.
+ ///
+ public LayerMask[] PointingRaycastLayerMasks => pointingRaycastLayerMasks;
+
+ [SerializeField]
+ private bool debugDrawPointingRays = false;
+
+ ///
+ /// Toggle to enable or disable debug pointing rays.
+ ///
+ public bool DebugDrawPointingRays => debugDrawPointingRays;
+
+ [SerializeField]
+ private Color[] debugDrawPointingRayColors = null;
+
+ ///
+ /// The colors to use when debugging pointer rays.
+ ///
+ public Color[] DebugDrawPointingRayColors => debugDrawPointingRayColors;
+
+ [Prefab]
+ [SerializeField]
+ [Tooltip("The gaze cursor prefab to use on the Gaze pointer.")]
+ private GameObject gazeCursorPrefab = null;
+
+ ///
+ /// The gaze cursor prefab to use on the Gaze pointer.
+ ///
+ public GameObject GazeCursorPrefab => gazeCursorPrefab;
+
+ [SerializeField]
+ [Tooltip("The concrete type of IMixedRealityGazeProvider to use.")]
+ [Implements(typeof(IMixedRealityGazeProvider), TypeGrouping.ByNamespaceFlat)]
+ private SystemType gazeProviderType;
+
+ ///
+ /// The concrete type of to use.
+ ///
+ public SystemType GazeProviderType
+ {
+ get { return gazeProviderType; }
+ internal set { gazeProviderType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("If true, platform-specific head gaze override is used, when available. Otherwise, the center of the camera frame is used by default.")]
+ private bool useHeadGazeOverride = false;
+
+ ///
+ /// If true, platform-specific head gaze override is used, when available. Otherwise, the center of the camera frame is used by default.
+ ///
+ public bool UseHeadGazeOverride => useHeadGazeOverride;
+
+ [SerializeField]
+ [Tooltip("If true, eye-based tracking will be used as gaze input when available. This field does not control whether eye tracking data is provided.")]
+ private bool isEyeTrackingEnabled = false;
+
+ ///
+ /// If true, eye-based tracking will be used as gaze input when available. This field does not control whether eye tracking data is provided.
+ ///
+ public bool IsEyeTrackingEnabled
+ {
+ get { return isEyeTrackingEnabled; }
+ internal set { isEyeTrackingEnabled = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("The Pointer options for this profile.")]
+ private PointerOption[] pointerOptions = System.Array.Empty();
+
+ ///
+ /// The Pointer options for this profile.
+ ///
+ public PointerOption[] PointerOptions => pointerOptions;
+
+ [SerializeField]
+ [Implements(typeof(IMixedRealityPointerMediator), TypeGrouping.ByNamespaceFlat)]
+ [Tooltip("The concrete Pointer Mediator component to use. This is a component that mediates all pointers in system, disabling / enabling them based on the state of other pointers.")]
+ private SystemType pointerMediator = null;
+
+ ///
+ /// The concrete Pointer Mediator component to use.
+ /// This is a component that mediates all pointers in system, disabling / enabling them based on the state of other pointers.
+ ///
+ public SystemType PointerMediator => pointerMediator;
+
+ [SerializeField]
+ [Implements(typeof(IMixedRealityPrimaryPointerSelector), TypeGrouping.ByNamespaceFlat)]
+ [Tooltip("Primary pointer selector implementation to use. This is used by the focus provider to choose the primary pointer.")]
+ private SystemType primaryPointerSelector = null;
+
+ ///
+ /// Primary pointer selector implementation to use. This is used by the focus provider to choose the primary pointer.
+ ///
+ public SystemType PrimaryPointerSelector => primaryPointerSelector;
+
+ void ISerializationCallbackReceiver.OnBeforeSerialize()
+ {
+ for (int i = 0; i < pointerOptions.Length; i++)
+ {
+ ref PointerOption pointerOption = ref pointerOptions[i];
+ IMixedRealityPointer pointer = pointerOption.PointerPrefab != null ? pointerOption.PointerPrefab.GetComponent() : null;
+
+ if (pointer.IsNull()
+ || (pointer.PrioritizedLayerMasksOverride != null
+ && pointer.PrioritizedLayerMasksOverride.Length > 0
+ && pointerOption.PrioritizedLayerMasks != null
+ && pointerOption.PrioritizedLayerMasks.SequenceEqual(pointer.PrioritizedLayerMasksOverride))
+ || (pointingRaycastLayerMasks != null
+ && pointingRaycastLayerMasks.Length > 0
+ && pointerOption.PrioritizedLayerMasks != null
+ && pointerOption.PrioritizedLayerMasks.SequenceEqual(pointingRaycastLayerMasks)))
+ {
+ continue;
+ }
+
+ // If the prefab has new LayerMasks, sync with prioritizedLayerMasks
+ int pointerPrioritizedLayerMasksOverrideCount = pointer.PrioritizedLayerMasksOverride?.Length ?? 0;
+ if (pointerPrioritizedLayerMasksOverrideCount != 0)
+ {
+ if (pointerOption.PrioritizedLayerMasks?.Length != pointerPrioritizedLayerMasksOverrideCount)
+ {
+ pointerOption.PrioritizedLayerMasks = new LayerMask[pointerPrioritizedLayerMasksOverrideCount];
+ }
+ Array.Copy(pointer.PrioritizedLayerMasksOverride, pointerOption.PrioritizedLayerMasks, pointerPrioritizedLayerMasksOverrideCount);
+ }
+ // If the prefab doesn't have any LayerMasks, initialize with the global default
+ else
+ {
+ int pointingRaycastLayerMasksCount = pointingRaycastLayerMasks.Length;
+ if (pointerOption.PrioritizedLayerMasks?.Length != pointingRaycastLayerMasksCount)
+ {
+ pointerOption.PrioritizedLayerMasks = new LayerMask[pointingRaycastLayerMasksCount];
+ }
+ Array.Copy(pointingRaycastLayerMasks, pointerOption.PrioritizedLayerMasks, pointingRaycastLayerMasksCount);
+ }
+ }
+ }
+
+ void ISerializationCallbackReceiver.OnAfterDeserialize() { }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityPointerProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityPointerProfile.cs.meta
new file mode 100644
index 0000000..39eae6b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityPointerProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: db393d206eab4604ab74278cb6cda355
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityRaycastHit.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityRaycastHit.cs
new file mode 100644
index 0000000..3429fe0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityRaycastHit.cs
@@ -0,0 +1,102 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// The resulting hit information from an IMixedRealityRaycastProvider.
+ ///
+ public struct MixedRealityRaycastHit
+ {
+ public Vector3 point;
+ public Vector3 normal;
+ public Vector3 barycentricCoordinate;
+ public float distance;
+ public int triangleIndex;
+ public Vector2 textureCoord;
+ public Vector2 textureCoord2;
+ public Transform transform;
+ public Vector2 lightmapCoord;
+ public bool raycastValid;
+ public Collider collider;
+
+ public MixedRealityRaycastHit(bool raycastValid, RaycastHit hitInfo)
+ {
+ this.raycastValid = raycastValid;
+ if (raycastValid)
+ {
+ point = hitInfo.point;
+ normal = hitInfo.normal;
+ barycentricCoordinate = hitInfo.barycentricCoordinate;
+ distance = hitInfo.distance;
+ triangleIndex = hitInfo.triangleIndex;
+
+ MeshCollider meshCollider = hitInfo.collider as MeshCollider;
+ if (meshCollider == null)
+ {
+ textureCoord = hitInfo.textureCoord;
+ textureCoord2 = hitInfo.textureCoord2;
+ lightmapCoord = hitInfo.lightmapCoord;
+ }
+ else
+ {
+ Mesh sharedMesh = meshCollider.sharedMesh;
+ if (sharedMesh != null && sharedMesh.isReadable)
+ {
+#if UNITY_2019_4_OR_NEWER
+ if (sharedMesh.HasVertexAttribute(UnityEngine.Rendering.VertexAttribute.TexCoord0))
+ {
+ textureCoord = hitInfo.textureCoord;
+ }
+ else
+ {
+ textureCoord = Vector2.zero;
+ }
+
+ // This checks for TexCoord1, since textureCoord2 and lightmapCoord both query that index
+ // via CalculateRaycastTexCoord(collider, m_UV, m_Point, m_FaceID, 1); (the last parameter is the index)
+ if (sharedMesh.HasVertexAttribute(UnityEngine.Rendering.VertexAttribute.TexCoord1))
+ {
+ textureCoord2 = hitInfo.textureCoord2;
+ lightmapCoord = hitInfo.lightmapCoord;
+ }
+ else
+ {
+ textureCoord2 = Vector2.zero;
+ lightmapCoord = Vector2.zero;
+ }
+#else
+ textureCoord = hitInfo.textureCoord;
+ textureCoord2 = hitInfo.textureCoord2;
+ lightmapCoord = hitInfo.lightmapCoord;
+#endif
+ }
+ else
+ {
+ textureCoord = Vector2.zero;
+ textureCoord2 = Vector2.zero;
+ lightmapCoord = Vector2.zero;
+ }
+ }
+
+ transform = hitInfo.transform;
+ collider = hitInfo.collider;
+ }
+ else
+ {
+ point = Vector3.zero;
+ normal = Vector3.zero;
+ barycentricCoordinate = Vector3.zero;
+ distance = 0;
+ triangleIndex = 0;
+ textureCoord = Vector2.zero;
+ textureCoord2 = Vector2.zero;
+ transform = null;
+ lightmapCoord = Vector2.zero;
+ collider = null;
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityRaycastHit.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityRaycastHit.cs.meta
new file mode 100644
index 0000000..5ce1633
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealityRaycastHit.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4c83a552926f36e45ab326756050af3f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealitySpeechCommandsProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealitySpeechCommandsProfile.cs
new file mode 100644
index 0000000..5cf1dc3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealitySpeechCommandsProfile.cs
@@ -0,0 +1,43 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Configuration profile settings for setting up and consuming Speech Commands.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Speech Commands Profile", fileName = "MixedRealitySpeechCommandsProfile", order = (int)CreateProfileMenuItemIndices.Speech)]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/speech")]
+ public class MixedRealitySpeechCommandsProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("Whether the recognizer should be activated on start.")]
+ private AutoStartBehavior startBehavior = AutoStartBehavior.AutoStart;
+
+ ///
+ /// The list of Speech Commands users use in your application.
+ ///
+ public AutoStartBehavior SpeechRecognizerStartBehavior => startBehavior;
+
+ [SerializeField]
+ [Tooltip("Select the minimum confidence level for recognized words")]
+ private RecognitionConfidenceLevel recognitionConfidenceLevel = RecognitionConfidenceLevel.Medium;
+
+ ///
+ /// The speech recognizer's minimum confidence level setting that will raise the action.
+ ///
+ public RecognitionConfidenceLevel SpeechRecognitionConfidenceLevel => recognitionConfidenceLevel;
+
+ [SerializeField]
+ [Tooltip("The list of Speech Commands users use in your application.")]
+ private SpeechCommands[] speechCommands = System.Array.Empty();
+
+ ///
+ /// The list of Speech Commands users use in your application.
+ ///
+ public SpeechCommands[] SpeechCommands => speechCommands;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealitySpeechCommandsProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealitySpeechCommandsProfile.cs.meta
new file mode 100644
index 0000000..dac15b6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/MixedRealitySpeechCommandsProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1f18fec9b55c4f818e284af454161962
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerBehavior.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerBehavior.cs
new file mode 100644
index 0000000..7ad157d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerBehavior.cs
@@ -0,0 +1,27 @@
+
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Specifies how a pointer in MRTK's default input system behaves.
+ ///
+ public enum PointerBehavior
+ {
+ ///
+ /// Pointer active state is managed by MRTK input system. If it is a near pointer (grab, poke), it
+ /// will be always enabled. If it is not a near pointer, it will get disabled if any near pointer on the
+ /// same hand is active. This is what allows rays to turn off when a hand is near a grabbable.
+ ///
+ Default = 0,
+ ///
+ /// Pointer is always on, regardless of what other pointers are active.
+ ///
+ AlwaysOn,
+ ///
+ /// Pointer is always off, regardless of what other pointers are active.
+ ///
+ AlwaysOff
+ };
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerBehavior.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerBehavior.cs.meta
new file mode 100644
index 0000000..5b6a41b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerBehavior.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: bd85cf387e8e41446ad2330a8caaa9b5
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerOption.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerOption.cs
new file mode 100644
index 0000000..7928647
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerOption.cs
@@ -0,0 +1,96 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Linq;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Defines a pointer option to assign to a controller.
+ ///
+ [Serializable]
+ public struct PointerOption : ISerializationCallbackReceiver
+ {
+ ///
+ /// Constructor.
+ ///
+ public PointerOption(SupportedControllerType controllerType, Handedness handedness, GameObject pointerPrefab, LayerMask[] prioritizedLayerMasks = null)
+ {
+ this.controllerType = controllerType;
+ this.handedness = handedness;
+ this.pointerPrefab = pointerPrefab;
+ this.prioritizedLayerMasks = prioritizedLayerMasks ?? new LayerMask[1] { UnityEngine.Physics.DefaultRaycastLayers };
+ }
+
+ [EnumFlags]
+ [SerializeField]
+ [Tooltip("The type of Controller this pointer can be attached to at runtime.")]
+ private SupportedControllerType controllerType;
+
+ ///
+ /// The type of Controller this pointer can be attached to at runtime.
+ ///
+ /// If is selected, then it will attach to any controller type
+ public SupportedControllerType ControllerType => controllerType;
+
+ [SerializeField]
+ [Tooltip("Defines valid hand(s) to create the pointer prefab on.")]
+ private Handedness handedness;
+
+ ///
+ /// Defines valid hand(s) to create the pointer prefab on.
+ ///
+ public Handedness Handedness => handedness;
+
+ [SerializeField]
+ [Tooltip("The prefab with an IMixedRealityPointer component to create when a valid controller becomes available.")]
+ private GameObject pointerPrefab;
+
+ ///
+ /// The prefab with an component to create when a valid controller becomes available.
+ ///
+ public GameObject PointerPrefab => pointerPrefab;
+
+ [SerializeField]
+ [Tooltip("The LayerMasks, in prioritized order, which are used to determine the target.")]
+ private LayerMask[] prioritizedLayerMasks;
+
+ ///
+ /// The LayerMasks, in prioritized order, which are used to determine the target
+ ///
+ public LayerMask[] PrioritizedLayerMasks
+ {
+ get => prioritizedLayerMasks;
+ internal set => prioritizedLayerMasks = value;
+ }
+
+ void ISerializationCallbackReceiver.OnBeforeSerialize()
+ {
+ if (pointerPrefab == null)
+ {
+ return;
+ }
+
+ IMixedRealityPointer pointer = pointerPrefab.GetComponent();
+ if (pointer.IsNull()
+ || pointer.PrioritizedLayerMasksOverride == null
+ || pointer.PrioritizedLayerMasksOverride.Length == 0
+ || (prioritizedLayerMasks != null && pointer.PrioritizedLayerMasksOverride.SequenceEqual(prioritizedLayerMasks)))
+ {
+ return;
+ }
+
+ int pointerPrioritizedLayerMasksOverrideCount = pointer.PrioritizedLayerMasksOverride.Length;
+ if (prioritizedLayerMasks?.Length != pointerPrioritizedLayerMasksOverrideCount)
+ {
+ prioritizedLayerMasks = new LayerMask[pointerPrioritizedLayerMasksOverrideCount];
+ }
+ Array.Copy(pointer.PrioritizedLayerMasksOverride, prioritizedLayerMasks, pointerPrioritizedLayerMasksOverrideCount);
+ }
+
+ void ISerializationCallbackReceiver.OnAfterDeserialize() { }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerOption.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerOption.cs.meta
new file mode 100644
index 0000000..c4b5c39
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/PointerOption.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b64872b3e5aa40cb862ecd0762edddc6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/SpeechCommands.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/SpeechCommands.cs
new file mode 100644
index 0000000..f6cf7ce
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/SpeechCommands.cs
@@ -0,0 +1,96 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Data structure for mapping Voice and Keyboard input to s that can be raised by the Input System.
+ ///
+ [Serializable]
+ public struct SpeechCommands
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The Keyword.
+ /// The KeyCode.
+ /// The Action to perform when Keyword or KeyCode is recognized.
+ /// An optional key to use to override the keyword with a localized version
+ public SpeechCommands(string keyword, KeyCode keyCode, MixedRealityInputAction action, string localizationKey = "")
+ {
+ this.keyword = keyword;
+ this.keyCode = keyCode;
+ this.action = action;
+ this.localizationKey = localizationKey;
+ this.localizedKeyword = null;
+ }
+
+ [SerializeField]
+ [Tooltip("The key to use to find a localized keyword")]
+ private string localizationKey;
+
+ private string localizedKeyword;
+
+ ///
+ /// The localized version of the keyword
+ ///
+ public string LocalizedKeyword
+ {
+ get
+ {
+#if WINDOWS_UWP
+ if (!string.IsNullOrWhiteSpace(localizationKey) && string.IsNullOrWhiteSpace(localizedKeyword))
+ {
+ try
+ {
+ var resourceLoader = global::Windows.ApplicationModel.Resources.ResourceLoader.GetForViewIndependentUse();
+ localizedKeyword = resourceLoader.GetString(localizationKey);
+ }
+ catch (System.Exception e)
+ {
+ // Ignore the exception and just use the fallback
+ Debug.LogError("GetLocalizedKeywordException: " + e.Message);
+ }
+ }
+#endif
+ return string.IsNullOrWhiteSpace(localizedKeyword) ? keyword : localizedKeyword;
+ }
+ }
+
+ [SerializeField]
+ [Tooltip("The Fallback keyword to listen for.")]
+ private string keyword;
+
+ ///
+ /// The Fallback Keyword to listen for, or the localization key if no fallback keyword was set.
+ ///
+ public string Keyword
+ {
+ get
+ {
+ return string.IsNullOrWhiteSpace(keyword) ? localizationKey : keyword;
+ }
+ }
+
+ [SerializeField]
+ [Tooltip("The corresponding KeyCode that also raises the same action as the Localized Keyword.")]
+ private KeyCode keyCode;
+
+ ///
+ /// The corresponding KeyCode that also raises the same action as the Keyword.
+ ///
+ public KeyCode KeyCode => keyCode;
+
+ [SerializeField]
+ [Tooltip("The Action that is raised by either the Localized Keyword or KeyCode.")]
+ private MixedRealityInputAction action;
+
+ ///
+ /// The that is raised by either the Keyword or KeyCode.
+ ///
+ public MixedRealityInputAction Action => action;
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/SpeechCommands.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/SpeechCommands.cs.meta
new file mode 100644
index 0000000..c12b8e1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/SpeechCommands.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2e3ab1f2131a421d8c3e05386d24074d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/TouchableEventType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/TouchableEventType.cs
new file mode 100644
index 0000000..017eabe
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/TouchableEventType.cs
@@ -0,0 +1,14 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Type of Events to receive from a PokePointer.
+ ///
+ public enum TouchableEventType
+ {
+ Touch,
+ Pointer,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/TouchableEventType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/TouchableEventType.cs.meta
new file mode 100644
index 0000000..c59c057
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/TouchableEventType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a5e75d2e7738c7748a605a5349a762b2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/WindowsGestureSettings.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/WindowsGestureSettings.cs
new file mode 100644
index 0000000..5ed6934
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/WindowsGestureSettings.cs
@@ -0,0 +1,64 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Windows.Input
+{
+ ///
+ /// Copy of Unity's GestureSettings
+ ///
+ [Flags]
+ public enum WindowsGestureSettings
+ {
+ ///
+ /// Enable support for the tap gesture.
+ ///
+ Tap = 1 << 0, // HEX: 0x00000001 | Decimal: 1
+
+ ///
+ /// Enable support for the double-tap gesture.
+ ///
+ DoubleTap = 1 << 1, // HEX: 0x00000002 | Decimal: 2
+
+ ///
+ /// Enable support for the hold gesture.
+ ///
+ Hold = 1 << 2, // HEX: 0x00000004 | Decimal: 4
+
+ ///
+ /// Enable support for the manipulation gesture which tracks changes to the hand's position. This gesture is relative to the start position of the gesture and measures an absolute movement through the world.
+ ///
+ ManipulationTranslate = 1 << 3, // HEX: 0x00000008 | Decimal: 8
+
+ ///
+ /// Enable support for the navigation gesture, in the horizontal axis.
+ ///
+ NavigationX = 1 << 4, // HEX: 0x00000010 | Decimal: 16
+
+ ///
+ /// Enable support for the navigation gesture, in the vertical axis.
+ ///
+ NavigationY = 1 << 5, // HEX: 0x00000020 | Decimal: 32
+
+ ///
+ /// Enable support for the navigation gesture, in the depth axis.
+ ///
+ NavigationZ = 1 << 6, // HEX: 0x00000040 | Decimal: 64
+
+ ///
+ /// Enable support for the navigation gesture, in the horizontal axis using rails (guides).
+ ///
+ NavigationRailsX = 1 << 7, // HEX: 0x00000080 | Decimal: 128
+
+ ///
+ /// Enable support for the navigation gesture, in the vertical axis using rails (guides).
+ ///
+ NavigationRailsY = 1 << 8, // HEX: 0x00000100 | Decimal: 256
+
+ ///
+ /// Enable support for the navigation gesture, in the depth axis using rails (guides).
+ ///
+ NavigationRailsZ = 1 << 9, // HEX: 0x00000200 | Decimal: 512
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/WindowsGestureSettings.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/WindowsGestureSettings.cs.meta
new file mode 100644
index 0000000..b1a8a6e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/InputSystem/WindowsGestureSettings.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 997ae55289cd25b41b389340334d2ba0
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines.meta
new file mode 100644
index 0000000..d605c87
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 8cbd134ae9ff4b1ab2b5b8def14d1053
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/DistortionMode.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/DistortionMode.cs
new file mode 100644
index 0000000..10e6010
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/DistortionMode.cs
@@ -0,0 +1,20 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// How to apply the distortion along the line.
+ ///
+ public enum DistortionMode
+ {
+ ///
+ /// Use the normalized length of the line plus its distortion strength curve to determine distortion strength
+ ///
+ NormalizedLength = 0,
+ ///
+ /// Use a single value to determine distortion strength
+ ///
+ Uniform,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/DistortionMode.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/DistortionMode.cs.meta
new file mode 100644
index 0000000..137ef52
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/DistortionMode.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8dd26458b9d6400b99f86ca6336ebed8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationMode.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationMode.cs
new file mode 100644
index 0000000..a558cbc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationMode.cs
@@ -0,0 +1,24 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Default options for how to distribute interpolated points in a line renderer
+ ///
+ public enum InterpolationMode
+ {
+ ///
+ /// Specify the number of interpolation steps manually
+ ///
+ FromSteps = 0,
+ ///
+ /// Create steps based on total length of line + manually specified length
+ ///
+ FromLength,
+ ///
+ /// Create steps based on total length of line + animation curve
+ ///
+ FromCurve
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationMode.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationMode.cs.meta
new file mode 100644
index 0000000..a5cdd7a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationMode.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: cf8e569c382e4377a831fd0f095830c2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationType.cs
new file mode 100644
index 0000000..22d9c64
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationType.cs
@@ -0,0 +1,15 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines the type of interpolation to use when calculating a spline.
+ ///
+ public enum InterpolationType
+ {
+ Bezier = 0,
+ CatmullRom,
+ Hermite,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationType.cs.meta
new file mode 100644
index 0000000..44c4ca1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/InterpolationType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e5291c0ac0fc496982ec8afeed071587
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LinePointTransformMode.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LinePointTransformMode.cs
new file mode 100644
index 0000000..2611a64
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LinePointTransformMode.cs
@@ -0,0 +1,20 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines how a base line data provider will transform its points
+ ///
+ public enum LinePointTransformMode
+ {
+ ///
+ /// Use the local line transform. More reliable but with a performance cost.
+ ///
+ UseTransform,
+ ///
+ /// Use a matrix. Lines that are not active and enabled will not update point positions.
+ ///
+ UseMatrix,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LinePointTransformMode.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LinePointTransformMode.cs.meta
new file mode 100644
index 0000000..286e64a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LinePointTransformMode.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 89d2256185b38ab4d94c2d07ba805ddc
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LineRotationMode.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LineRotationMode.cs
new file mode 100644
index 0000000..e9a1e5e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LineRotationMode.cs
@@ -0,0 +1,24 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines how to calculate the line's rotation at any given point.
+ ///
+ public enum LineRotationMode
+ {
+ ///
+ /// Don't rotate
+ ///
+ None = 0,
+ ///
+ /// Use velocity to calculate the line's rotation
+ ///
+ Velocity,
+ ///
+ /// Rotate relative to direction from origin point
+ ///
+ RelativeToOrigin,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LineRotationMode.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LineRotationMode.cs.meta
new file mode 100644
index 0000000..a61ab46
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/LineRotationMode.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 3b2c29c9e22e4e62b81b27026ed45019
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/PointDistributionMode.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/PointDistributionMode.cs
new file mode 100644
index 0000000..c0e902c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/PointDistributionMode.cs
@@ -0,0 +1,28 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines how to get an interpolated point along a line
+ ///
+ public enum PointDistributionMode
+ {
+ ///
+ /// Don't adjust placement
+ ///
+ None = 0,
+ ///
+ /// Adjust placement automatically (default)
+ ///
+ Auto,
+ ///
+ /// Place based on distance
+ ///
+ DistanceSingleValue,
+ ///
+ /// Place based on curve
+ ///
+ DistanceCurveValue,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/PointDistributionMode.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/PointDistributionMode.cs.meta
new file mode 100644
index 0000000..4039a2a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/PointDistributionMode.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b3d54b2ce3b84633ab7196ba563bf842
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/StepMode.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/StepMode.cs
new file mode 100644
index 0000000..aeeb946
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/StepMode.cs
@@ -0,0 +1,20 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines how to generate points in a line renderer
+ ///
+ public enum StepMode
+ {
+ ///
+ /// Draw points based on LineStepCount
+ ///
+ Interpolated = 0,
+ ///
+ /// Draw only the points available in the source - use this for hard edges
+ ///
+ FromSource,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/StepMode.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/StepMode.cs.meta
new file mode 100644
index 0000000..80fa57d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Lines/StepMode.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c25f5594e515484d9eb9eb91cc7434e9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityExperienceSettingsProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityExperienceSettingsProfile.cs
new file mode 100644
index 0000000..52fd36a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityExperienceSettingsProfile.cs
@@ -0,0 +1,42 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Experience settings profile for the Mixed Reality Toolkit.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality Toolkit/Profiles/Mixed Reality Toolkit Experience Settings Profile", fileName = "MixedRealityToolkitExperienceSettingsProfile", order = (int)CreateProfileMenuItemIndices.Configuration)]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/configuration/mixed-reality-configuration-guide")]
+ public class MixedRealityExperienceSettingsProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("The scale of the Mixed Reality experience.")]
+ private ExperienceScale targetExperienceScale = ExperienceScale.Room;
+
+ ///
+ /// The desired the scale of the experience.
+ ///
+ public ExperienceScale TargetExperienceScale
+ {
+ get { return targetExperienceScale; }
+ set { targetExperienceScale = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("The amount to offset the MixedRealitySceneContent for the target experience. 1 unit = 1 meter")]
+ private float contentOffset = 0.0f;
+
+ ///
+ /// The height above the floor for the Mixed Reality Experience. 1 unit = 1 meter
+ ///
+ public float ContentOffset
+ {
+ get { return contentOffset; }
+ set { contentOffset = value; }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityExperienceSettingsProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityExperienceSettingsProfile.cs.meta
new file mode 100644
index 0000000..48ee80f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityExperienceSettingsProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 3193d07953d1c3e44ad7ad26908fe6b2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityInputDataProviderConfiguration.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityInputDataProviderConfiguration.cs
new file mode 100644
index 0000000..c211a2f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityInputDataProviderConfiguration.cs
@@ -0,0 +1,73 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ [Serializable]
+ public struct MixedRealityInputDataProviderConfiguration : IMixedRealityServiceConfiguration
+ {
+ [SerializeField]
+ [Implements(typeof(IMixedRealityInputDeviceManager), TypeGrouping.ByNamespaceFlat)]
+ private SystemType componentType;
+
+ ///
+ public SystemType ComponentType => componentType;
+
+ [SerializeField]
+ private string componentName;
+
+ ///
+ public string ComponentName => componentName;
+
+ [SerializeField]
+ private uint priority;
+
+ ///
+ public uint Priority => priority;
+
+ [SerializeField]
+ [EnumFlags]
+ private SupportedPlatforms runtimePlatform;
+
+ ///
+ public SupportedPlatforms RuntimePlatform => runtimePlatform;
+
+ [SerializeField]
+ private BaseMixedRealityProfile deviceManagerProfile;
+
+ ///
+ public BaseMixedRealityProfile Profile => deviceManagerProfile;
+
+ ///
+ /// Device manager specific configuration profile.
+ ///
+ [Obsolete("Use the Profile property instead.")]
+ public BaseMixedRealityProfile DeviceManagerProfile => deviceManagerProfile;
+
+ ///
+ /// Constructor.
+ ///
+ /// The of the data provider.
+ /// The friendly name of the data provider.
+ /// The load priority of the data provider.
+ /// The runtime platform(s) supported by the data provider.
+ /// The configuration profile for the data provider.
+ public MixedRealityInputDataProviderConfiguration(
+ SystemType componentType,
+ string componentName,
+ uint priority,
+ SupportedPlatforms runtimePlatform,
+ BaseMixedRealityProfile profile)
+ {
+ this.componentType = componentType;
+ this.componentName = componentName;
+ this.priority = priority;
+ this.runtimePlatform = runtimePlatform;
+ deviceManagerProfile = profile;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityInputDataProviderConfiguration.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityInputDataProviderConfiguration.cs.meta
new file mode 100644
index 0000000..c320d6b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityInputDataProviderConfiguration.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 502de787a9d657e439beedf2a862e7ae
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityRegisteredServiceProvidersProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityRegisteredServiceProvidersProfile.cs
new file mode 100644
index 0000000..9d81fc5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityRegisteredServiceProvidersProfile.cs
@@ -0,0 +1,20 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Registered Service Providers Profile", fileName = "MixedRealityRegisteredServiceProvidersProfile", order = (int)CreateProfileMenuItemIndices.RegisteredServiceProviders)]
+ public class MixedRealityRegisteredServiceProvidersProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ private MixedRealityServiceConfiguration[] configurations = null;
+
+ ///
+ /// Currently registered system and manager configurations.
+ ///
+ public MixedRealityServiceConfiguration[] Configurations => configurations;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityRegisteredServiceProvidersProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityRegisteredServiceProvidersProfile.cs.meta
new file mode 100644
index 0000000..6469814
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityRegisteredServiceProvidersProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: eebbca41bb0b40d298ef201735d08616
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityServiceConfiguration.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityServiceConfiguration.cs
new file mode 100644
index 0000000..5aad4b9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityServiceConfiguration.cs
@@ -0,0 +1,76 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Defines a system, feature, or manager to be registered with as a on startup.
+ ///
+ [Serializable]
+ public struct MixedRealityServiceConfiguration : IMixedRealityServiceConfiguration
+ {
+ ///
+ /// Constructor.
+ ///
+ /// The concrete type for the system, feature or manager.
+ /// The simple, human readable name for the system, feature, or manager.
+ /// The priority this system, feature, or manager will be initialized in.
+ /// The runtime platform(s) to run this system, feature, or manager on.
+ /// The configuration profile for the service.
+ public MixedRealityServiceConfiguration(
+ SystemType componentType,
+ string componentName,
+ uint priority,
+ SupportedPlatforms runtimePlatform,
+ BaseMixedRealityProfile configurationProfile)
+ {
+ this.componentType = componentType;
+ this.componentName = componentName;
+ this.priority = priority;
+ this.runtimePlatform = runtimePlatform;
+ this.configurationProfile = configurationProfile;
+ }
+
+ [SerializeField]
+ [Implements(typeof(IMixedRealityExtensionService), TypeGrouping.ByNamespaceFlat)]
+ private SystemType componentType;
+
+ ///
+ public SystemType ComponentType => componentType;
+
+ [SerializeField]
+ private string componentName;
+
+ ///
+ public string ComponentName => componentName;
+
+ [SerializeField]
+ private uint priority;
+
+ ///
+ public uint Priority => priority;
+
+ [EnumFlags]
+ [SerializeField]
+ private SupportedPlatforms runtimePlatform;
+
+ ///
+ public SupportedPlatforms RuntimePlatform => runtimePlatform;
+
+ [SerializeField]
+ private BaseMixedRealityProfile configurationProfile;
+
+ ///
+ public BaseMixedRealityProfile Profile => configurationProfile;
+
+ ///
+ /// The configuration profile for the service.
+ ///
+ [Obsolete("Use the Profile property instead.")]
+ public BaseMixedRealityProfile ConfigurationProfile => configurationProfile;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityServiceConfiguration.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityServiceConfiguration.cs.meta
new file mode 100644
index 0000000..520358a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityServiceConfiguration.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a0d7ee77470e4f228ba6eb25fd883db8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealitySpatialObserverConfiguration.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealitySpatialObserverConfiguration.cs
new file mode 100644
index 0000000..ee1bfc3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealitySpatialObserverConfiguration.cs
@@ -0,0 +1,72 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ [Serializable]
+ public struct MixedRealitySpatialObserverConfiguration : IMixedRealityServiceConfiguration
+ {
+ [SerializeField]
+ [Implements(typeof(IMixedRealitySpatialAwarenessObserver), TypeGrouping.ByNamespaceFlat)]
+ private SystemType componentType;
+
+ ///
+ public SystemType ComponentType => componentType;
+
+ [SerializeField]
+ private string componentName;
+
+ ///
+ public string ComponentName => componentName;
+
+ [SerializeField]
+ private uint priority;
+
+ ///
+ public uint Priority => priority;
+
+ [SerializeField]
+ [EnumFlags]
+ private SupportedPlatforms runtimePlatform;
+
+ ///
+ public SupportedPlatforms RuntimePlatform => runtimePlatform;
+
+ [SerializeField]
+ private BaseSpatialAwarenessObserverProfile observerProfile;
+
+ ///
+ public BaseMixedRealityProfile Profile => observerProfile;
+
+ ///
+ /// Spatial Observer specific configuration profile.
+ ///
+ public BaseSpatialAwarenessObserverProfile ObserverProfile => observerProfile;
+
+ ///
+ /// Constructor.
+ ///
+ /// The of the observer.
+ /// The friendly name of the observer.
+ /// The load priority of the observer.
+ /// The runtime platform(s) supported by the observer.
+ /// The configuration profile for the observer.
+ public MixedRealitySpatialObserverConfiguration(
+ SystemType componentType,
+ string componentName,
+ uint priority,
+ SupportedPlatforms runtimePlatform,
+ BaseSpatialAwarenessObserverProfile configurationProfile)
+ {
+ this.componentType = componentType;
+ this.componentName = componentName;
+ this.priority = priority;
+ this.runtimePlatform = runtimePlatform;
+ this.observerProfile = configurationProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealitySpatialObserverConfiguration.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealitySpatialObserverConfiguration.cs.meta
new file mode 100644
index 0000000..ae4320c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealitySpatialObserverConfiguration.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a1988fca42a2f664b870efd7a39bb203
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityToolkitConfigurationProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityToolkitConfigurationProfile.cs
new file mode 100644
index 0000000..6eb3bee
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityToolkitConfigurationProfile.cs
@@ -0,0 +1,381 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Boundary;
+using Microsoft.MixedReality.Toolkit.CameraSystem;
+using Microsoft.MixedReality.Toolkit.Diagnostics;
+using Microsoft.MixedReality.Toolkit.Input;
+using Microsoft.MixedReality.Toolkit.SceneSystem;
+using Microsoft.MixedReality.Toolkit.SpatialAwareness;
+using Microsoft.MixedReality.Toolkit.Teleport;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+using UnityEngine.Serialization;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Configuration profile settings for the Mixed Reality Toolkit.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Toolkit Configuration Profile", fileName = "MixedRealityToolkitConfigurationProfile", order = (int)CreateProfileMenuItemIndices.Configuration)]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/configuration/mixed-reality-configuration-guide")]
+ public class MixedRealityToolkitConfigurationProfile : BaseMixedRealityProfile
+ {
+ #region Mixed Reality Toolkit configurable properties
+
+ [SerializeField]
+ [Tooltip("Experience Settings profile.")]
+ private MixedRealityExperienceSettingsProfile experienceSettingsProfile;
+
+ ///
+ /// Profile for configuring the experience settings of your project.
+ /// Determines whether your project targers AR/VR, the scale of your experience, and the height of the user where applicable
+ ///
+ public MixedRealityExperienceSettingsProfile ExperienceSettingsProfile
+ {
+ get { return experienceSettingsProfile; }
+ internal set { experienceSettingsProfile = value; }
+ }
+
+
+ [SerializeField]
+ [Tooltip("The scale of the Mixed Reality experience.")]
+ private ExperienceScale targetExperienceScale = ExperienceScale.Room;
+
+ ///
+ /// The desired the scale of the experience.
+ /// Profile for configuring the experience settings of your project.
+ /// Determines whether your project targers AR/VR, the scale of your experience, and the height of the user where applicable
+ ///
+ ///
+ /// The target experience scale is now configured via ExperienceSettingsProfile
+ ///
+ [Obsolete("The target experience scale is now configured via ExperienceSettingsProfile, please use the ExperienceSettingsProfile.TargetExperienceScale instead")]
+ public ExperienceScale TargetExperienceScale
+ {
+ get { return targetExperienceScale; }
+ set { targetExperienceScale = value; }
+ }
+
+ [SerializeField]
+ [FormerlySerializedAs("enableCameraProfile")]
+ [Tooltip("Enable the Camera System on Startup.")]
+ private bool enableCameraSystem = false;
+
+ ///
+ /// Enable and configure the Camera Profile for the Mixed Reality Toolkit
+ ///
+ public bool IsCameraSystemEnabled
+ {
+ get { return CameraProfile != null && cameraSystemType != null && cameraSystemType.Type != null && enableCameraSystem; }
+ internal set { enableCameraSystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Camera profile.")]
+ private MixedRealityCameraProfile cameraProfile;
+
+ ///
+ /// Profile for customizing your camera and quality settings based on if your
+ /// head mounted display (HMD) is a transparent device or an occluded device.
+ ///
+ public MixedRealityCameraProfile CameraProfile
+ {
+ get { return cameraProfile; }
+ internal set { cameraProfile = value; }
+ }
+
+ ///
+ /// Camera System class to instantiate at runtime.
+ ///
+ public SystemType CameraSystemType
+ {
+ get { return cameraSystemType; }
+ internal set { cameraSystemType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Camera System Class to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealityCameraSystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType cameraSystemType;
+
+ [SerializeField]
+ [Tooltip("Enable the Input System on Startup.")]
+ private bool enableInputSystem = false;
+
+ ///
+ /// Enable and configure the Input System component for the Mixed Reality Toolkit
+ ///
+ public bool IsInputSystemEnabled
+ {
+ get { return inputSystemProfile != null && inputSystemType != null && inputSystemType.Type != null && enableInputSystem; }
+ internal set { enableInputSystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Input System profile for setting wiring up events and actions to input devices.")]
+ private MixedRealityInputSystemProfile inputSystemProfile;
+
+ ///
+ /// Input System profile for configuring events and actions to input devices.
+ ///
+ public MixedRealityInputSystemProfile InputSystemProfile
+ {
+ get { return inputSystemProfile; }
+ internal set { inputSystemProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Input System class to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealityInputSystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType inputSystemType;
+
+ ///
+ /// Input System class to instantiate at runtime.
+ ///
+ public SystemType InputSystemType
+ {
+ get { return inputSystemType; }
+ internal set { inputSystemType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Enable the Boundary on Startup")]
+ private bool enableBoundarySystem = false;
+
+ ///
+ /// Enable and configure the boundary system.
+ ///
+ public bool IsBoundarySystemEnabled
+ {
+ get { return BoundarySystemSystemType != null && BoundarySystemSystemType.Type != null && enableBoundarySystem && boundaryVisualizationProfile != null; }
+ internal set { enableBoundarySystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Boundary system class to instantiate at runtime for legacy XR.")]
+ [Implements(typeof(IMixedRealityBoundarySystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType boundarySystemType = null;
+
+ [SerializeField]
+ [Tooltip("Boundary system class to instantiate at runtime for XR SDK.")]
+ [Implements(typeof(IMixedRealityBoundarySystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType xrsdkBoundarySystemType = null;
+
+ ///
+ /// Boundary system class to instantiate at runtime.
+ ///
+ public SystemType BoundarySystemSystemType => (XRSettingsUtilities.XRSDKEnabled && xrsdkBoundarySystemType?.Type != null) ? xrsdkBoundarySystemType : boundarySystemType;
+
+ [SerializeField]
+ [Tooltip("Profile for wiring up boundary visualization assets.")]
+ private MixedRealityBoundaryVisualizationProfile boundaryVisualizationProfile;
+
+ ///
+ /// Active profile for boundary visualization.
+ ///
+ public MixedRealityBoundaryVisualizationProfile BoundaryVisualizationProfile
+ {
+ get { return boundaryVisualizationProfile; }
+ internal set { boundaryVisualizationProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Enable the Teleport System on Startup")]
+ private bool enableTeleportSystem = false;
+
+ ///
+ /// Enable and configure the teleport system.
+ ///
+ public bool IsTeleportSystemEnabled
+ {
+ get { return teleportSystemType != null && teleportSystemType.Type != null && enableTeleportSystem; }
+ internal set { enableTeleportSystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Boundary System Class to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealityTeleportSystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType teleportSystemType;
+
+ ///
+ /// Teleport System class to instantiate at runtime.
+ ///
+ public SystemType TeleportSystemSystemType
+ {
+ get { return teleportSystemType; }
+ internal set { teleportSystemType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Enable the Spatial Awareness system on startup")]
+ private bool enableSpatialAwarenessSystem = false;
+
+ ///
+ /// Enable and configure the spatial awareness system.
+ ///
+ public bool IsSpatialAwarenessSystemEnabled
+ {
+ get { return spatialAwarenessSystemType != null && spatialAwarenessSystemType.Type != null && enableSpatialAwarenessSystem; }
+ internal set { enableSpatialAwarenessSystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Spatial Awareness System Class to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealitySpatialAwarenessSystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType spatialAwarenessSystemType;
+
+ ///
+ /// Spatial Awareness System class to instantiate at runtime.
+ ///
+ public SystemType SpatialAwarenessSystemSystemType
+ {
+ get { return spatialAwarenessSystemType; }
+ internal set { spatialAwarenessSystemType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Profile for configuring the spatial awareness system.")]
+ private MixedRealitySpatialAwarenessSystemProfile spatialAwarenessSystemProfile;
+
+ ///
+ /// Active profile for spatial awareness system
+ ///
+ public MixedRealitySpatialAwarenessSystemProfile SpatialAwarenessSystemProfile
+ {
+ get { return spatialAwarenessSystemProfile; }
+ internal set { spatialAwarenessSystemProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Profile for configuring diagnostic components.")]
+ private MixedRealityDiagnosticsProfile diagnosticsSystemProfile;
+
+ ///
+ /// Active profile for diagnostic configuration
+ ///
+ public MixedRealityDiagnosticsProfile DiagnosticsSystemProfile
+ {
+ get { return diagnosticsSystemProfile; }
+ internal set { diagnosticsSystemProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Enable diagnostic system")]
+ private bool enableDiagnosticsSystem = false;
+
+ ///
+ /// Is the Diagnostics System enabled?
+ ///
+ public bool IsDiagnosticsSystemEnabled
+ {
+ get { return enableDiagnosticsSystem && DiagnosticsSystemSystemType?.Type != null && diagnosticsSystemProfile != null; }
+ internal set { enableDiagnosticsSystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Diagnostics System class to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealityDiagnosticsSystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType diagnosticsSystemType;
+
+ ///
+ /// Diagnostics System Script File to instantiate at runtime
+ ///
+ public SystemType DiagnosticsSystemSystemType
+ {
+ get { return diagnosticsSystemType; }
+ internal set { diagnosticsSystemType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Profile for configuring scene system components.")]
+ private MixedRealitySceneSystemProfile sceneSystemProfile;
+
+ ///
+ /// Active profile for scene configuration
+ ///
+ public MixedRealitySceneSystemProfile SceneSystemProfile
+ {
+ get { return sceneSystemProfile; }
+ internal set { sceneSystemProfile = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Enable scene system")]
+ private bool enableSceneSystem = false;
+
+ ///
+ /// Is the Scene System enabled?
+ ///
+ public bool IsSceneSystemEnabled
+ {
+ get { return enableSceneSystem && SceneSystemSystemType?.Type != null && sceneSystemProfile != null; }
+ internal set { enableSceneSystem = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("Scene System class to instantiate at runtime.")]
+ [Implements(typeof(IMixedRealitySceneSystem), TypeGrouping.ByNamespaceFlat)]
+ private SystemType sceneSystemType;
+
+ ///
+ /// Scene System Script File to instantiate at runtime
+ ///
+ public SystemType SceneSystemSystemType
+ {
+ get { return sceneSystemType; }
+ internal set { sceneSystemType = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("All the additional non-required services registered with the Mixed Reality Toolkit.")]
+ private MixedRealityRegisteredServiceProvidersProfile registeredServiceProvidersProfile = null;
+
+ ///
+ /// All the additional non-required systems, features, and managers registered with the Mixed Reality Toolkit.
+ ///
+ public MixedRealityRegisteredServiceProvidersProfile RegisteredServiceProvidersProfile => registeredServiceProvidersProfile;
+
+ ///
+ /// If true, MRTK will generate components that let you to view the state of running services. These objects will not be generated at runtime.
+ ///
+ [Obsolete("Service inspectors will be removed in an upcoming release")]
+ public bool UseServiceInspectors
+ {
+ get { return useServiceInspectors; }
+ }
+
+ [Obsolete("Service inspectors will be removed in an upcoming release")]
+ [SerializeField]
+ [Tooltip("Deprecated: If true, MRTK will generate components that let you to view the state of running services. These objects will not be generated at runtime.")]
+ private bool useServiceInspectors = false;
+
+ ///
+ /// If true, MRTK will render the depth buffer as color. Only valid in editor.
+ ///
+ public bool RenderDepthBuffer
+ {
+ get { return renderDepthBuffer; }
+ }
+
+ [SerializeField]
+ [Tooltip("If true, MRTK will render the depth buffer as color. Only valid in editor.")]
+ private bool renderDepthBuffer = false;
+
+ ///
+ /// If true, verbose logging will be enabled for MRTK components.
+ ///
+ public bool EnableVerboseLogging
+ {
+ get { return enableVerboseLogging; }
+ set { enableVerboseLogging = value; }
+ }
+
+ [SerializeField]
+ [Tooltip("If true, verbose logging will be enabled for MRTK components.")]
+ private bool enableVerboseLogging = false;
+
+ #endregion Mixed Reality Toolkit configurable properties
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityToolkitConfigurationProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityToolkitConfigurationProfile.cs.meta
new file mode 100644
index 0000000..c1545a8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/MixedRealityToolkitConfigurationProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7612acbc1a4a4ed0afa5f4ccbe42bee4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics.meta
new file mode 100644
index 0000000..51cc593
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 7f75169e4f3e4dda85043966a2490b3b
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/ComparableRaycastResult.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/ComparableRaycastResult.cs
new file mode 100644
index 0000000..d0e96a4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/ComparableRaycastResult.cs
@@ -0,0 +1,19 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Physics
+{
+ public struct ComparableRaycastResult
+ {
+ public readonly int LayerMaskIndex;
+ public readonly RaycastResult RaycastResult;
+
+ public ComparableRaycastResult(RaycastResult raycastResult, int layerMaskIndex = 0)
+ {
+ RaycastResult = raycastResult;
+ LayerMaskIndex = layerMaskIndex;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/ComparableRaycastResult.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/ComparableRaycastResult.cs.meta
new file mode 100644
index 0000000..bb5b26d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/ComparableRaycastResult.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8e21752251bc4090a3f41b86244bcd8b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/FocusDetails.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/FocusDetails.cs
new file mode 100644
index 0000000..ffd66fa
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/FocusDetails.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Input;
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Physics
+{
+ ///
+ /// Contains information about which game object has the focus currently.
+ /// Also contains information about the normal of that point.
+ ///
+ public struct FocusDetails
+ {
+ ///
+ /// Distance along the ray until a hit, or until the end of the ray if no hit
+ ///
+ public float RayDistance { get; set; }
+
+ ///
+ /// The hit point of the raycast.
+ ///
+ public Vector3 Point { get; set; }
+
+ ///
+ /// The normal of the raycast.
+ ///
+ public Vector3 Normal { get; set; }
+
+ ///
+ /// The object hit by the last raycast.
+ ///
+ public GameObject Object { get; set; }
+
+ ///
+ /// The last raycast hit info.
+ ///
+ public MixedRealityRaycastHit LastRaycastHit { get; set; }
+
+ ///
+ /// The last raycast hit info for graphic raycast
+ ///
+ public RaycastResult LastGraphicsRaycastResult { get; set; }
+
+ public Vector3 PointLocalSpace { get; set; }
+ public Vector3 NormalLocalSpace { get; set; }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/FocusDetails.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/FocusDetails.cs.meta
new file mode 100644
index 0000000..4fdca78
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/FocusDetails.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d367ed81359c4a2cb435788d6e4b533b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/RayStep.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/RayStep.cs
new file mode 100644
index 0000000..cfd0b2a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/RayStep.cs
@@ -0,0 +1,181 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Physics
+{
+ [Serializable]
+ public struct RayStep
+ {
+ // Re-use static space to avoid additional allocation
+ private static Vector3 dist;
+ private static Vector3 dir;
+ private static Vector3 pos;
+
+ public RayStep(Vector3 origin, Vector3 terminus) : this()
+ {
+ UpdateRayStep(ref origin, ref terminus);
+
+ epsilon = 0.01f;
+ }
+
+ public Vector3 Origin { get; private set; }
+ public Vector3 Terminus { get; private set; }
+ public Vector3 Direction { get; private set; }
+
+ public float Length { get; private set; }
+
+ private readonly float epsilon;
+
+ public Vector3 GetPoint(float distance)
+ {
+ if (Length <= distance || Length == 0f)
+ {
+ return Origin;
+ }
+
+ pos.x = Origin.x + Direction.x * distance;
+ pos.y = Origin.y + Direction.y * distance;
+ pos.z = Origin.z + Direction.z * distance;
+
+ return pos;
+ }
+
+ ///
+ /// Update current raystep with new origin and terminus points.
+ /// Pass by ref to avoid unnecessary struct copy into function since values will be copied anyways locally
+ ///
+ /// beginning of raystep origin
+ /// end of raystep
+ public void UpdateRayStep(ref Vector3 origin, ref Vector3 terminus)
+ {
+ Origin = origin;
+ Terminus = terminus;
+
+ dist.x = Terminus.x - Origin.x;
+ dist.y = Terminus.y - Origin.y;
+ dist.z = Terminus.z - Origin.z;
+
+ Length = Mathf.Sqrt((dist.x * dist.x) + (dist.y * dist.y) + (dist.z * dist.z));
+
+ if (Length > 0)
+ {
+ dir.x = dist.x / Length;
+ dir.y = dist.y / Length;
+ dir.z = dist.z / Length;
+ }
+ else
+ {
+ dir = dist;
+ }
+
+ Direction = dir;
+ }
+
+ public void CopyRay(Ray ray, float rayLength)
+ {
+ Length = rayLength;
+ Origin = ray.origin;
+ Direction = ray.direction;
+
+ pos.x = Origin.x + Direction.x * Length;
+ pos.y = Origin.y + Direction.y * Length;
+ pos.z = Origin.z + Direction.z * Length;
+
+ Terminus = pos;
+ }
+
+ public bool Contains(Vector3 point)
+ {
+ dist.x = Origin.x - point.x;
+ dist.y = Origin.y - point.y;
+ dist.z = Origin.z - point.z;
+ float sqrMagOriginPoint = (dist.x * dist.x) + (dist.y * dist.y) + (dist.z * dist.z);
+
+ dist.x = point.x - Terminus.x;
+ dist.y = point.y - Terminus.y;
+ dist.z = point.z - Terminus.z;
+ float sqrMagPointTerminus = (dist.x * dist.x) + (dist.y * dist.y) + (dist.z * dist.z);
+
+ float sqrLength = Length * Length;
+ float sqrEpsilon = epsilon * epsilon;
+
+ return (sqrMagOriginPoint + sqrMagPointTerminus) - sqrLength > sqrEpsilon;
+ }
+
+ public static implicit operator Ray(RayStep r)
+ {
+ return new Ray(r.Origin, r.Direction);
+ }
+
+ #region static utility functions
+
+ ///
+ /// Returns a point along an array of RaySteps by distance
+ ///
+ public static Vector3 GetPointByDistance(RayStep[] steps, float distance)
+ {
+ Debug.Assert(steps != null);
+ Debug.Assert(steps.Length > 0);
+
+ float remainingDistance = 0;
+ RayStep rayStep = GetStepByDistance(steps, distance, ref remainingDistance);
+ if (remainingDistance > 0)
+ {
+ return Vector3.Lerp(rayStep.Origin, rayStep.Terminus, remainingDistance / rayStep.Length);
+ }
+ else
+ {
+ return rayStep.Terminus;
+ }
+ }
+
+ ///
+ /// Returns a RayStep along an array of RaySteps by distance
+ ///
+ public static RayStep GetStepByDistance(RayStep[] steps, float distance, ref float remainingDistance)
+ {
+ Debug.Assert(steps != null && steps.Length > 0);
+
+ float traveledDistance = 0;
+ float stepLength = 0;
+ RayStep currentStep = new RayStep();
+
+
+ foreach (var step in steps)
+ {
+ currentStep = step;
+ stepLength = step.Length;
+
+ if (distance > traveledDistance + stepLength)
+ {
+ traveledDistance += stepLength;
+ }
+ else
+ {
+ remainingDistance = Mathf.Clamp(distance - traveledDistance, 0f, stepLength);
+ return currentStep;
+ }
+ }
+
+ remainingDistance = 0;
+ return currentStep;
+ }
+
+ ///
+ /// Returns a direction along an array of RaySteps by distance
+ ///
+ public static Vector3 GetDirectionByDistance(RayStep[] steps, float distance)
+ {
+ Debug.Assert(steps != null);
+ Debug.Assert(steps.Length > 0);
+
+ float traveledDistance = 0;
+ return GetStepByDistance(steps, distance, ref traveledDistance).Direction;
+ }
+
+ #endregion
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/RayStep.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/RayStep.cs.meta
new file mode 100644
index 0000000..35510e4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/RayStep.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6f24b53b8c6c4029819e1fe282bdae74
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/SceneQueryType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/SceneQueryType.cs
new file mode 100644
index 0000000..33bbbdb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/SceneQueryType.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Physics
+{
+ ///
+ /// Defines the different scene query types. Mostly used by pointers.
+ ///
+ public enum SceneQueryType
+ {
+ ///
+ /// Use a simple raycast from a single point in a given direction.
+ ///
+ SimpleRaycast,
+
+ ///
+ /// Complex raycast from multiple points using a box collider.
+ ///
+ BoxRaycast,
+
+ ///
+ /// Use Sphere cast.
+ ///
+ SphereCast,
+
+ ///
+ /// Check for colliders within a specific radius.
+ ///
+ SphereOverlap
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/SceneQueryType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/SceneQueryType.cs.meta
new file mode 100644
index 0000000..ae332df
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/SceneQueryType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7fc42c41eca042b4e9f5ebebe595e099
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/TeleportSurfaceResult.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/TeleportSurfaceResult.cs
new file mode 100644
index 0000000..a932e4a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/TeleportSurfaceResult.cs
@@ -0,0 +1,16 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Physics
+{
+ [Serializable]
+ public enum TeleportSurfaceResult
+ {
+ None = 0,
+ Valid,
+ Invalid,
+ HotSpot,
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/TeleportSurfaceResult.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/TeleportSurfaceResult.cs.meta
new file mode 100644
index 0000000..7911bf6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Physics/TeleportSurfaceResult.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2e66db5ac6aa4ad898424ce73f433e80
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem.meta
new file mode 100644
index 0000000..97fbc92
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 12e18c40067f70c44b00e556c871441f
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/LightingSceneTransitionType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/LightingSceneTransitionType.cs
new file mode 100644
index 0000000..2f6a749
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/LightingSceneTransitionType.cs
@@ -0,0 +1,15 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ ///
+ /// Used by scene service to control how to transition from one lighting scene to another.
+ ///
+ public enum LightingSceneTransitionType
+ {
+ None, // Previous lighting scene is unloaded, new lighting scene is loaded. No transition.
+ FadeToBlack, // Previous lighting scene fades out to black. New lighting scene is loaded, then faded up from black. Useful for smooth transitions between locations.
+ CrossFade, // Previous lighting scene fades out as new lighting scene fades in. Useful for smooth transitions between lighting setups in the same location. NOTE: This will not work with different skyboxes.
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/LightingSceneTransitionType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/LightingSceneTransitionType.cs.meta
new file mode 100644
index 0000000..519ed63
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/LightingSceneTransitionType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 5e9c320c7cffbfd4291e04b3d3344fe3
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/MixedRealitySceneSystemProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/MixedRealitySceneSystemProfile.cs
new file mode 100644
index 0000000..75eec4f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/MixedRealitySceneSystemProfile.cs
@@ -0,0 +1,391 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+#if UNITY_EDITOR
+using UnityEditor;
+#endif
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ ///
+ /// Configuration profile settings for setting up scene system.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Scene System Profile", fileName = "MixedRealitySceneSystemProfile", order = (int)CreateProfileMenuItemIndices.SceneSystem)]
+ [MixedRealityServiceProfile(typeof(IMixedRealitySceneSystem))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/scene-system/scene-system-getting-started")]
+ public class MixedRealitySceneSystemProfile : BaseMixedRealityProfile
+ {
+ ///
+ /// Internal class used to cache lighting settings associated with a scene.
+ ///
+ [Serializable]
+ internal sealed class CachedLightingSettings
+ {
+ public string SceneName;
+ public RuntimeRenderSettings RenderSettings;
+ public RuntimeLightingSettings LightingSettings;
+ public RuntimeSunlightSettings SunlightSettings;
+ public DateTime TimeStamp;
+ }
+
+ public bool UseManagerScene { get { return useManagerScene && !managerScene.IsEmpty; } }
+
+ public bool UseLightingScene { get { return useLightingScene && lightingScenes.Count > 0; } }
+
+ public SceneInfo ManagerScene => managerScene;
+
+ public SceneInfo DefaultLightingScene { get { return lightingScenes[defaultLightingSceneIndex]; } }
+
+ public IEnumerable LightingScenes { get { return lightingScenes; } }
+
+ public IEnumerable ContentScenes { get { return contentScenes; } }
+
+ public IEnumerable ContentTags { get { return contentTags; } }
+
+ public int NumLightingScenes { get { return lightingScenes.Count; } }
+
+ public int NumContentScenes { get { return contentScenes.Count; } }
+
+ public IEnumerable PermittedLightingSceneComponentTypes
+ {
+ get
+ {
+ foreach (SystemType systemType in permittedLightingSceneComponentTypes) { yield return systemType.Type; }
+ }
+ }
+
+#if UNITY_EDITOR
+ public bool EditorManageBuildSettings => editorManageBuildSettings;
+
+ public bool EditorManageLoadedScenes => editorManageLoadedScenes;
+
+ public bool EditorEnforceSceneOrder => editorEnforceSceneOrder;
+
+ public bool EditorEnforceLightingSceneTypes => editorEnforceLightingSceneTypes;
+
+ public bool EditorLightingCacheOutOfDate => editorLightingCacheOutOfDate;
+
+ public bool EditorLightingCacheUpdateRequested { get; set; }
+#endif
+
+ [SerializeField]
+ private bool useManagerScene = true;
+
+ [SerializeField]
+ private SceneInfo managerScene = default(SceneInfo);
+
+ [SerializeField]
+ private bool useLightingScene = true;
+
+ [SerializeField]
+ private int defaultLightingSceneIndex = 0;
+
+ [SerializeField]
+ private List lightingScenes = new List();
+
+ [SerializeField]
+ private List contentScenes = new List();
+
+ [SerializeField]
+ private SystemType[] permittedLightingSceneComponentTypes = new SystemType[] {
+ new SystemType(typeof(Transform)),
+ new SystemType(typeof(GameObject)),
+ new SystemType(typeof(Light)),
+ new SystemType(typeof(ReflectionProbe)),
+ new SystemType(typeof(LightProbeGroup)),
+ new SystemType(typeof(LightProbeProxyVolume)),
+ };
+
+ // These will be hidden by the default inspector.
+ [SerializeField]
+ [Tooltip("Cached content tags found in your content scenes")]
+ private List contentTags = new List();
+
+ // These will be hidden by the default inspector.
+ [SerializeField]
+ [Tooltip("Cached lighting settings from your lighting scenes")]
+ private List cachedLightingSettings = new List();
+
+ #region editor settings
+
+ // CS414 is disabled during this section because these properties are being used in the editor
+ // scenario - when this file is build for player scenario, these serialized fields still exist
+ // but are not used.
+#pragma warning disable 414
+ [SerializeField]
+ [Tooltip("If true, the service will update your build settings automatically, ensuring that all manager, lighting and content scenes are added. Disable this if you want total control over build settings.")]
+ private bool editorManageBuildSettings = true;
+
+ [SerializeField]
+ [Tooltip("If true, the service will ensure manager scene is displayed first in scene hierarchy, followed by lighting and then content. Disable this if you want total control over scene hierarchy.")]
+ private bool editorEnforceSceneOrder = true;
+
+ [SerializeField]
+ [Tooltip("If true, service will ensure that manager scenes and lighting scenes are always loaded. Disable if you want total control over which scenes are loaded in editor.")]
+ private bool editorManageLoadedScenes = true;
+
+ [SerializeField]
+ [Tooltip("If true, service will ensure that only lighting-related components are allowed in lighting scenes. Disable if you want total control over the content of lighting scenes.")]
+ private bool editorEnforceLightingSceneTypes = true;
+
+ [SerializeField]
+ private bool editorLightingCacheOutOfDate = false;
+#pragma warning restore 414
+
+ #endregion
+
+ public bool GetLightingSceneSettings(
+ string lightingSceneName,
+ out SceneInfo lightingScene,
+ out RuntimeLightingSettings lightingSettings,
+ out RuntimeRenderSettings renderSettings,
+ out RuntimeSunlightSettings sunlightSettings)
+ {
+ lightingSettings = default(RuntimeLightingSettings);
+ renderSettings = default(RuntimeRenderSettings);
+ sunlightSettings = default(RuntimeSunlightSettings);
+ lightingScene = SceneInfo.Empty;
+
+ for (int i = 0; i < lightingScenes.Count; i++)
+ {
+ if (lightingScenes[i].Name == lightingSceneName)
+ {
+ lightingScene = lightingScenes[i];
+ break;
+ }
+ }
+
+ if (lightingScene.IsEmpty)
+ { // If we didn't find a lighting scene, don't bother looking for a cache
+ return false;
+ }
+
+ bool foundCache = false;
+ for (int i = 0; i < cachedLightingSettings.Count; i++)
+ {
+ CachedLightingSettings cache = cachedLightingSettings[i];
+ if (cache.SceneName == lightingSceneName)
+ {
+ lightingSettings = cache.LightingSettings;
+ renderSettings = cache.RenderSettings;
+ sunlightSettings = cache.SunlightSettings;
+ foundCache = true;
+ break;
+ }
+ }
+
+ return foundCache;
+ }
+
+ public IEnumerable GetContentSceneNamesByTag(string tag)
+ {
+ foreach (SceneInfo contentScene in contentScenes)
+ {
+ if (contentScene.Tag == tag)
+ yield return contentScene.Name;
+ }
+ }
+
+#if UNITY_EDITOR
+ #region validation
+ private void OnValidate()
+ {
+ if (Application.isPlaying || EditorApplication.isCompiling)
+ {
+ return;
+ }
+
+ bool saveChanges = false;
+
+ // Remove any duplicate entries from our lighting and content scene lists
+ saveChanges |= (RemoveOrClearDuplicateEntries(lightingScenes) || RemoveOrClearDuplicateEntries(contentScenes));
+
+ // Ensure that manager scenes are not contained in content or lighting scenes
+ saveChanges |= (UseManagerScene && (RemoveScene(lightingScenes, managerScene) || RemoveScene(contentScenes, managerScene)));
+
+ // Ensure that content scenes are not included in lighting scenes
+ saveChanges |= (UseLightingScene && RemoveScenes(lightingScenes, contentScenes));
+
+ // Build our content tags
+ List newContentTags = new List();
+ foreach (SceneInfo contentScene in contentScenes)
+ {
+ if (string.IsNullOrEmpty(contentScene.Tag))
+ {
+ continue;
+ }
+
+ if (contentScene.Tag == "Untagged")
+ {
+ continue;
+ }
+
+ if (!newContentTags.Contains(contentScene.Tag))
+ {
+ newContentTags.Add(contentScene.Tag);
+ }
+ }
+
+ // See if our content tags have changed
+ if (!contentTags.SequenceEqual(newContentTags))
+ {
+ contentTags = newContentTags;
+ saveChanges = true;
+ }
+
+ defaultLightingSceneIndex = Mathf.Clamp(defaultLightingSceneIndex, 0, lightingScenes.Count - 1);
+
+ if (saveChanges)
+ { // We need to tie this directly to lighting scenes somehow
+ editorLightingCacheOutOfDate = true;
+ // Make sure our changes are saved to disk!
+ AssetDatabase.Refresh();
+ EditorUtility.SetDirty(this);
+ AssetDatabase.SaveAssets();
+ }
+ }
+
+ ///
+ /// Clears cached lighting settings.
+ /// Used to ensure we don't end up with 'dead' cached data.
+ ///
+ public void ClearLightingCache()
+ {
+ cachedLightingSettings.Clear();
+ }
+
+ ///
+ /// Used to update the cached lighting / render settings.
+ /// Since extracting them is complex and requires scene loading, I thought it best to avoid having the profile do it.
+ ///
+ /// The scene these settings belong to.
+ public void SetLightingCache(SceneInfo sceneInfo, RuntimeLightingSettings lightingSettings, RuntimeRenderSettings renderSettings, RuntimeSunlightSettings sunlightSettings)
+ {
+ CachedLightingSettings settings = new CachedLightingSettings();
+ settings.SceneName = sceneInfo.Name;
+ settings.LightingSettings = lightingSettings;
+ settings.RenderSettings = renderSettings;
+ settings.SunlightSettings = sunlightSettings;
+ settings.TimeStamp = DateTime.Now;
+
+ cachedLightingSettings.Add(settings);
+
+ editorLightingCacheOutOfDate = false;
+ }
+
+ ///
+ /// Sets editorLightingCacheOutOfDate to true and saves the profile.
+ ///
+ public void SetLightingCacheDirty()
+ {
+ editorLightingCacheOutOfDate = true;
+
+ AssetDatabase.Refresh();
+ EditorUtility.SetDirty(this);
+ AssetDatabase.SaveAssets();
+ }
+
+ public DateTime GetEarliestLightingCacheTimestamp()
+ {
+ if (cachedLightingSettings.Count <= 0)
+ {
+ return DateTime.MinValue;
+ }
+
+ DateTime earliestTimeStamp = DateTime.MaxValue;
+ foreach (CachedLightingSettings settings in cachedLightingSettings)
+ {
+ if (settings.TimeStamp < earliestTimeStamp)
+ {
+ earliestTimeStamp = settings.TimeStamp;
+ }
+ }
+
+ return earliestTimeStamp;
+ }
+
+ private static bool RemoveScenes(List sceneList, List scenesToRemove)
+ {
+ bool changed = false;
+
+ for (int i = sceneList.Count - 1; i >= 0; i--)
+ {
+ if (sceneList[i].IsEmpty)
+ {
+ continue;
+ }
+
+ foreach (SceneInfo sceneToRemove in scenesToRemove)
+ {
+ if (sceneToRemove.IsEmpty)
+ {
+ continue;
+ }
+
+ if (sceneList[i].Asset == sceneToRemove.Asset)
+ {
+ Debug.LogWarning("Removing scene " + sceneToRemove.Name + " from scene list.");
+ sceneList[i] = SceneInfo.Empty;
+ changed = true;
+ break;
+ }
+ }
+ }
+
+ return changed;
+ }
+
+ private static bool RemoveScene(List sceneList, SceneInfo sceneToRemove)
+ {
+ bool changed = false;
+
+ for (int i = sceneList.Count - 1; i >= 0; i--)
+ {
+ if (sceneList[i].IsEmpty)
+ {
+ continue;
+ }
+
+ if (sceneList[i].Asset == sceneToRemove.Asset)
+ {
+ Debug.LogWarning("Removing manager scene " + sceneToRemove.Name + " from scene list.");
+ sceneList[i] = SceneInfo.Empty;
+ changed = true;
+ }
+ }
+
+ return changed;
+ }
+
+ private static bool RemoveOrClearDuplicateEntries(List sceneList)
+ {
+ HashSet scenePaths = new HashSet();
+ bool changed = false;
+
+ for (int i = 0; i < sceneList.Count; i++)
+ {
+ if (sceneList[i].IsEmpty)
+ {
+ continue;
+ }
+
+ if (!scenePaths.Add(sceneList[i].Path))
+ { // If we encounter a duplicate, just set it to empty.
+ // This will ensure we don't get duplicates when we add new elements to the array.
+ Debug.LogWarning("Found duplicate entry in scene list at " + i + ", removing");
+ sceneList[i] = SceneInfo.Empty;
+ changed = true;
+ }
+ }
+
+ return changed;
+ }
+ #endregion
+#endif
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/MixedRealitySceneSystemProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/MixedRealitySceneSystemProfile.cs.meta
new file mode 100644
index 0000000..ce12cb8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/MixedRealitySceneSystemProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b81dae1d46879b444aa9847f7961649f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeLightingSettings.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeLightingSettings.cs
new file mode 100644
index 0000000..a2bffac
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeLightingSettings.cs
@@ -0,0 +1,51 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ ///
+ /// A struct that mimics the lighting settings stored in a scene.
+ /// Used to store, retrieve and interpolate lighting settings.
+ /// Omits any editor-only settings.
+ ///
+ [Serializable]
+ public struct RuntimeLightingSettings
+ {
+ public float BounceScale;
+ public float IndirectOutputScale;
+ public float AlbedoBoost;
+ public MixedLightingMode EnvironmentLightingMode;
+ public bool EnableBakedLightmaps;
+ public bool EnabledRealtimeLightmaps;
+
+ ///
+ /// Lerps between two settings
+ ///
+ /// Value from 0 to 1
+ public static RuntimeLightingSettings Lerp(RuntimeLightingSettings from, RuntimeLightingSettings to, float t)
+ {
+ bool notStarted = t <= 0;
+ to.AlbedoBoost = Mathf.Lerp(from.AlbedoBoost, to.AlbedoBoost, t);
+ to.BounceScale = Mathf.Lerp(from.BounceScale, to.BounceScale, t);
+ to.EnableBakedLightmaps = notStarted ? from.EnableBakedLightmaps : to.EnableBakedLightmaps;
+ to.EnabledRealtimeLightmaps = notStarted ? from.EnabledRealtimeLightmaps : to.EnabledRealtimeLightmaps;
+ to.EnvironmentLightingMode = notStarted ? from.EnvironmentLightingMode : to.EnvironmentLightingMode;
+ to.IndirectOutputScale = Mathf.Lerp(from.IndirectOutputScale, to.IndirectOutputScale, t);
+ return to;
+ }
+
+ ///
+ /// Sets continuous settings to 'black' without changing any discrete features.
+ ///
+ public static RuntimeLightingSettings Black(RuntimeLightingSettings source)
+ {
+ source.AlbedoBoost = 0;
+ source.BounceScale = 0;
+ source.IndirectOutputScale = 0;
+ return source;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeLightingSettings.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeLightingSettings.cs.meta
new file mode 100644
index 0000000..b6fb80c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeLightingSettings.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b0148fadb9ada61458e61015e93f5848
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeRenderSettings.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeRenderSettings.cs
new file mode 100644
index 0000000..bc9eaf6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeRenderSettings.cs
@@ -0,0 +1,88 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+using UnityEngine.Rendering;
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ ///
+ /// A struct that mimics the render settings stored in a scene.
+ /// Used to store, retrieve and interpolate render settings.
+ /// Omits any editor-only settings, as well as some settings that are seldom used.
+ ///
+ [Serializable]
+ public struct RuntimeRenderSettings
+ {
+ public bool Fog;
+ public Color FogColor;
+ public FogMode FogMode;
+ public float FogDensity;
+ public float LinearFogStart;
+ public float LinearFogEnd;
+ public Color AmbientSkyColor;
+ public Color AmbientEquatorColor;
+ public Color AmbientGroundColor;
+ public Color AmbientLight;
+ public float AmbientIntensity;
+ public int AmbientMode;
+ public Color SubtractiveShadowColor;
+ public Material SkyboxMaterial;
+ public DefaultReflectionMode DefaultReflectionMode;
+ public int DefaultReflectionResolution;
+ public int ReflectionBounces;
+ public float ReflectionIntensity;
+ public Cubemap CustomReflection;
+ public bool UseRadianceAmbientProbe;
+
+ ///
+ /// Lerps between two settings
+ ///
+ /// Value from 0 to 1
+ public static RuntimeRenderSettings Lerp(RuntimeRenderSettings from, RuntimeRenderSettings to, float t)
+ {
+ bool notStarted = t <= 0;
+ to.AmbientEquatorColor = Color.Lerp(from.AmbientEquatorColor, to.AmbientEquatorColor, t);
+ to.AmbientGroundColor = Color.Lerp(from.AmbientGroundColor, to.AmbientGroundColor, t);
+ to.AmbientIntensity = Mathf.Lerp(from.AmbientIntensity, to.AmbientIntensity, t);
+ to.AmbientLight = Color.Lerp(from.AmbientLight, to.AmbientLight, t);
+ to.AmbientMode = notStarted ? from.AmbientMode : to.AmbientMode;
+ to.AmbientSkyColor = Color.Lerp(from.AmbientSkyColor, to.AmbientSkyColor, t);
+ to.CustomReflection = notStarted ? from.CustomReflection : to.CustomReflection;
+ to.DefaultReflectionMode = notStarted ? from.DefaultReflectionMode : to.DefaultReflectionMode;
+ to.DefaultReflectionResolution = notStarted ? from.DefaultReflectionResolution : to.DefaultReflectionResolution;
+ to.Fog = notStarted ? from.Fog : to.Fog;
+ to.FogColor = Color.Lerp(from.FogColor, to.FogColor, t);
+ to.FogDensity = Mathf.Lerp(from.FogDensity, to.FogDensity, t);
+ to.FogMode = notStarted ? from.FogMode : to.FogMode;
+ to.LinearFogEnd = Mathf.Lerp(from.LinearFogEnd, to.LinearFogEnd, t);
+ to.LinearFogStart = Mathf.Lerp(from.LinearFogStart, to.LinearFogStart, t);
+ to.ReflectionBounces = notStarted ? from.ReflectionBounces : to.ReflectionBounces;
+ to.ReflectionIntensity = Mathf.Lerp(from.ReflectionIntensity, to.ReflectionIntensity, t);
+ to.SkyboxMaterial = notStarted ? from.SkyboxMaterial : to.SkyboxMaterial;
+ to.SubtractiveShadowColor = Color.Lerp(from.SubtractiveShadowColor, to.SubtractiveShadowColor, t);
+ to.UseRadianceAmbientProbe = notStarted ? from.UseRadianceAmbientProbe : to.UseRadianceAmbientProbe;
+ return to;
+ }
+
+ ///
+ /// Sets continuous settings to 'black' without changing any discrete features.
+ ///
+ public static RuntimeRenderSettings Black(RuntimeRenderSettings source)
+ {
+ source.AmbientEquatorColor = Color.clear;
+ source.AmbientGroundColor = Color.clear;
+ source.AmbientIntensity = 0;
+ source.AmbientLight = Color.clear;
+ source.AmbientSkyColor = Color.clear;
+ source.FogColor = Color.clear;
+ source.FogDensity = 0;
+ source.LinearFogEnd = 0;
+ source.LinearFogStart = 0;
+ source.ReflectionIntensity = 0;
+ source.SubtractiveShadowColor = Color.clear;
+ return source;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeRenderSettings.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeRenderSettings.cs.meta
new file mode 100644
index 0000000..62a9799
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeRenderSettings.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: aaa90ba5cea178647b2aeb4c1bcc7440
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeSunlightSettings.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeSunlightSettings.cs
new file mode 100644
index 0000000..e7ff74d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeSunlightSettings.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ ///
+ /// Struct for storing directional sunlight settings stored in a scene.
+ ///
+ [Serializable]
+ public struct RuntimeSunlightSettings
+ {
+ public bool UseSunlight;
+ public Color Color;
+ public float Intensity;
+ public float XRotation;
+ public float YRotation;
+ public float ZRotation;
+
+ ///
+ /// Lerps between two settings
+ ///
+ /// Value from 0 to 1
+ public static RuntimeSunlightSettings Lerp(RuntimeSunlightSettings from, RuntimeSunlightSettings to, float t)
+ {
+ bool notStarted = t <= 0;
+ to.Color = Color.Lerp(from.Color, to.Color, t);
+ to.Intensity = Mathf.Lerp(from.Intensity, to.Intensity, t);
+ to.XRotation = Mathf.Lerp(from.XRotation, to.XRotation, t);
+ to.YRotation = Mathf.Lerp(from.YRotation, to.YRotation, t);
+ to.ZRotation = Mathf.Lerp(from.ZRotation, to.ZRotation, t);
+ to.UseSunlight = notStarted ? from.UseSunlight : to.UseSunlight;
+ return to;
+ }
+
+ ///
+ /// Sets continuous settings to 'black' without changing any discrete features.
+ ///
+ public static RuntimeSunlightSettings Black(RuntimeSunlightSettings source)
+ {
+ source.Color = Color.clear;
+ source.Intensity = 0;
+
+ return source;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeSunlightSettings.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeSunlightSettings.cs.meta
new file mode 100644
index 0000000..bcf0982
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/RuntimeSunlightSettings.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2a6317eb707db6d478a060aaaf845ee9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneActivationToken.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneActivationToken.cs
new file mode 100644
index 0000000..855d9b2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneActivationToken.cs
@@ -0,0 +1,29 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ ///
+ /// Used by scene system to control when newly loaded scenes are activated.
+ ///
+ public class SceneActivationToken
+ {
+ ///
+ /// When true, the operation is waiting on AllowSceneActivation to be set to true before proceeding.
+ ///
+ public bool ReadyToProceed { get; private set; } = false;
+
+ ///
+ /// Setting this to true grants permission for scene operation to activate loaded scenes.
+ ///
+ public bool AllowSceneActivation { get; set; } = false;
+
+ ///
+ /// Sets ReadyToProceed value
+ ///
+ public void SetReadyToProceed(bool readyToProceed)
+ {
+ ReadyToProceed = readyToProceed;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneActivationToken.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneActivationToken.cs.meta
new file mode 100644
index 0000000..5cb981f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneActivationToken.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6cb3aff58cf814344b557534ef336ff4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneInfo.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneInfo.cs
new file mode 100644
index 0000000..4dc63a9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneInfo.cs
@@ -0,0 +1,81 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.SceneSystem
+{
+ [Serializable]
+ public struct SceneInfo
+ {
+ public static SceneInfo Empty { get { return empty; } }
+ private static SceneInfo empty = default(SceneInfo);
+
+ ///
+ /// Scene asset is not set.
+ ///
+ public bool IsEmpty
+ {
+ get
+ {
+ return Asset == null;
+ }
+ }
+
+ ///
+ /// Returns true if the asset is not null and if the scene has a valid build index. Doesn't respect whether scene is enabled in build settings.
+ ///
+ public bool IsInBuildSettings
+ {
+ get
+ {
+ return Asset != null && Included;
+ }
+ }
+
+ public bool IsEnabled
+ {
+ get
+ {
+ return IsInBuildSettings & BuildIndex >= 0;
+ }
+ }
+
+ ///
+ /// Name of the scene. Set by the property drawer.
+ ///
+ public string Name;
+
+ ///
+ /// Path of the scene. Set by the property drawer.
+ ///
+ public string Path;
+
+ ///
+ /// True if scene is included in build (NOT necessarily enabled)
+ ///
+ public bool Included;
+
+ ///
+ /// Build index of the scene. If included in build settings and enabled, this will be a value greater than zero.
+ /// If not included or disabled, this will be -1
+ ///
+ public int BuildIndex;
+
+ ///
+ /// Optional tag used to load and unload scenes in groups.
+ ///
+#if UNITY_EDITOR
+ [TagProperty]
+#endif
+ public string Tag;
+
+#if UNITY_EDITOR
+ [SceneAssetReference]
+#endif
+ ///
+ /// SceneAsset reference. Since SceneAsset is an editor-only asset, we store an object reference instead.
+ ///
+ public UnityEngine.Object Asset;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneInfo.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneInfo.cs.meta
new file mode 100644
index 0000000..3764a50
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SceneSystem/SceneInfo.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e3b67e72d22c83344ba12cefe88200b0
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness.meta
new file mode 100644
index 0000000..7dc3116
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: cbb3f0c327f621645b5c28b4f44410db
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObject.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObject.cs
new file mode 100644
index 0000000..011b4f2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObject.cs
@@ -0,0 +1,37 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ public partial class BaseSpatialAwarenessObject : IMixedRealitySpatialAwarenessObject
+ {
+ ///
+ public int Id { get; set; }
+
+ ///
+ public GameObject GameObject { get; set; }
+
+ ///
+ public MeshRenderer Renderer { get; set; }
+
+ ///
+ /// The MeshFilter associated with this spatial object's renderer.
+ ///
+ public MeshFilter Filter { get; set; }
+
+ ///
+ public virtual void CleanObject()
+ {
+ // todo: consider if this should be virtual, and what params it should contain
+ }
+
+ ///
+ /// Constructor.
+ ///
+ protected BaseSpatialAwarenessObject()
+ {
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObject.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObject.cs.meta
new file mode 100644
index 0000000..88f7ff1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObject.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c10b9af3de3f9c14ca6462bec3d582e2
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObserverProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObserverProfile.cs
new file mode 100644
index 0000000..8617acd
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObserverProfile.cs
@@ -0,0 +1,61 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Abstract class that provides base profile information for Spatial Awareness Observers and their configuration
+ ///
+ [Serializable]
+ public abstract class BaseSpatialAwarenessObserverProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ [Tooltip("How should the observer behave at startup?")]
+ private AutoStartBehavior startupBehavior = AutoStartBehavior.AutoStart;
+
+ ///
+ /// Indicates if the observer is to start immediately or wait for manual startup.
+ ///
+ public AutoStartBehavior StartupBehavior => startupBehavior;
+
+ [SerializeField]
+ [Tooltip("Should the spatial observer remain in a fixed location?")]
+ private bool isStationaryObserver = false;
+
+ ///
+ /// Indicates whether or not the spatial observer is to remain in a fixed location.
+ ///
+ public bool IsStationaryObserver => isStationaryObserver;
+
+ [SerializeField]
+ [Tooltip("The dimensions of the spatial observer volume, in meters.")]
+ private Vector3 observationExtents = Vector3.one * 3;
+
+ ///
+ /// The size of the volume, in meters per axis, from which individual observations will be made.
+ ///
+ public Vector3 ObservationExtents => observationExtents;
+
+ [SerializeField]
+ [Tooltip("The shape of observation volume")]
+ private VolumeType observerVolumeType = VolumeType.AxisAlignedCube;
+
+ ///
+ /// The shape (ex: axis aligned cube) of the observation volume.
+ ///
+ public VolumeType ObserverVolumeType => observerVolumeType;
+
+ [SerializeField]
+ [Tooltip("How often, in seconds, should the spatial observer update?")]
+ private float updateInterval = 3.5f;
+
+ ///
+ /// The frequency, in seconds, at which the spatial observer updates.
+ ///
+ public float UpdateInterval => updateInterval;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObserverProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObserverProfile.cs.meta
new file mode 100644
index 0000000..3e63194
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/BaseSpatialAwarenessObserverProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 92a6c43827fff704ca802cc4777484ae
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental.meta
new file mode 100644
index 0000000..d0bf930
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 61b4c36149b20484c8004220a8267762
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental/SpatialAwarenessSceneObject.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental/SpatialAwarenessSceneObject.cs
new file mode 100644
index 0000000..0a18ee3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental/SpatialAwarenessSceneObject.cs
@@ -0,0 +1,151 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.SpatialAwareness;
+using System.Collections.Generic;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Experimental.SpatialAwareness
+{
+ ///
+ /// Object encapsulating the components of a spatial awareness scene object.
+ ///
+ public class SpatialAwarenessSceneObject : BaseSpatialAwarenessObject
+ {
+ ///
+ /// Constructor.
+ ///
+ private SpatialAwarenessSceneObject() : base() { }
+
+ ///
+ /// Creates a .
+ ///
+ ///
+ /// A SpatialAwarenessSceneObject containing the fields that describe the scene object.
+ ///
+ public static SpatialAwarenessSceneObject Create(
+ int id,
+ SpatialAwarenessSurfaceTypes surfaceType,
+ Vector3 position,
+ Quaternion rotation,
+ List quads,
+ List meshData
+ )
+ {
+ SpatialAwarenessSceneObject newObject = new SpatialAwarenessSceneObject
+ {
+ Id = id
+ };
+
+ newObject.SurfaceType = surfaceType;
+ newObject.Position = position;
+ newObject.Rotation = rotation;
+ newObject.Quads = quads;
+ newObject.Meshes = meshData;
+
+ return newObject;
+ }
+
+ ///
+ /// The world position for the scene object.
+ ///
+ public Vector3 Position
+ {
+ get;
+ private set;
+ }
+
+ ///
+ /// The world rotation for the scene object.
+ ///
+ public Quaternion Rotation
+ {
+ get;
+ private set;
+ }
+
+ ///
+ /// The list of quads associated with the scene object.
+ ///
+ public List Quads
+ {
+ get;
+ private set;
+ }
+
+ ///
+ /// The list of meshes associated with the scene object.
+ ///
+ public List Meshes
+ {
+ get;
+ private set;
+ }
+
+ ///
+ /// The surface type of the scene object.
+ ///
+ public SpatialAwarenessSurfaceTypes SurfaceType
+ {
+ get;
+ private set;
+ }
+
+ ///
+ /// Object encapsulating data of a mesh.
+ ///
+ public class MeshData
+ {
+ ///
+ /// Id of the mesh.
+ ///
+ public int Id { get; set; }
+
+ ///
+ /// Indices of the mesh.
+ ///
+ public int[] Indices { get; set; }
+
+ ///
+ /// Vertices of the mesh.
+ ///
+ public Vector3[] Vertices { get; set; }
+
+ ///
+ /// UVs of the mesh.
+ ///
+ public Vector2[] UVs { get; set; }
+
+ ///
+ /// The gameObject associated with the mesh.
+ ///
+ public GameObject GameObject { get; set; }
+ }
+
+ ///
+ /// Object encapsulating data of a quad.
+ ///
+ public class QuadData
+ {
+ ///
+ /// Id of the quad.
+ ///
+ public int Id { get; set; }
+
+ ///
+ /// Extents of the quad.
+ ///
+ public Vector2 Extents { get; set; }
+
+ ///
+ /// The occlusion mask of the quad.
+ ///
+ public byte[] OcclusionMask { get; set; }
+
+ ///
+ /// The gameObject associated with the quad.
+ ///
+ public GameObject GameObject { get; set; }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental/SpatialAwarenessSceneObject.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental/SpatialAwarenessSceneObject.cs.meta
new file mode 100644
index 0000000..eb97452
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/Experimental/SpatialAwarenessSceneObject.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7c8f04162227c0c4abfbee62a058608a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessMeshObserverProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessMeshObserverProfile.cs
new file mode 100644
index 0000000..0a6f608
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessMeshObserverProfile.cs
@@ -0,0 +1,141 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Physics;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Configuration profile settings for spatial awareness mesh observers.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Spatial Awareness Mesh Observer Profile", fileName = "MixedRealitySpatialAwarenessMeshObserverProfile", order = (int)CreateProfileMenuItemIndices.SpatialAwarenessMeshObserver)]
+ [MixedRealityServiceProfile(typeof(IMixedRealitySpatialAwarenessMeshObserver))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/spatial-awareness/configuring-spatial-awareness-mesh-observer")]
+ public class MixedRealitySpatialAwarenessMeshObserverProfile : BaseSpatialAwarenessObserverProfile
+ {
+ #region IMixedRealitySpatialAwarenessMeshObserver settings
+
+ [PhysicsLayer]
+ [SerializeField]
+ [Tooltip("Physics layer on which to set observed meshes.")]
+ private int meshPhysicsLayer = BaseSpatialObserver.DefaultSpatialAwarenessLayer;
+
+ ///
+ /// The Unity Physics Layer on which to set observed meshes.
+ ///
+ public int MeshPhysicsLayer => meshPhysicsLayer;
+
+ [SerializeField]
+ [Tooltip("Level of detail used when creating the mesh")]
+ private SpatialAwarenessMeshLevelOfDetail levelOfDetail = SpatialAwarenessMeshLevelOfDetail.Coarse;
+
+ ///
+ /// The level of detail used when creating the mesh.
+ ///
+ public SpatialAwarenessMeshLevelOfDetail LevelOfDetail => levelOfDetail;
+
+ [SerializeField]
+ [Tooltip("Level of detail, in triangles per cubic meter.\nIgnored unless LevelOfDetail is set to Custom.")]
+ private int trianglesPerCubicMeter = 0;
+
+ ///
+ /// The level of detail, in triangles per cubic meter, for the returned spatial mesh.
+ ///
+ /// This value is ignored, unless is set to Custom.
+ public int TrianglesPerCubicMeter => trianglesPerCubicMeter;
+
+ [SerializeField]
+ [Tooltip("Should normals be recalculated when a mesh is added or updated?")]
+ private bool recalculateNormals = true;
+
+ ///
+ /// Indicates if the spatial awareness system to generate normal for the returned meshes
+ /// as some platforms may not support returning normal along with the spatial mesh.
+ ///
+ public bool RecalculateNormals => recalculateNormals;
+
+ [SerializeField]
+ [Tooltip("How should spatial meshes be displayed?")]
+ private SpatialAwarenessMeshDisplayOptions displayOption = SpatialAwarenessMeshDisplayOptions.Visible;
+
+ ///
+ /// Indicates how the mesh subsystem is to display surface meshes within the application.
+ ///
+ public SpatialAwarenessMeshDisplayOptions DisplayOption => displayOption;
+
+ [SerializeField]
+ [Tooltip("Material to use when displaying observed meshes")]
+ private Material visibleMaterial = null;
+
+ ///
+ /// The material to be used when displaying observed meshes.
+ ///
+ public Material VisibleMaterial => visibleMaterial;
+
+ [SerializeField]
+ [Tooltip("Material to use when observed meshes should occlude other objects")]
+ private Material occlusionMaterial = null;
+
+ ///
+ /// The material to be used when observed meshes should occlude other objects.
+ ///
+ public Material OcclusionMaterial => occlusionMaterial;
+
+ [SerializeField]
+ [Tooltip("Optional physics material to apply to spatial mesh")]
+ private PhysicMaterial physicsMaterial = null;
+
+ public PhysicMaterial PhysicsMaterial => physicsMaterial;
+
+ [SerializeField]
+ [Tooltip("Optional prefab that is added to the runtime spatial mesh hierarchy. This prefab will only" +
+ " be instantiated and appended to the hierarchy in a build, the behavior associated with this property is not" +
+ " enabled in editor.\n" +
+ "\n" +
+ "Default Runtime Spatial Awareness Hierarchy: \n" +
+ "\n" +
+ "Spatial Awareness System \n" +
+ " Windows Mixed Reality Spatial Mesh Observer\n" +
+ " Spatial Mesh - ID\n" +
+ " Spatial Mesh - ID\n" +
+ " ...\n" +
+ "\n" +
+ "Runtime Spatial Awareness Hierarchy with this prefab:\n" +
+ "\n" +
+ "Spatial Awareness System \n" +
+ " Runtime Spatial Mesh Prefab\n" +
+ " Windows Mixed Reality Spatial Mesh Observer\n" +
+ " Spatial Mesh - ID\n" +
+ " Spatial Mesh - ID\n" +
+ " ...\n")]
+ private GameObject runtimeSpatialMeshPrefab = null;
+
+ ///
+ /// Optional prefab that is added to the runtime spatial mesh hierarchy. This prefab will only
+ /// be instantiated and appended to the hierarchy in a build, the behavior associated with this property is not
+ /// enabled in editor.
+ ///
+ /// Runtime Spatial Awareness Hierarchy without this prefab:
+ ///
+ /// Spatial Awareness System
+ /// Windows Mixed Reality Spatial Mesh Observer
+ /// Spatial Mesh - ID
+ /// Spatial Mesh - ID
+ /// ...
+ ///
+ /// Runtime Spatial Awareness Hierarchy with this prefab:
+ ///
+ /// Spatial Awareness System
+ /// Runtime Spatial Mesh Prefab
+ /// Windows Mixed Reality Spatial Mesh Observer
+ /// Spatial Mesh - ID
+ /// Spatial Mesh - ID
+ /// ...
+ ///
+ public GameObject RuntimeSpatialMeshPrefab => runtimeSpatialMeshPrefab;
+
+ #endregion IMixedRealitySpatialAwarenessMeshObserver settings
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessMeshObserverProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessMeshObserverProfile.cs.meta
new file mode 100644
index 0000000..599409e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessMeshObserverProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d561f5f963b45674aa423435fb820879
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessSystemProfile.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessSystemProfile.cs
new file mode 100644
index 0000000..4398e22
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessSystemProfile.cs
@@ -0,0 +1,26 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Configuration profile settings for spatial awareness mesh observers.
+ ///
+ [CreateAssetMenu(menuName = "Mixed Reality/Toolkit/Profiles/Mixed Reality Spatial Awareness System Profile", fileName = "MixedRealitySpatialAwarenessSystemProfile", order = (int)CreateProfileMenuItemIndices.SpatialAwareness)]
+ [MixedRealityServiceProfile(typeof(IMixedRealitySpatialAwarenessSystem))]
+ [HelpURL("https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/spatial-awareness/spatial-awareness-getting-started")]
+ public class MixedRealitySpatialAwarenessSystemProfile : BaseMixedRealityProfile
+ {
+ [SerializeField]
+ private MixedRealitySpatialObserverConfiguration[] observerConfigurations = System.Array.Empty();
+
+ public MixedRealitySpatialObserverConfiguration[] ObserverConfigurations
+ {
+ get { return observerConfigurations; }
+ internal set { observerConfigurations = value; }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessSystemProfile.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessSystemProfile.cs.meta
new file mode 100644
index 0000000..68b0d95
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/MixedRealitySpatialAwarenessSystemProfile.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 19f279aded72cb741b4de89a54359dd4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {fileID: 2800000, guid: 4f9f54f9478441228dea18a2c828cfc6, type: 3}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshLevelOfDetail.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshLevelOfDetail.cs
new file mode 100644
index 0000000..faeb6dc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshLevelOfDetail.cs
@@ -0,0 +1,40 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Enumeration defining levels of detail for the spatial awareness mesh subsystem.
+ ///
+ public enum SpatialAwarenessMeshLevelOfDetail
+ {
+ ///
+ /// The custom level of detail allows specifying a custom value for
+ /// TrianglesPerCubicMeter.
+ ///
+ Custom = -1,
+
+ ///
+ /// The coarse level of detail is well suited for identifying large
+ /// environmental features, such as floors and walls.
+ ///
+ Coarse = 0,
+
+ ///
+ /// The medium level of detail is often useful for experiences that
+ /// continually scan the environment (ex: a virtual pet).
+ ///
+ Medium,
+
+ ///
+ /// The fine level of detail is well suited for using as an occlusion
+ /// mesh.
+ ///
+ Fine,
+
+ ///
+ /// The unlimited level of detail requests meshes as detailed as possible from the device.
+ ///
+ Unlimited = 255
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshLevelOfDetail.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshLevelOfDetail.cs.meta
new file mode 100644
index 0000000..1d6565c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshLevelOfDetail.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 46f2975a9da9d1648849a773339b0732
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshObject.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshObject.cs
new file mode 100644
index 0000000..814d594
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshObject.cs
@@ -0,0 +1,114 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Object encapsulating the components of a spatial awareness mesh object.
+ ///
+ public class SpatialAwarenessMeshObject : BaseSpatialAwarenessObject
+ {
+ ///
+ /// When a mesh is created we will need to create a game object with a minimum
+ /// set of components to contain the mesh. These are the required component types.
+ ///
+ private static readonly Type[] RequiredMeshComponents =
+ {
+ typeof(MeshFilter),
+ typeof(MeshRenderer),
+ typeof(MeshCollider)
+ };
+
+ ///
+ /// The collider for the mesh object.
+ ///
+ public MeshCollider Collider { get; set; }
+
+ ///
+ /// Constructor.
+ ///
+ private SpatialAwarenessMeshObject() : base() { }
+
+ ///
+ /// Creates a .
+ ///
+ ///
+ /// SpatialMeshObject containing the fields that describe the mesh.
+ ///
+ public static SpatialAwarenessMeshObject Create(
+ Mesh mesh,
+ int layer,
+ string name,
+ int meshId,
+ GameObject meshParent = null)
+ {
+ SpatialAwarenessMeshObject newMesh = new SpatialAwarenessMeshObject
+ {
+ Id = meshId,
+ GameObject = new GameObject(name, RequiredMeshComponents)
+ {
+ layer = layer
+ }
+ };
+
+ /// Preserve local transform when attaching to parent.
+ newMesh.GameObject.transform.SetParent(meshParent != null ? meshParent.transform : null, false);
+
+ newMesh.Filter = newMesh.GameObject.GetComponent();
+ newMesh.Filter.sharedMesh = mesh;
+
+ newMesh.Renderer = newMesh.GameObject.GetComponent();
+
+ // Reset the surface mesh collider to fit the updated mesh.
+ // Unity tribal knowledge indicates that to change the mesh assigned to a
+ // mesh collider, the mesh must first be set to null. Presumably there
+ // is a side effect in the setter when setting the shared mesh to null.
+ newMesh.Collider = newMesh.GameObject.GetComponent();
+ newMesh.Collider.sharedMesh = null;
+ newMesh.Collider.sharedMesh = newMesh.Filter.sharedMesh;
+
+ return newMesh;
+ }
+
+ ///
+ /// Clean up the resources associated with the surface.
+ ///
+ /// The whose resources will be cleaned up.
+ public static void Cleanup(SpatialAwarenessMeshObject meshObject, bool destroyGameObject = true, bool destroyMeshes = true)
+ {
+ if (meshObject.GameObject == null)
+ {
+ return;
+ }
+
+ if (destroyGameObject)
+ {
+ UnityEngine.Object.Destroy(meshObject.GameObject);
+ meshObject.GameObject = null;
+ return;
+ }
+
+ if (destroyMeshes)
+ {
+ Mesh filterMesh = meshObject.Filter.sharedMesh;
+ Mesh colliderMesh = meshObject.Collider.sharedMesh;
+
+ if (filterMesh != null)
+ {
+ UnityEngine.Object.Destroy(filterMesh);
+ meshObject.Filter.sharedMesh = null;
+ }
+
+ if ((colliderMesh != null) && (colliderMesh != filterMesh))
+ {
+ UnityEngine.Object.Destroy(colliderMesh);
+ meshObject.Collider.sharedMesh = null;
+ }
+ }
+ }
+
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshObject.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshObject.cs.meta
new file mode 100644
index 0000000..e56958e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessMeshObject.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 90274f04089fd9f48bf49eefd1fa16c8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessPlanarObject.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessPlanarObject.cs
new file mode 100644
index 0000000..ad16d71
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessPlanarObject.cs
@@ -0,0 +1,68 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ public partial class SpatialAwarenessPlanarObject : BaseSpatialAwarenessObject
+ {
+ ///
+ /// The BoxCollider associated with this plane's GameObject.
+ ///
+ public BoxCollider Collider { get; set; }
+
+ private SpatialAwarenessSurfaceTypes planeType = SpatialAwarenessSurfaceTypes.Unknown;
+
+ ///
+ /// The type of surface (ex: wall) represented by this object.
+ ///
+ public SpatialAwarenessSurfaceTypes SurfaceType
+ {
+ get => planeType;
+ private set => planeType = value;
+ }
+
+ ///
+ /// Constructor.
+ ///
+ private SpatialAwarenessPlanarObject() : base() { }
+
+ ///
+ /// Creates a .
+ ///
+ ///
+ /// SpatialAwarenessPlanarObject containing the fields that describe the plane.
+ ///
+ public static SpatialAwarenessPlanarObject CreateSpatialObject(
+ Vector3 center,
+ Vector3 size,
+ Quaternion rotation,
+ int layer,
+ string name,
+ int planeId,
+ SpatialAwarenessSurfaceTypes surfaceType = SpatialAwarenessSurfaceTypes.Unknown)
+ {
+ SpatialAwarenessPlanarObject newPlane = new SpatialAwarenessPlanarObject();
+
+ newPlane.Id = planeId;
+ newPlane.SurfaceType = surfaceType;
+
+
+ GameObject planeObject = GameObject.CreatePrimitive(PrimitiveType.Cube);
+ planeObject.transform.position = center;
+ planeObject.transform.rotation = rotation;
+ planeObject.transform.localScale = size;
+ planeObject.name = name;
+ planeObject.layer = layer;
+
+ newPlane.GameObject = planeObject;
+ newPlane.Filter = newPlane.GameObject.GetComponent();
+ newPlane.Renderer = newPlane.GameObject.GetComponent();
+ newPlane.Renderer.shadowCastingMode = UnityEngine.Rendering.ShadowCastingMode.Off;
+ newPlane.Collider = newPlane.GameObject.GetComponent();
+
+ return newPlane;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessPlanarObject.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessPlanarObject.cs.meta
new file mode 100644
index 0000000..b85141e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessPlanarObject.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2d9842a78541d2045aad3a8161195c62
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessSurfaceTypes.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessSurfaceTypes.cs
new file mode 100644
index 0000000..d714735
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessSurfaceTypes.cs
@@ -0,0 +1,85 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Enumeration defining the types of planar surfaces supported by Spatial Awareness.
+ ///
+ [System.Flags]
+ public enum SpatialAwarenessSurfaceTypes
+ {
+ ///
+ /// A surface that cannot yet be categorized.
+ ///
+ ///
+ /// Unknown should not be confused with Background. Unknown surfaces may
+ /// have no classification or there may not yet be enough data to assign
+ /// a surface type. Additional environmental scanning may provide the necessary
+ /// data to classify the surface.
+ ///
+ Unknown = 1 << 0,
+
+ ///
+ /// The environment’s floor.
+ ///
+ Floor = 1 << 1,
+
+ ///
+ /// The environment’s ceiling.
+ ///
+ Ceiling = 1 << 2,
+
+ ///
+ /// A vertical surface within the user’s space.
+ ///
+ Wall = 1 << 3,
+
+ ///
+ /// A large, raised surface upon which objects can be placed.
+ ///
+ ///
+ /// Platforms can represent tables, countertops, shelves or other horizontal surfaces.
+ ///
+ Platform = 1 << 4,
+
+ ///
+ /// A surface that does not fit one of the defined surface types.
+ ///
+ ///
+ /// Platforms, like floors, that can be used for placing objects requiring a horizontal surface.
+ /// Background should not be confused with Unknown. There is sufficient data to
+ /// classify the surface and it has been found to not correspond to a defined type.
+ ///
+ Background = 1 << 5,
+
+ ///
+ /// A boundless world mesh.
+ ///
+ World = 1 << 6,
+
+ ///
+ /// Objects for which we have no observations
+ ///
+ Inferred = 1 << 7
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class SpatialAwarenessSurfaceTypesExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this SpatialAwarenessSurfaceTypes a, SpatialAwarenessSurfaceTypes b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessSurfaceTypes.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessSurfaceTypes.cs.meta
new file mode 100644
index 0000000..8fbb5de
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialAwarenessSurfaceTypes.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4928868ecc4076d40880a37740b8e492
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialMeshDisplayOptions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialMeshDisplayOptions.cs
new file mode 100644
index 0000000..567fed5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialMeshDisplayOptions.cs
@@ -0,0 +1,26 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Options for how the spatial mesh is to be displayed by the spatial awareness system.
+ ///
+ public enum SpatialAwarenessMeshDisplayOptions
+ {
+ ///
+ /// Do not display the spatial mesh
+ ///
+ None = 0,
+
+ ///
+ /// Display the spatial mesh using the configured material
+ ///
+ Visible,
+
+ ///
+ /// Display the spatial mesh using the configured occlusion material
+ ///
+ Occlusion
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialMeshDisplayOptions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialMeshDisplayOptions.cs.meta
new file mode 100644
index 0000000..ec41f7c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/SpatialAwareness/SpatialMeshDisplayOptions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 63d8cc27a28ed1946a4690c625b93806
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities.meta
new file mode 100644
index 0000000..250831a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: f64522f38b6743178c98c60d303aeb23
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AnimatorParameter.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AnimatorParameter.cs
new file mode 100644
index 0000000..4b105e2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AnimatorParameter.cs
@@ -0,0 +1,97 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// A copy of the AnimatorControllerParameter because that class is not Serializable and cannot be modified in the editor.
+ ///
+ [Serializable]
+ public struct AnimatorParameter
+ {
+ ///
+ /// Constructor.
+ ///
+ /// Name of the animation parameter to modify.
+ /// Type of the animation parameter to modify.
+ /// If the animation parameter type is an int, value to set. Ignored otherwise.
+ /// If the animation parameter type is a float, value to set. Ignored otherwise.
+ /// "If the animation parameter type is a bool, value to set. Ignored otherwise.
+ public AnimatorParameter(string name, AnimatorControllerParameterType parameterType, int defaultInt = 0, float defaultFloat = 0f, bool defaultBool = false)
+ {
+ this.parameterType = parameterType;
+ this.defaultInt = defaultInt;
+ this.defaultFloat = defaultFloat;
+ this.defaultBool = defaultBool;
+ this.name = name;
+ nameStringHash = null;
+ }
+
+ [SerializeField]
+ [Tooltip("Type of the animation parameter to modify.")]
+ private AnimatorControllerParameterType parameterType;
+
+ ///
+ /// Type of the animation parameter to modify.
+ ///
+ public AnimatorControllerParameterType ParameterType => parameterType;
+
+ [SerializeField]
+ [Tooltip("If the animation parameter type is an int, value to set. Ignored otherwise.")]
+ private int defaultInt;
+
+ ///
+ /// If the animation parameter type is an int, value to set. Ignored otherwise.
+ ///
+ public int DefaultInt => defaultInt;
+
+ [SerializeField]
+ [Tooltip("If the animation parameter type is a float, value to set. Ignored otherwise.")]
+ private float defaultFloat;
+
+ ///
+ /// If the animation parameter type is a float, value to set. Ignored otherwise.
+ ///
+ public float DefaultFloat => defaultFloat;
+
+ [SerializeField]
+ [Tooltip("If the animation parameter type is a bool, value to set. Ignored otherwise.")]
+ private bool defaultBool;
+
+ ///
+ /// If the animation parameter type is a bool, value to set. Ignored otherwise.
+ ///
+ public bool DefaultBool => defaultBool;
+
+ [SerializeField]
+ [Tooltip("Name of the animation parameter to modify.")]
+ private string name;
+
+ ///
+ /// Name of the animation parameter to modify.
+ ///
+ public string Name => name;
+
+ private int? nameStringHash;
+
+ ///
+ /// Animator Name String to Hash.
+ ///
+ public int NameHash
+ {
+ get
+ {
+ if (!nameStringHash.HasValue && !string.IsNullOrEmpty(Name))
+ {
+ nameStringHash = Animator.StringToHash(Name);
+ }
+
+ Debug.Assert(nameStringHash != null);
+ return nameStringHash.Value;
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AnimatorParameter.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AnimatorParameter.cs.meta
new file mode 100644
index 0000000..4eebcd7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AnimatorParameter.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 5a06e28a39fe4c848d5c994848297c41
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ArticulatedHandPose.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ArticulatedHandPose.cs
new file mode 100644
index 0000000..8521db2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ArticulatedHandPose.cs
@@ -0,0 +1,368 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections.Generic;
+using System.IO;
+using UnityEngine;
+
+#if UNITY_EDITOR
+using UnityEditor;
+#endif
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Shape of an articulated hand defined by joint poses.
+ ///
+ public class ArticulatedHandPose
+ {
+ private static readonly TrackedHandJoint[] Joints = Enum.GetValues(typeof(TrackedHandJoint)) as TrackedHandJoint[];
+
+ ///
+ /// Represents the maximum number of tracked hand joints.
+ ///
+ public static int JointCount { get; } = Joints.Length;
+
+ ///
+ /// Joint poses are stored as right-hand poses in camera space.
+ /// Output poses are computed in world space, and mirroring on the x axis for the left hand.
+ ///
+ private readonly MixedRealityPose[] localJointPoses;
+
+ public ArticulatedHandPose()
+ {
+ localJointPoses = new MixedRealityPose[JointCount];
+ SetZero();
+ }
+
+ public ArticulatedHandPose(MixedRealityPose[] localJointPoses)
+ {
+ this.localJointPoses = new MixedRealityPose[JointCount];
+ Array.Copy(localJointPoses, this.localJointPoses, JointCount);
+ }
+
+ public MixedRealityPose GetLocalJointPose(TrackedHandJoint joint, Handedness handedness)
+ {
+ MixedRealityPose pose = localJointPoses[(int)joint];
+
+ // Pose offset are for right hand, mirror on X axis if left hand is needed
+ if (handedness == Handedness.Left)
+ {
+ pose = new MixedRealityPose(
+ new Vector3(-pose.Position.x, pose.Position.y, pose.Position.z),
+ new Quaternion(pose.Rotation.x, -pose.Rotation.y, -pose.Rotation.z, pose.Rotation.w));
+ }
+
+ return pose;
+ }
+
+ ///
+ /// Compute world space poses from camera-space joint data.
+ ///
+ /// Handedness of the resulting pose
+ /// Rotational offset of the resulting pose
+ /// Translational offset of the resulting pose
+ /// Output array of joint poses
+ public void ComputeJointPoses(
+ Handedness handedness,
+ Quaternion rotation,
+ Vector3 position,
+ MixedRealityPose[] jointsOut)
+ {
+ for (int i = 0; i < JointCount; i++)
+ {
+ // Initialize from local offsets
+ MixedRealityPose pose = GetLocalJointPose(Joints[i], handedness);
+ Vector3 p = pose.Position;
+ Quaternion r = pose.Rotation;
+
+ // Apply external transform
+ p = position + rotation * p;
+ r = rotation * r;
+
+ jointsOut[i] = new MixedRealityPose(p, r);
+ }
+ }
+
+ ///
+ /// Take world space joint poses from any hand and convert into right-hand, camera-space poses.
+ ///
+ /// Input joint poses
+ /// Handedness of the input data
+ /// Rotational offset of the input data
+ /// Translational offset of the input data
+ public void ParseFromJointPoses(
+ MixedRealityPose[] joints,
+ Handedness handedness,
+ Quaternion rotation,
+ Vector3 position)
+ {
+ Quaternion invRotation = Quaternion.Inverse(rotation);
+ Quaternion invCameraRotation = Quaternion.Inverse(CameraCache.Main.transform.rotation);
+
+ for (int i = 0; i < JointCount; i++)
+ {
+ Vector3 p = joints[i].Position;
+ Quaternion r = joints[i].Rotation;
+
+ // Apply inverse external transform
+ p = invRotation * (p - position);
+ r = invRotation * r;
+
+ // To camera space
+ p = invCameraRotation * p;
+ r = invCameraRotation * r;
+
+ // Pose offset are for right hand, mirror on X axis if left hand is given
+ if (handedness == Handedness.Left)
+ {
+ p.x = -p.x;
+ r.y = -r.y;
+ r.z = -r.z;
+ }
+
+ localJointPoses[i] = new MixedRealityPose(p, r);
+ }
+ }
+
+ ///
+ /// Set all poses to zero.
+ ///
+ public void SetZero()
+ {
+ for (int i = 0; i < JointCount; i++)
+ {
+ localJointPoses[i] = MixedRealityPose.ZeroIdentity;
+ }
+ }
+
+ ///
+ /// Copy data from another articulated hand pose.
+ ///
+ public void Copy(ArticulatedHandPose other)
+ {
+ Array.Copy(other.localJointPoses, localJointPoses, JointCount);
+ }
+
+ ///
+ /// Blend between two hand poses.
+ ///
+ public void InterpolateOffsets(ArticulatedHandPose poseA, ArticulatedHandPose poseB, float value)
+ {
+ for (int i = 0; i < JointCount; i++)
+ {
+ var p = Vector3.Lerp(poseA.localJointPoses[i].Position, poseB.localJointPoses[i].Position, value);
+ var r = Quaternion.Slerp(poseA.localJointPoses[i].Rotation, poseB.localJointPoses[i].Rotation, value);
+ localJointPoses[i] = new MixedRealityPose(p, r);
+ }
+ }
+
+ ///
+ /// Supported hand gestures.
+ ///
+ public enum GestureId
+ {
+ ///
+ /// Unspecified hand shape
+ ///
+ None = 0,
+ ///
+ /// Flat hand with fingers spread out
+ ///
+ Flat,
+ ///
+ /// Relaxed hand pose
+ ///
+ Open,
+ ///
+ /// Index finger and Thumb touching, grab point does not move
+ ///
+ Pinch,
+ ///
+ /// Index finger and Thumb touching, wrist does not move
+ ///
+ PinchSteadyWrist,
+ ///
+ /// Index finger stretched out
+ ///
+ Poke,
+ ///
+ /// Grab with whole hand, fist shape
+ ///
+ Grab,
+ ///
+ /// OK sign
+ ///
+ ThumbsUp,
+ ///
+ /// Victory sign
+ ///
+ Victory,
+ ///
+ /// Relaxed hand pose, grab point does not move
+ ///
+ OpenSteadyGrabPoint,
+ ///
+ /// Hand facing upwards, Index and Thumb stretched out to start a teleport
+ ///
+ TeleportStart,
+ ///
+ /// Hand facing upwards, Index curled in to finish a teleport
+ ///
+ TeleportEnd,
+ }
+
+ [Obsolete("Use SimulatedArticulatedHandPoses class or other custom class")]
+ private static readonly Dictionary handPoses = new Dictionary();
+
+ ///
+ /// Get pose data for a supported gesture.
+ ///
+ [Obsolete("Use SimulatedArticulatedHandPoses.GetGesturePose() or other custom class")]
+ public static ArticulatedHandPose GetGesturePose(GestureId gesture)
+ {
+ if (handPoses.TryGetValue(gesture, out ArticulatedHandPose pose))
+ {
+ return pose;
+ }
+ return null;
+ }
+
+#if UNITY_EDITOR
+ ///
+ /// Load pose data from files.
+ ///
+ [Obsolete("Use SimulatedArticulatedHandPoses or other custom class")]
+ public static void LoadGesturePoses()
+ {
+ string[] gestureNames = Enum.GetNames(typeof(GestureId));
+ for (int i = 0; i < gestureNames.Length; ++i)
+ {
+ string gestureFileName = string.Format("ArticulatedHandPose_{0}", gestureNames[i]);
+ string[] gestureGuids = AssetDatabase.FindAssets(gestureFileName);
+ string gesturePath = string.Empty;
+ foreach (string guid in gestureGuids)
+ {
+ string tempPath = AssetDatabase.GUIDToAssetPath(guid);
+ if (tempPath.Contains("InputSimulation")
+ && tempPath.Contains("ArticulatedHandPoses")
+ && tempPath.Contains(gestureFileName + ".json"))
+ {
+ gesturePath = tempPath;
+ break;
+ }
+ }
+
+ if (!string.IsNullOrWhiteSpace(gesturePath))
+ {
+ LoadGesturePose((GestureId)i, gesturePath);
+ }
+ }
+ }
+
+ [Obsolete("Use SimulatedArticulatedHandPoses class or other custom class")]
+ private static ArticulatedHandPose LoadGesturePose(GestureId gesture, string filePath)
+ {
+ if (!string.IsNullOrWhiteSpace(filePath))
+ {
+ var pose = new ArticulatedHandPose();
+ pose.FromJson(File.ReadAllText(filePath));
+ handPoses.Add(gesture, pose);
+ return pose;
+ }
+ return null;
+ }
+
+ [Obsolete("Use SimulatedArticulatedHandPoses class or other custom class")]
+ public static void ResetGesturePoses()
+ {
+ handPoses.Clear();
+ }
+#endif
+
+ /// Utility class to serialize hand pose as a dictionary with full joint names
+ [Serializable]
+ internal struct ArticulatedHandPoseItem
+ {
+ private static readonly string[] jointNames = Enum.GetNames(typeof(TrackedHandJoint));
+
+ public string joint;
+ public MixedRealityPose pose;
+
+ public TrackedHandJoint JointIndex
+ {
+ get
+ {
+ int nameIndex = Array.FindIndex(jointNames, IsJointName);
+ if (nameIndex < 0)
+ {
+ Debug.LogError($"Joint name {joint} not in TrackedHandJoint enum");
+ return TrackedHandJoint.None;
+ }
+ return (TrackedHandJoint)nameIndex;
+ }
+ set { joint = jointNames[(int)value]; }
+ }
+
+ private bool IsJointName(string s)
+ {
+ return s == joint;
+ }
+
+ public ArticulatedHandPoseItem(TrackedHandJoint joint, MixedRealityPose pose)
+ {
+ this.joint = jointNames[(int)joint];
+ this.pose = pose;
+ }
+ }
+
+ /// Utility class to serialize hand pose as a dictionary with full joint names
+ [Serializable]
+ internal class ArticulatedHandPoseDictionary
+ {
+ public ArticulatedHandPoseItem[] items = null;
+
+ public void FromJointPoses(MixedRealityPose[] jointPoses)
+ {
+ items = new ArticulatedHandPoseItem[JointCount];
+ for (int i = 0; i < JointCount; ++i)
+ {
+ items[i].JointIndex = (TrackedHandJoint)i;
+ items[i].pose = jointPoses[i];
+ }
+ }
+
+ public void ToJointPoses(MixedRealityPose[] jointPoses)
+ {
+ for (int i = 0; i < JointCount; ++i)
+ {
+ jointPoses[i] = MixedRealityPose.ZeroIdentity;
+ }
+ foreach (var item in items)
+ {
+ jointPoses[(int)item.JointIndex] = item.pose;
+ }
+ }
+ }
+
+ ///
+ /// Serialize pose data to JSON format.
+ ///
+ public string ToJson()
+ {
+ var dict = new ArticulatedHandPoseDictionary();
+ dict.FromJointPoses(localJointPoses);
+ return JsonUtility.ToJson(dict, true);
+ }
+
+ ///
+ /// Deserialize pose data from JSON format.
+ ///
+ public void FromJson(string json)
+ {
+ var dict = JsonUtility.FromJson(json);
+ dict.ToJointPoses(localJointPoses);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ArticulatedHandPose.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ArticulatedHandPose.cs.meta
new file mode 100644
index 0000000..50ecd4e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ArticulatedHandPose.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 521d0baa937716a4fbcfed7bd6bbde0e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AutoStartBehavior.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AutoStartBehavior.cs
new file mode 100644
index 0000000..b9d2ea9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AutoStartBehavior.cs
@@ -0,0 +1,22 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// This enumeration identifies two different ways to handle the startup behavior for a feature.
+ /// Both will warm up the component, ready for its use (e.g. connecting backend services or registering for events.
+ /// The first causes the feature to start immediately. The second allows the feature to be manually started at a later time.
+ ///
+ public enum AutoStartBehavior
+ {
+ ///
+ /// Automatically start the feature
+ ///
+ AutoStart = 0,
+ ///
+ /// Delay the start of the feature until the user requests it to begin
+ ///
+ ManualStart
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AutoStartBehavior.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AutoStartBehavior.cs.meta
new file mode 100644
index 0000000..db8a3ce
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AutoStartBehavior.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 22ae876eef4f0e647a461b56ef37893e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisFlags.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisFlags.cs
new file mode 100644
index 0000000..adc6654
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisFlags.cs
@@ -0,0 +1,35 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Flags used to represent a set of 3D axes
+ ///
+ [System.Flags]
+ public enum AxisFlags
+ {
+ XAxis = 1 << 0,
+ YAxis = 1 << 1,
+ ZAxis = 1 << 2
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class AxisFlagsExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this AxisFlags a, AxisFlags b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisFlags.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisFlags.cs.meta
new file mode 100644
index 0000000..6b2cf43
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisFlags.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 0b4e4a9d78a31d044857f2fdabe5a4fc
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisType.cs
new file mode 100644
index 0000000..a1b8587
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisType.cs
@@ -0,0 +1,45 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The AxisType identifies the type of button or input being sent to the framework from a controller.
+ /// This is mainly information only or for advanced users to understand the input coming directly from the controller.
+ ///
+ public enum AxisType
+ {
+ ///
+ /// No Specified type.
+ ///
+ None = 0,
+ ///
+ /// Raw stream from input (proxy only).
+ ///
+ Raw,
+ ///
+ /// Digital On/Off input.
+ ///
+ Digital,
+ ///
+ /// Single Axis analogue input.
+ ///
+ SingleAxis,
+ ///
+ /// Dual Axis analogue input.
+ ///
+ DualAxis,
+ ///
+ /// Position only Axis analogue input.
+ ///
+ ThreeDofPosition,
+ ///
+ /// Rotation only Axis analogue input.
+ ///
+ ThreeDofRotation,
+ ///
+ /// Position AND Rotation analogue input.
+ ///
+ SixDof
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisType.cs.meta
new file mode 100644
index 0000000..f1de1a9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/AxisType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 15ab48ddceae418a8c8d072ee1ea658c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/CollationOrder.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/CollationOrder.cs
new file mode 100644
index 0000000..1434e93
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/CollationOrder.cs
@@ -0,0 +1,32 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Collation order type used for sorting
+ ///
+ public enum CollationOrder
+ {
+ ///
+ /// Don't sort, just display in order received
+ ///
+ None = 0,
+ ///
+ /// Sort by child order of parent
+ ///
+ ChildOrder,
+ ///
+ /// Sort by transform name
+ ///
+ Alphabetical,
+ ///
+ /// Sort by child order of parent, reversed
+ ///
+ ChildOrderReversed,
+ ///
+ /// Sort by transform name, reversed
+ ///
+ AlphabeticalReversed
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/CollationOrder.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/CollationOrder.cs.meta
new file mode 100644
index 0000000..37bcd2b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/CollationOrder.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1a692aa62f838a94988eb5b913d5939b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ExperienceScale.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ExperienceScale.cs
new file mode 100644
index 0000000..7d62218
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ExperienceScale.cs
@@ -0,0 +1,33 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The ExperienceScale identifies the environment for which the experience is designed.
+ ///
+ [System.Serializable]
+ public enum ExperienceScale
+ {
+ ///
+ /// An experience which utilizes only the headset orientation and is gravity aligned. The coordinate system origin is at head level.
+ ///
+ OrientationOnly = 0,
+ ///
+ /// An experience designed for seated use. The coordinate system origin is at head level.
+ ///
+ Seated,
+ ///
+ /// An experience designed for stationary standing use. The coordinate system origin is at floor level.
+ ///
+ Standing,
+ ///
+ /// An experience designed to support movement throughout a room. The coordinate system origin is at floor level.
+ ///
+ Room,
+ ///
+ /// An experience designed to utilize and move through the physical world. The coordinate system origin is at head level.
+ ///
+ World
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ExperienceScale.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ExperienceScale.cs.meta
new file mode 100644
index 0000000..6cbd14d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ExperienceScale.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 76a29d9960bc445289c706baa75160d7
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Handedness.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Handedness.cs
new file mode 100644
index 0000000..45d5ec8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Handedness.cs
@@ -0,0 +1,61 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The Handedness defines which hand a controller is currently operating in.
+ /// It is up to the developer to determine whether this affects the use of a controller or not.
+ /// "Other" defines potential controllers that will offer a "third" hand, e.g. a full body tracking suit.
+ ///
+ [Flags]
+ public enum Handedness : byte
+ {
+ ///
+ /// No hand specified by the SDK for the controller
+ ///
+ None = 0 << 0,
+ ///
+ /// The controller is identified as being provided in a Left hand
+ ///
+ Left = 1 << 0,
+ ///
+ /// The controller is identified as being provided in a Right hand
+ ///
+ Right = 1 << 1,
+ ///
+ /// The controller is identified as being either left and/or right handed.
+ ///
+ Both = Left | Right,
+ ///
+ /// Reserved, for systems that provide alternate hand state.
+ ///
+ Other = 1 << 2,
+ ///
+ /// Global catchall, used to map actions to any controller (provided the controller supports it)
+ ///
+ /// Note, by default the specific hand actions will override settings mapped as both
+ Any = Other | Left | Right,
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class HandednessExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this Handedness a, Handedness b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Handedness.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Handedness.cs.meta
new file mode 100644
index 0000000..a12d3ce
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Handedness.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f7c9d27c99e649e983cab045ea9aaea1
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationHandFlags.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationHandFlags.cs
new file mode 100644
index 0000000..5f3145e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationHandFlags.cs
@@ -0,0 +1,34 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Flags used to represent the number of hands that can be used in manipulation
+ ///
+ [System.Flags]
+ public enum ManipulationHandFlags
+ {
+ OneHanded = 1 << 0,
+ TwoHanded = 1 << 1,
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class ManipulationHandFlagsExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this ManipulationHandFlags a, ManipulationHandFlags b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationHandFlags.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationHandFlags.cs.meta
new file mode 100644
index 0000000..e490945
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationHandFlags.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 331dc46d2942d1947964376312ddc59c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationProximityFlags.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationProximityFlags.cs
new file mode 100644
index 0000000..1b7ede4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationProximityFlags.cs
@@ -0,0 +1,34 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Flags used to represent whether manipulation can be far, near or both
+ ///
+ [System.Flags]
+ public enum ManipulationProximityFlags
+ {
+ Near = 1 << 0,
+ Far = 1 << 1,
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class ManipulationProximityFlagsExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this ManipulationProximityFlags a, ManipulationProximityFlags b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationProximityFlags.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationProximityFlags.cs.meta
new file mode 100644
index 0000000..c1e51fe
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ManipulationProximityFlags.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c3cf8def1e3e5484c86b3dc60576feba
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityCapability.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityCapability.cs
new file mode 100644
index 0000000..5ff68df
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityCapability.cs
@@ -0,0 +1,56 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Mixed reality platform capabilities.
+ ///
+ public enum MixedRealityCapability
+ {
+ ///
+ /// Articulated hand input
+ ///
+ ArticulatedHand = 0,
+
+ ///
+ /// Gaze-Gesture-Voice hand input
+ ///
+ GGVHand,
+
+ ///
+ /// Motion controller input
+ ///
+ MotionController,
+
+ ///
+ /// Eye gaze targeting
+ ///
+ EyeTracking,
+
+ ///
+ /// Voice commands using app defined keywords
+ ///
+ VoiceCommand,
+
+ ///
+ /// Voice to text dictation
+ ///
+ VoiceDictation,
+
+ ///
+ /// Spatial meshes
+ ///
+ SpatialAwarenessMesh,
+
+ ///
+ /// Spatial planes
+ ///
+ SpatialAwarenessPlane,
+
+ ///
+ /// Spatial points
+ ///
+ SpatialAwarenessPoint
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityCapability.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityCapability.cs.meta
new file mode 100644
index 0000000..1c343e9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityCapability.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 05049f6c0dbfa71458a983f2bdc55920
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityPose.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityPose.cs
new file mode 100644
index 0000000..a6b6818
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityPose.cs
@@ -0,0 +1,144 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ [Serializable]
+ public struct MixedRealityPose : IEqualityComparer
+ {
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityPose(Vector3 position, Quaternion rotation)
+ {
+ this.position = position;
+ this.rotation = rotation;
+ }
+
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityPose(Vector3 position)
+ {
+ this.position = position;
+ rotation = Quaternion.identity;
+ }
+
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityPose(Quaternion rotation)
+ {
+ position = Vector3.zero;
+ this.rotation = rotation;
+ }
+
+ ///
+ /// The default value for a Six DoF Transform.
+ ///
+ ///
+ /// Vector3.zero and Quaternion.identity.
+ ///
+ public static MixedRealityPose ZeroIdentity { get; } = new MixedRealityPose(Vector3.zero, Quaternion.identity);
+
+ [SerializeField]
+ [Tooltip("The position of the pose")]
+ private Vector3 position;
+
+ ///
+ /// The position of the pose.
+ ///
+ public Vector3 Position { get { return position; } set { position = value; } }
+
+ [SerializeField]
+ [Tooltip("The rotation of the pose.")]
+ private Quaternion rotation;
+
+ ///
+ /// The rotation of the pose.
+ ///
+ public Quaternion Rotation { get { return rotation; } set { rotation = value; } }
+
+ public static MixedRealityPose operator +(MixedRealityPose left, MixedRealityPose right)
+ {
+ return new MixedRealityPose(left.Position + right.Position, left.Rotation * right.Rotation);
+ }
+
+ ///
+ /// Returns right-hand MixedRealityPose transformed by left-hand MixedRealityPose.
+ ///
+ public static MixedRealityPose operator *(MixedRealityPose left, MixedRealityPose right)
+ {
+ return new MixedRealityPose(left.Position + (left.Rotation * right.Position), left.Rotation * right.Rotation);
+ }
+
+ public static bool operator ==(MixedRealityPose left, MixedRealityPose right)
+ {
+ return left.Equals(right);
+ }
+
+ public static bool operator !=(MixedRealityPose left, MixedRealityPose right)
+ {
+ return !left.Equals(right);
+ }
+
+ public override string ToString()
+ {
+ return $"{position} | {rotation}";
+ }
+
+ ///
+ /// The Z axis of the pose in world space.
+ ///
+ public Vector3 Forward => rotation * Vector3.forward;
+
+ ///
+ /// The Y axis of the pose in world space.
+ ///
+ public Vector3 Up => rotation * Vector3.up;
+
+ ///
+ /// The X axis of the pose in world space.
+ ///
+ public Vector3 Right => rotation * Vector3.right;
+
+ #region IEqualityComparer Implementation
+
+ ///
+ bool IEqualityComparer.Equals(object left, object right)
+ {
+ if (ReferenceEquals(null, left) || ReferenceEquals(null, right)) { return false; }
+ if (!(left is MixedRealityPose) || !(right is MixedRealityPose)) { return false; }
+ return ((MixedRealityPose)left).Equals((MixedRealityPose)right);
+ }
+
+ public bool Equals(MixedRealityPose other)
+ {
+ return Position == other.Position &&
+ Rotation.Equals(other.Rotation);
+ }
+
+ public override bool Equals(object obj)
+ {
+ if (ReferenceEquals(null, obj)) { return false; }
+ return obj is MixedRealityPose pose && Equals(pose);
+ }
+
+ ///
+ int IEqualityComparer.GetHashCode(object obj)
+ {
+ return obj is MixedRealityPose pose ? pose.GetHashCode() : 0;
+ }
+
+ public override int GetHashCode()
+ {
+ return base.GetHashCode();
+ }
+
+ #endregion IEqualityComparer Implementation
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityPose.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityPose.cs.meta
new file mode 100644
index 0000000..8a57ce7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityPose.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f0c78c64d2484ac98ee537d88340eaac
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityTransform.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityTransform.cs
new file mode 100644
index 0000000..35b7457
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityTransform.cs
@@ -0,0 +1,154 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ [Serializable]
+ public struct MixedRealityTransform : IEqualityComparer
+ {
+
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityTransform(Transform transform)
+ {
+ this.pose = new MixedRealityPose(transform.position, transform.rotation);
+ this.scale = transform.localScale;
+ }
+
+ ///
+ /// Constructor.
+ ///
+ public MixedRealityTransform(Vector3 position, Quaternion rotation, Vector3 scale)
+ {
+ this.pose = new MixedRealityPose(position, rotation);
+ this.scale = scale;
+ }
+
+ ///
+ /// Create a transform with only given position
+ ///
+ public static MixedRealityTransform NewTranslate(Vector3 position)
+ {
+ return new MixedRealityTransform(position, Quaternion.identity, Vector3.one);
+ }
+
+ ///
+ /// Create a transform with only given rotation
+ ///
+ public static MixedRealityTransform NewRotate(Quaternion rotation)
+ {
+ return new MixedRealityTransform(Vector3.zero, rotation, Vector3.one);
+ }
+
+ ///
+ /// Create a transform with only given scale
+ ///
+ public static MixedRealityTransform NewScale(Vector3 scale)
+ {
+ return new MixedRealityTransform(Vector3.zero, Quaternion.identity, scale);
+ }
+
+ ///
+ /// The default value for a Six DoF Transform.
+ ///
+ public static MixedRealityTransform Identity { get; } = new MixedRealityTransform(Vector3.zero, Quaternion.identity, Vector3.one);
+
+ [SerializeField]
+ [Tooltip("The pose (position and rotation) of the transform")]
+ private MixedRealityPose pose;
+
+ ///
+ /// The position of the transform.
+ ///
+ public Vector3 Position { get { return pose.Position; } set { pose.Position = value; } }
+
+ ///
+ /// The rotation of the transform.
+ ///
+ public Quaternion Rotation { get { return pose.Rotation; } set { pose.Rotation = value; } }
+
+ [SerializeField]
+ [Tooltip("The scale of the transform.")]
+ private Vector3 scale;
+
+ ///
+ /// The scale of the transform.
+ ///
+ public Vector3 Scale { get { return scale; } set { scale = value; } }
+
+ public static MixedRealityTransform operator +(MixedRealityTransform left, MixedRealityTransform right)
+ {
+ return new MixedRealityTransform(left.Position + right.Position, left.Rotation * right.Rotation, Vector3.Scale(left.Scale, right.Scale));
+ }
+
+ public static bool operator ==(MixedRealityTransform left, MixedRealityTransform right)
+ {
+ return left.Equals(right);
+ }
+
+ public static bool operator !=(MixedRealityTransform left, MixedRealityTransform right)
+ {
+ return !left.Equals(right);
+ }
+
+ public override string ToString()
+ {
+ return $"{pose.Position} | {pose.Rotation} | {scale}";
+ }
+
+ ///
+ /// The Z axis of the pose in world space.
+ ///
+ public Vector3 Forward => (pose.Rotation * Vector3.Scale(scale, Vector3.forward)).normalized;
+
+ ///
+ /// The Y axis of the pose in world space.
+ ///
+ public Vector3 Up => (pose.Rotation * Vector3.Scale(scale, Vector3.up)).normalized;
+
+ ///
+ /// The X axis of the pose in world space.
+ ///
+ public Vector3 Right => (pose.Rotation * Vector3.Scale(scale, Vector3.right)).normalized;
+
+ #region IEqualityComparer Implementation
+
+ ///
+ bool IEqualityComparer.Equals(object left, object right)
+ {
+ if (ReferenceEquals(null, left) || ReferenceEquals(null, right)) { return false; }
+ if (!(left is MixedRealityTransform) || !(right is MixedRealityTransform)) { return false; }
+ return ((MixedRealityTransform)left).Equals((MixedRealityTransform)right);
+ }
+
+ public bool Equals(MixedRealityTransform other)
+ {
+ return Position == other.Position &&
+ Rotation.Equals(other.Rotation);
+ }
+
+ public override bool Equals(object obj)
+ {
+ if (ReferenceEquals(null, obj)) { return false; }
+ return obj is MixedRealityTransform transform && Equals(transform);
+ }
+
+ ///
+ int IEqualityComparer.GetHashCode(object obj)
+ {
+ return obj is MixedRealityTransform transform ? transform.GetHashCode() : 0;
+ }
+
+ public override int GetHashCode()
+ {
+ return base.GetHashCode();
+ }
+
+ #endregion IEqualityComparer Implementation
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityTransform.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityTransform.cs.meta
new file mode 100644
index 0000000..5f988b4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MixedRealityTransform.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1d7fa471ae4170f47b4c1a95066463de
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MovementConstraintType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MovementConstraintType.cs
new file mode 100644
index 0000000..2d8950e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MovementConstraintType.cs
@@ -0,0 +1,11 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ public enum MovementConstraintType
+ {
+ None,
+ FixDistanceFromHead
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MovementConstraintType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MovementConstraintType.cs.meta
new file mode 100644
index 0000000..062b7b0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/MovementConstraintType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 5fdbd32439c550040b98b93849f33078
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/OrientationType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/OrientationType.cs
new file mode 100644
index 0000000..66e778e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/OrientationType.cs
@@ -0,0 +1,48 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Orientation type enum
+ ///
+ public enum OrientationType
+ {
+ ///
+ /// Don't rotate at all
+ ///
+ None = 0,
+ ///
+ /// Rotate towards the origin
+ ///
+ FaceOrigin,
+ ///
+ /// Rotate towards the origin + 180 degrees
+ ///
+ FaceOriginReversed,
+ ///
+ /// Zero rotation
+ ///
+ FaceParentFoward,
+ ///
+ /// Zero rotation + 180 degrees
+ ///
+ FaceParentForwardReversed,
+ ///
+ /// Parent Relative Up
+ ///
+ FaceParentUp,
+ ///
+ /// Parent Relative Down
+ ///
+ FaceParentDown,
+ ///
+ /// Lay flat on the surface, facing in
+ ///
+ FaceCenterAxis,
+ ///
+ /// Lay flat on the surface, facing out
+ ///
+ FaceCenterAxisReversed
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/OrientationType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/OrientationType.cs.meta
new file mode 100644
index 0000000..1ead34d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/OrientationType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6941589aece774f4a9d93c1bceaca1e0
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/PivotAxis.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/PivotAxis.cs
new file mode 100644
index 0000000..7ea694c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/PivotAxis.cs
@@ -0,0 +1,23 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Rotational Pivot axis for orientating an object
+ ///
+ public enum PivotAxis
+ {
+ // Most common options, preserving current functionality with the same enum order.
+ XY,
+ Y,
+ // Rotate about an individual axis.
+ X,
+ Z,
+ // Rotate about a pair of axes.
+ XZ,
+ YZ,
+ // Rotate about all axes.
+ Free
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/PivotAxis.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/PivotAxis.cs.meta
new file mode 100644
index 0000000..2674250
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/PivotAxis.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 49901ddaa7d249c1b4430e2ba899aff8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProcessResult.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProcessResult.cs
new file mode 100644
index 0000000..07d57fc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProcessResult.cs
@@ -0,0 +1,39 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Result from a completed asynchronous process.
+ ///
+ public struct ProcessResult
+ {
+ ///
+ /// Exit code from completed process.
+ ///
+ public int ExitCode { get; }
+
+ ///
+ /// Errors from completed process.
+ ///
+ public string[] Errors { get; }
+
+ ///
+ /// Output from completed process.
+ ///
+ public string[] Output { get; }
+
+ ///
+ /// Constructor for Process Result.
+ ///
+ /// Exit code from completed process.
+ /// Errors from completed process.
+ /// Output from completed process.
+ public ProcessResult(int exitCode, string[] errors, string[] output) : this()
+ {
+ ExitCode = exitCode;
+ Errors = errors;
+ Output = output;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProcessResult.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProcessResult.cs.meta
new file mode 100644
index 0000000..f3cbc76
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProcessResult.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: de687c792c284b8da7c1fd44165967f6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProfileMenuItemIndices.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProfileMenuItemIndices.cs
new file mode 100644
index 0000000..a6ad1a2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProfileMenuItemIndices.cs
@@ -0,0 +1,36 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Defines the display order of the Assets > Create > Mixed Reality Toolkit > Profiles menu items.
+ ///
+ public enum CreateProfileMenuItemIndices
+ {
+ Configuration = 0,
+ Camera,
+ Input,
+ Pointer,
+ ControllerMapping,
+ InputActions,
+ InputActionRules,
+ Speech,
+ BoundaryVisualization,
+ ControllerVisualization,
+ SpatialAwareness, // todo: remove
+ SpatialAwarenessMeshObserver,
+ SpatialAwarenessSurfaceObserver,
+ Gestures,
+ Diagnostics,
+ RegisteredServiceProviders,
+ InputSimulation,
+ HandTracking,
+ EyeTracking,
+ MouseInput,
+ SceneSystem,
+ SceneUnderstandingObserver,
+
+ Assembly = 99, // This should stay at the end
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProfileMenuItemIndices.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProfileMenuItemIndices.cs.meta
new file mode 100644
index 0000000..a923b3d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ProfileMenuItemIndices.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 87707e41c37bc4848a02261a1c24e8ae
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RadialViewReferenceDirection.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RadialViewReferenceDirection.cs
new file mode 100644
index 0000000..77c9b32
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RadialViewReferenceDirection.cs
@@ -0,0 +1,24 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Which direction to orient the radial view object.
+ ///
+ public enum RadialViewReferenceDirection
+ {
+ ///
+ /// Orient towards the target including roll, pitch and yaw
+ ///
+ ObjectOriented,
+ ///
+ /// Orient toward the target but ignore roll
+ ///
+ FacingWorldUp,
+ ///
+ /// Orient towards the target but remain vertical or gravity aligned
+ ///
+ GravityAligned
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RadialViewReferenceDirection.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RadialViewReferenceDirection.cs.meta
new file mode 100644
index 0000000..dce3de5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RadialViewReferenceDirection.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c8285c2d275087649879c53375090ec1
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RecognitionConfidenceLevel.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RecognitionConfidenceLevel.cs
new file mode 100644
index 0000000..d177682
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RecognitionConfidenceLevel.cs
@@ -0,0 +1,16 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Indicates the confidence level of a recognized event.
+ ///
+ public enum RecognitionConfidenceLevel
+ {
+ High = 0,
+ Medium,
+ Low,
+ Unknown
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RecognitionConfidenceLevel.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RecognitionConfidenceLevel.cs.meta
new file mode 100644
index 0000000..ccc8fa8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RecognitionConfidenceLevel.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: cdb18394f18caaf4494b3f283304d6a9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RotationConstraintType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RotationConstraintType.cs
new file mode 100644
index 0000000..f1dbfe0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RotationConstraintType.cs
@@ -0,0 +1,37 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ public enum RotationConstraintType
+ {
+ None,
+ XAxisOnly,
+ YAxisOnly,
+ ZAxisOnly
+ }
+
+ ///
+ /// Helper class used to convert from RotationConstraintType to AxisFlags
+ ///
+ public class RotationConstraintHelper
+ {
+ ///
+ /// Returns corresponding AxisFlags for given RotationConstraintType
+ ///
+ public static AxisFlags ConvertToAxisFlags(RotationConstraintType type)
+ {
+ switch (type)
+ {
+ case RotationConstraintType.XAxisOnly:
+ return AxisFlags.YAxis | AxisFlags.ZAxis;
+ case RotationConstraintType.YAxisOnly:
+ return AxisFlags.XAxis | AxisFlags.ZAxis;
+ case RotationConstraintType.ZAxisOnly:
+ return AxisFlags.XAxis | AxisFlags.YAxis;
+ default:
+ return 0;
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RotationConstraintType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RotationConstraintType.cs.meta
new file mode 100644
index 0000000..44af2d3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/RotationConstraintType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2197541262093a249b139e0fc9cead7f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ScaleState.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ScaleState.cs
new file mode 100644
index 0000000..f9e9c41
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ScaleState.cs
@@ -0,0 +1,12 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ public enum ScaleState
+ {
+ Static = 0,
+ Shrinking,
+ Growing
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ScaleState.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ScaleState.cs.meta
new file mode 100644
index 0000000..f9a645e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/ScaleState.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c00bb86474e9d2240a0cf235cd8798cc
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SceneAssetReferenceAttribute.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SceneAssetReferenceAttribute.cs
new file mode 100644
index 0000000..43d1da1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SceneAssetReferenceAttribute.cs
@@ -0,0 +1,11 @@
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Attribute for using a SceneAssetReference property drawer.
+ ///
+ [AttributeUsage(AttributeTargets.Field, AllowMultiple = false)]
+ public class SceneAssetReferenceAttribute : PropertyAttribute { }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SceneAssetReferenceAttribute.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SceneAssetReferenceAttribute.cs.meta
new file mode 100644
index 0000000..077681a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SceneAssetReferenceAttribute.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d8c1459d933a37948b973c8875ff6a51
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SolverOrientationType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SolverOrientationType.cs
new file mode 100644
index 0000000..25793e9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SolverOrientationType.cs
@@ -0,0 +1,33 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ public enum SolverOrientationType
+ {
+ ///
+ /// Use the tracked object's pitch, yaw, and roll
+ ///
+ FollowTrackedObject = 0,
+ ///
+ /// Face toward the tracked object
+ ///
+ FaceTrackedObject,
+ ///
+ /// Orient towards SolverHandler's tracked object or TargetTransform
+ ///
+ YawOnly,
+ ///
+ /// Leave the object's rotation alone
+ ///
+ Unmodified,
+ ///
+ /// Orient toward the main camera instead of SolverHandler's properties.
+ ///
+ CameraFacing,
+ ///
+ /// Align parallel to the direction the camera is facing
+ ///
+ CameraAligned
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SolverOrientationType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SolverOrientationType.cs.meta
new file mode 100644
index 0000000..4a8f6d7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SolverOrientationType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c62bd8110c3bfcf4eadfada5bc89722a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedApplicationModes.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedApplicationModes.cs
new file mode 100644
index 0000000..7a989c1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedApplicationModes.cs
@@ -0,0 +1,27 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The supported Application modes for specific features.
+ ///
+ ///
+ /// This enum can be used to configure specific features to have differing behaviors when run in editor.
+ ///
+ [Flags]
+ public enum SupportedApplicationModes
+ {
+ ///
+ /// This indicates that the feature is relevant in editor scenarios.
+ ///
+ Editor = 1 << 0,
+
+ ///
+ /// This indicates that the feature is relevant in player scenarios.
+ ///
+ Player = 1 << 1,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedApplicationModes.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedApplicationModes.cs.meta
new file mode 100644
index 0000000..4cda1ef
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedApplicationModes.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8a74339227fff1e42aebfa58eaae1f19
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedPlatforms.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedPlatforms.cs
new file mode 100644
index 0000000..07e877a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedPlatforms.cs
@@ -0,0 +1,58 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The supported platforms for Mixed Reality Toolkit components and features.
+ ///
+ [Flags]
+ public enum SupportedPlatforms
+ {
+ WindowsStandalone = 1 << 0,
+ MacStandalone = 1 << 1,
+ LinuxStandalone = 1 << 2,
+ WindowsUniversal = 1 << 3,
+ WindowsEditor = 1 << 4,
+ Android = 1 << 5,
+ MacEditor = 1 << 6,
+ LinuxEditor = 1 << 7,
+ IOS = 1 << 8,
+ Web = 1 << 9,
+ Lumin = 1 << 10
+ }
+
+ ///
+ /// The supported Unity XR pipelines for Mixed Reality Toolkit components and features.
+ ///
+ [Flags]
+ public enum SupportedUnityXRPipelines
+ {
+#if UNITY_2020_1_OR_NEWER
+ [Obsolete("The legacy XR pipeline has been removed in Unity 2020 or newer. Please migrate to XR SDK.")]
+#endif // UNITY_2020_1_OR_NEWER
+ LegacyXR = 1 << 0,
+ XRSDK = 1 << 1,
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class SupportedUnityXRPipelinesExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this SupportedUnityXRPipelines a, SupportedUnityXRPipelines b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedPlatforms.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedPlatforms.cs.meta
new file mode 100644
index 0000000..2e7d161
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SupportedPlatforms.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6dca6753a2924f5d9301605e4f662702
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SystemType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SystemType.cs
new file mode 100644
index 0000000..9c4d622
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SystemType.cs
@@ -0,0 +1,162 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using Microsoft.MixedReality.Toolkit;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+using System;
+using System.Collections.Generic;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Reference to a class with support for Unity serialization.
+ ///
+ [Serializable]
+ public sealed class SystemType : ISerializationCallbackReceiver
+ {
+ [SerializeField]
+ private string reference = string.Empty;
+
+ private Type type;
+
+ public static string GetReference(Type type)
+ {
+ if (type == null || string.IsNullOrEmpty(type.AssemblyQualifiedName))
+ {
+ return string.Empty;
+ }
+
+ string[] qualifiedNameComponents = type.AssemblyQualifiedName.Split(',');
+ Debug.Assert(qualifiedNameComponents.Length >= 2);
+ return $"{qualifiedNameComponents[0]}, {qualifiedNameComponents[1].Trim()}";
+ }
+
+ ///
+ /// Initializes a new instance of the class.
+ ///
+ /// Assembly qualified class name.
+ public SystemType(string assemblyQualifiedClassName)
+ {
+ if (!string.IsNullOrEmpty(assemblyQualifiedClassName))
+ {
+ Type = Type.GetType(assemblyQualifiedClassName);
+
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+ if (Type != null && Type.IsAbstract())
+#else
+ if (Type != null && Type.IsAbstract)
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+ {
+ Type = null;
+ }
+ }
+ else
+ {
+ Type = null;
+ }
+ }
+
+ ///
+ /// Initializes a new instance of the class.
+ ///
+ /// Class type.
+ ///
+ /// If is not a class type.
+ ///
+ public SystemType(Type type)
+ {
+ Type = type;
+ }
+
+ #region ISerializationCallbackReceiver Members
+
+ void ISerializationCallbackReceiver.OnAfterDeserialize()
+ {
+ // Class references may move between asmdef or be renamed throughout MRTK development
+ // Check to see if we need to update our reference value
+ reference = TryMigrateReference(reference);
+
+ type = !string.IsNullOrEmpty(reference) ? Type.GetType(reference) : null;
+ }
+
+ void ISerializationCallbackReceiver.OnBeforeSerialize() { }
+
+ #endregion ISerializationCallbackReceiver Members
+
+ ///
+ /// Gets or sets type of class reference.
+ ///
+ ///
+ /// If is not a class type.
+ ///
+ public Type Type
+ {
+ get => type;
+ set
+ {
+ if (value != null)
+ {
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+ bool isValid = value.IsValueType() && !value.IsEnum() && !value.IsAbstract() || value.IsClass();
+#else
+ bool isValid = value.IsValueType && !value.IsEnum && !value.IsAbstract || value.IsClass;
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
+ if (!isValid)
+ {
+ Debug.LogError($"'{value.FullName}' is not a valid class or struct type.");
+ }
+ }
+
+ type = value;
+ reference = GetReference(value);
+ }
+ }
+
+ public static implicit operator string(SystemType type)
+ {
+ return type.reference;
+ }
+
+ public static implicit operator Type(SystemType type)
+ {
+ return type.Type;
+ }
+
+ public static implicit operator SystemType(Type type)
+ {
+ return new SystemType(type);
+ }
+
+ public override string ToString()
+ {
+ return Type?.FullName ?? "(None)";
+ }
+
+ // Key == original reference string entry, value == new migrated placement
+ // String values are broken into {namespace.classname, asmdef}
+ private static Dictionary ReferenceMappings = new Dictionary()
+ {
+ { "Microsoft.MixedReality.Toolkit.Input.InputSimulationService, Microsoft.MixedReality.Toolkit.Services.InputSimulation.Editor",
+ "Microsoft.MixedReality.Toolkit.Input.InputSimulationService, Microsoft.MixedReality.Toolkit.Services.InputSimulation" },
+
+ { "Microsoft.MixedReality.Toolkit.Input.InputPlaybackService, Microsoft.MixedReality.Toolkit.Services.InputSimulation.Editor",
+ "Microsoft.MixedReality.Toolkit.Input.InputPlaybackService, Microsoft.MixedReality.Toolkit.Services.InputSimulation" },
+ };
+
+ ///
+ /// This function checks if there are any known migrations for old class names, namespaces, and/or asmdef files
+ /// If so, the new reference string is returned and utilized for editor runtime and will be serialized to disk
+ ///
+ private static string TryMigrateReference(string reference)
+ {
+ if (ReferenceMappings.ContainsKey(reference))
+ {
+ return ReferenceMappings[reference];
+ }
+
+ return reference;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SystemType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SystemType.cs.meta
new file mode 100644
index 0000000..9399501
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/SystemType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 05816c45081244eda99c58788b66f0e9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedHandJoint.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedHandJoint.cs
new file mode 100644
index 0000000..bc7ee7b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedHandJoint.cs
@@ -0,0 +1,118 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The supported tracked hand joints.
+ ///
+ /// See https://en.wikipedia.org/wiki/Interphalangeal_joints_of_the_hand#/media/File:Scheme_human_hand_bones-en.svg for joint name definitions.
+ public enum TrackedHandJoint
+ {
+ None = 0,
+ ///
+ /// The wrist.
+ ///
+ Wrist,
+ ///
+ /// The palm.
+ ///
+ Palm,
+ ///
+ /// The lowest joint in the thumb (down in your palm).
+ ///
+ ThumbMetacarpalJoint,
+ ///
+ /// The thumb's second (middle-ish) joint.
+ ///
+ ThumbProximalJoint,
+ ///
+ /// The thumb's first (furthest) joint.
+ ///
+ ThumbDistalJoint,
+ ///
+ /// The tip of the thumb.
+ ///
+ ThumbTip,
+ ///
+ /// The lowest joint of the index finger.
+ ///
+ IndexMetacarpal,
+ ///
+ /// The knuckle joint of the index finger.
+ ///
+ IndexKnuckle,
+ ///
+ /// The middle joint of the index finger.
+ ///
+ IndexMiddleJoint,
+ ///
+ /// The joint nearest the tip of the index finger.
+ ///
+ IndexDistalJoint,
+ ///
+ /// The tip of the index finger.
+ ///
+ IndexTip,
+ ///
+ /// The lowest joint of the middle finger.
+ ///
+ MiddleMetacarpal,
+ ///
+ /// The knuckle joint of the middle finger.
+ ///
+ MiddleKnuckle,
+ ///
+ /// The middle joint of the middle finger.
+ ///
+ MiddleMiddleJoint,
+ ///
+ /// The joint nearest the tip of the finger.
+ ///
+ MiddleDistalJoint,
+ ///
+ /// The tip of the middle finger.
+ ///
+ MiddleTip,
+ ///
+ /// The lowest joint of the ring finger.
+ ///
+ RingMetacarpal,
+ ///
+ /// The knuckle of the ring finger.
+ ///
+ RingKnuckle,
+ ///
+ /// The middle joint of the ring finger.
+ ///
+ RingMiddleJoint,
+ ///
+ /// The joint nearest the tip of the ring finger.
+ ///
+ RingDistalJoint,
+ ///
+ /// The tip of the ring finger.
+ ///
+ RingTip,
+ ///
+ /// The lowest joint of the pinky finger.
+ ///
+ PinkyMetacarpal,
+ ///
+ /// The knuckle joint of the pinky finger.
+ ///
+ PinkyKnuckle,
+ ///
+ /// The middle joint of the pinky finger.
+ ///
+ PinkyMiddleJoint,
+ ///
+ /// The joint nearest the tip of the pink finger.
+ ///
+ PinkyDistalJoint,
+ ///
+ /// The tip of the pinky.
+ ///
+ PinkyTip
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedHandJoint.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedHandJoint.cs.meta
new file mode 100644
index 0000000..edeb484
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedHandJoint.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 05dbaa414de760145a4a85c115550377
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedObjectType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedObjectType.cs
new file mode 100644
index 0000000..bbad797
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedObjectType.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ public enum TrackedObjectType
+ {
+ ///
+ /// Calculates position and orientation from the main camera.
+ ///
+ Head = 0,
+
+ ///
+ /// (Obsolete) Calculates position and orientation from the left motion-tracked controller.
+ ///
+ [Obsolete("Use TrackedObjectType.ControllerRay and TrackedHandedness instead")]
+ MotionControllerLeft = 1,
+ ///
+ /// (Obsolete) Calculates position and orientation from the right motion-tracked controller.
+ ///
+ [Obsolete("Use TrackedObjectType.ControllerRay and TrackedHandedness instead")]
+ MotionControllerRight = 2,
+ ///
+ /// (Obsolete) Calculates position and orientation from a tracked hand joint on the left hand.
+ ///
+ [Obsolete("Use TrackedObjectType.HandJoint and TrackedHandedness instead")]
+ HandJointLeft = 3,
+ ///
+ /// (Obsolete) Calculates position and orientation from a tracked hand joint on the right hand.
+ ///
+ [Obsolete("Use TrackedObjectType.HandJoint and TrackedHandedness instead")]
+ HandJointRight = 4,
+
+ ///
+ /// Calculates position and orientation from the system-calculated ray of available controller (i.e motion controllers, hands, etc.)
+ ///
+ ControllerRay = 5,
+ ///
+ /// Calculates position and orientation from a tracked hand joint
+ ///
+ HandJoint = 6,
+ ///
+ /// Calculates position and orientation from a tracked hand joint
+ ///
+ CustomOverride = 7,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedObjectType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedObjectType.cs.meta
new file mode 100644
index 0000000..a4f79f0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TrackedObjectType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 04704bde8ca35fd459c4ae5e88fde2ac
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TransformFlags.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TransformFlags.cs
new file mode 100644
index 0000000..1285d6e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TransformFlags.cs
@@ -0,0 +1,35 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Flags used to represent a combination of different types of transformation
+ ///
+ [System.Flags]
+ public enum TransformFlags
+ {
+ Move = 1 << 0,
+ Rotate = 1 << 1,
+ Scale = 1 << 2
+ }
+
+ ///
+ /// Extension methods specific to the enum.
+ ///
+ public static class TransformFlagsExtensions
+ {
+ ///
+ /// Checks to determine if all bits in a provided mask are set.
+ ///
+ /// value.
+ /// mask.
+ ///
+ /// True if all of the bits in the specified mask are set in the current value.
+ ///
+ public static bool IsMaskSet(this TransformFlags a, TransformFlags b)
+ {
+ return (a & b) == b;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TransformFlags.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TransformFlags.cs.meta
new file mode 100644
index 0000000..3ac0a42
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TransformFlags.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1d5ea1b268c1f344b808b6ab28fc551e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TypeGrouping.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TypeGrouping.cs
new file mode 100644
index 0000000..6e0f02e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TypeGrouping.cs
@@ -0,0 +1,33 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// Indicates how selectable classes should be collated in drop-down menu.
+ ///
+ public enum TypeGrouping
+ {
+ ///
+ /// No grouping, just show type names in a list; for instance, "Some.Nested.Namespace.SpecialClass".
+ ///
+ None,
+
+ ///
+ /// Group classes by namespace and show foldout menus for nested namespaces; for
+ /// instance, "Some > Nested > Namespace > SpecialClass".
+ ///
+ ByNamespace,
+
+ ///
+ /// Group classes by namespace; for instance, "Some.Nested.Namespace > SpecialClass".
+ ///
+ ByNamespaceFlat,
+
+ ///
+ /// Group classes in the same way as Unity does for its component menu. This
+ /// grouping method must only be used for MonoBehaviour types.
+ ///
+ ByAddComponentMenu,
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TypeGrouping.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TypeGrouping.cs.meta
new file mode 100644
index 0000000..2076a1b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/TypeGrouping.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: eea43dd3544c41ec99fbd9875645181d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Vector3Smoothed.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Vector3Smoothed.cs
new file mode 100644
index 0000000..0f72a4a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Vector3Smoothed.cs
@@ -0,0 +1,33 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ [Serializable]
+ public struct Vector3Smoothed
+ {
+ public Vector3 Current { get; set; }
+ public Vector3 Goal { get; set; }
+ public float SmoothTime { get; set; }
+
+ public Vector3Smoothed(Vector3 value, float smoothingTime) : this()
+ {
+ Current = value;
+ Goal = value;
+ SmoothTime = smoothingTime;
+ }
+
+ public void Update(float deltaTime)
+ {
+ Current = Vector3.Lerp(Current, Goal, (Math.Abs(SmoothTime) < Mathf.Epsilon) ? 1.0f : deltaTime / SmoothTime);
+ }
+
+ public void SetGoal(Vector3 newGoal)
+ {
+ Goal = newGoal;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Vector3Smoothed.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Vector3Smoothed.cs.meta
new file mode 100644
index 0000000..617a0de
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/Vector3Smoothed.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d0388f3cdf0f6ad42b5d2e84e2d5bd75
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/VolumeType.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/VolumeType.cs
new file mode 100644
index 0000000..81ba1e4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/VolumeType.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit.Utilities
+{
+ ///
+ /// The possible shapes of bounding volumes for spatial awareness of the user's surroundings.
+ ///
+ public enum VolumeType
+ {
+ ///
+ /// No specified type.
+ ///
+ None = 0,
+
+ ///
+ /// Cubic volume aligned with the coordinate axes.
+ ///
+ AxisAlignedCube,
+
+ ///
+ /// Cubic volume aligned with the user.
+ ///
+ UserAlignedCube,
+
+ ///
+ /// Spherical volume.
+ ///
+ Sphere
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/VolumeType.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/VolumeType.cs.meta
new file mode 100644
index 0000000..7da18c1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Definitions/Utilities/VolumeType.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9627954d6427f6b448a405c9563585eb
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum.meta
new file mode 100644
index 0000000..6bfdf90
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: e43e2587caef422b96a81b40c69f1ca4
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary.meta
new file mode 100644
index 0000000..ecb1e2c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: cdf6caa6c97341258406c76ca0fb7cb2
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary/BoundaryEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary/BoundaryEventData.cs
new file mode 100644
index 0000000..5c9c7f1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary/BoundaryEventData.cs
@@ -0,0 +1,62 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Boundary
+{
+ ///
+ /// The data describing the boundary system event.
+ ///
+ public class BoundaryEventData : GenericBaseEventData
+ {
+ ///
+ /// Is the floor being visualized by the boundary system.
+ ///
+ public bool IsFloorVisualized { get; private set; }
+
+ ///
+ /// Is the play area being visualized by the boundary system.
+ ///
+ public bool IsPlayAreaVisualized { get; private set; }
+
+ ///
+ /// Is the tracked area being visualized by the boundary system.
+ ///
+ public bool IsTrackedAreaVisualized { get; private set; }
+
+ ///
+ /// Are the boundary walls being visualized by the boundary system.
+ ///
+ public bool AreBoundaryWallsVisualized { get; private set; }
+
+ ///
+ /// Is the ceiling being visualized by the boundary system.
+ ///
+ ///
+ /// The boundary system defines the ceiling as a plane set at above the floor.
+ ///
+ public bool IsCeilingVisualized { get; private set; }
+
+ ///
+ /// Constructor.
+ ///
+ public BoundaryEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ public void Initialize(
+ IMixedRealityBoundarySystem boundarySystem,
+ bool isFloorVisualized,
+ bool isPlayAreaVisualized,
+ bool isTrackedAreaVisualized,
+ bool areBoundaryWallsVisualized,
+ bool isCeilingVisualized)
+ {
+ base.BaseInitialize(boundarySystem);
+ IsFloorVisualized = isFloorVisualized;
+ IsPlayAreaVisualized = isPlayAreaVisualized;
+ IsTrackedAreaVisualized = isTrackedAreaVisualized;
+ AreBoundaryWallsVisualized = areBoundaryWallsVisualized;
+ IsCeilingVisualized = isCeilingVisualized;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary/BoundaryEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary/BoundaryEventData.cs.meta
new file mode 100644
index 0000000..c72e8ef
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Boundary/BoundaryEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8c45c81c4f8f46b3af6f9f8a475c1688
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics.meta
new file mode 100644
index 0000000..7fe00aa
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: d8a2128563c03e146aa0eb67da229dcd
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics/DiagnosticsEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics/DiagnosticsEventData.cs
new file mode 100644
index 0000000..8851183
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics/DiagnosticsEventData.cs
@@ -0,0 +1,25 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Diagnostics
+{
+ public class DiagnosticsEventData : GenericBaseEventData
+ {
+ ///
+ /// Constructor
+ ///
+ public DiagnosticsEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Constructor
+ ///
+ /// The instance of the Diagnostic System that raised the event.
+ public void Initialize(
+ IMixedRealityDiagnosticsSystem diagnosticsSystem)
+ {
+ BaseInitialize(diagnosticsSystem);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics/DiagnosticsEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics/DiagnosticsEventData.cs.meta
new file mode 100644
index 0000000..27bc550
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Diagnostics/DiagnosticsEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a756df3fdd3951a4e9b2414b8edc5187
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/GenericBaseEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/GenericBaseEventData.cs
new file mode 100644
index 0000000..a4ab511
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/GenericBaseEventData.cs
@@ -0,0 +1,54 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Generic Base Event Data for Sending Events through the Event System.
+ ///
+ public class GenericBaseEventData : BaseEventData
+ {
+ ///
+ /// The Event Source that the event originates from.
+ ///
+ public IMixedRealityEventSource EventSource { get; private set; }
+
+ ///
+ /// The UTC time at which the event occurred.
+ ///
+ public DateTime EventTime { get; private set; }
+
+ ///
+ /// The BaseEventData.selectedObject is explicitly hidden because access to it
+ /// (either via get or set) throws a NullReferenceException in typical usage within
+ /// the MRTK. Prefer using the subclasses own fields to access information about
+ /// the event instead of fields on BaseEventData.
+ ///
+ ///
+ /// BaseEventData is only used because it's part of Unity's EventSystem dispatching,
+ /// so this code must subclass it in order to leverage EventSystem.ExecuteEvents
+ ///
+ public new GameObject selectedObject { get; protected set; }
+
+ ///
+ /// Constructor.
+ ///
+ /// Usually EventSystems.EventSystem.current
+ public GenericBaseEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ /// The source of the event.
+ protected void BaseInitialize(IMixedRealityEventSource eventSource)
+ {
+ Reset();
+ EventTime = DateTime.UtcNow;
+ EventSource = eventSource;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/GenericBaseEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/GenericBaseEventData.cs.meta
new file mode 100644
index 0000000..97bbe1e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/GenericBaseEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ea8bc3c3fca64f75bd82aacf62120a0b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input.meta
new file mode 100644
index 0000000..8c1f0e0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 1adbba004b214082aed1c8dab09ac2fc
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/BaseInputEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/BaseInputEventData.cs
new file mode 100644
index 0000000..0ff4cf6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/BaseInputEventData.cs
@@ -0,0 +1,51 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Base class of all input events.
+ ///
+ public abstract class BaseInputEventData : BaseEventData
+ {
+ ///
+ /// The UTC time at which the event occurred.
+ ///
+ public DateTime EventTime { get; private set; }
+
+ ///
+ /// The source the input event originates from.
+ ///
+ public IMixedRealityInputSource InputSource { get; private set; }
+
+ ///
+ /// The id of the source the event is from, for instance the hand id.
+ ///
+ public uint SourceId => InputSource.SourceId;
+
+ ///
+ /// The input action for this event.
+ ///
+ public MixedRealityInputAction MixedRealityInputAction { get; private set; }
+
+ ///
+ /// Constructor.
+ ///
+ /// Typically will be EventSystems.EventSystem.current
+ protected BaseInputEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ protected void BaseInitialize(IMixedRealityInputSource inputSource, MixedRealityInputAction inputAction)
+ {
+ Reset();
+ EventTime = DateTime.UtcNow;
+ InputSource = inputSource;
+ MixedRealityInputAction = inputAction;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/BaseInputEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/BaseInputEventData.cs.meta
new file mode 100644
index 0000000..cbf82ab
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/BaseInputEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: bf8c148379a64553b62a25a20d79495f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/DictationEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/DictationEventData.cs
new file mode 100644
index 0000000..713fe9d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/DictationEventData.cs
@@ -0,0 +1,37 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes an Input Event with voice dictation.
+ ///
+ public class DictationEventData : BaseInputEventData
+ {
+ ///
+ /// String result of the current dictation.
+ ///
+ public string DictationResult { get; private set; }
+
+ ///
+ /// Audio Clip of the last Dictation recording Session.
+ ///
+ public AudioClip DictationAudioClip { get; private set; }
+
+ ///
+ public DictationEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityInputSource inputSource, string dictationResult, AudioClip dictationAudioClip = null)
+ {
+ BaseInitialize(inputSource, MixedRealityInputAction.None);
+ DictationResult = dictationResult;
+ DictationAudioClip = dictationAudioClip;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/DictationEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/DictationEventData.cs.meta
new file mode 100644
index 0000000..202e232
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/DictationEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 80421fea75cc4dfdab2a329edb08977d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/FocusEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/FocusEventData.cs
new file mode 100644
index 0000000..ec57424
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/FocusEventData.cs
@@ -0,0 +1,53 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes an Input Event associated with a specific pointer's focus state change.
+ ///
+ public class FocusEventData : BaseEventData
+ {
+ ///
+ /// The pointer associated with this event.
+ ///
+ public IMixedRealityPointer Pointer { get; private set; }
+
+ ///
+ /// The old focused object.
+ ///
+ public GameObject OldFocusedObject { get; private set; }
+
+ ///
+ /// The new focused object.
+ ///
+ public GameObject NewFocusedObject { get; private set; }
+
+ ///
+ public FocusEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityPointer pointer)
+ {
+ Reset();
+ Pointer = pointer;
+ }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityPointer pointer, GameObject oldFocusedObject, GameObject newFocusedObject)
+ {
+ Reset();
+ Pointer = pointer;
+ OldFocusedObject = oldFocusedObject;
+ NewFocusedObject = newFocusedObject;
+ }
+ }
+}
+
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/FocusEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/FocusEventData.cs.meta
new file mode 100644
index 0000000..f09a7c3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/FocusEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 6d7e1f629f764c588ee755b2e0eff5a7
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/HandTrackingInputEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/HandTrackingInputEventData.cs
new file mode 100644
index 0000000..17c2ac8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/HandTrackingInputEventData.cs
@@ -0,0 +1,33 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ public class HandTrackingInputEventData : InputEventData
+ {
+ ///
+ /// Constructor creates a default EventData object.
+ /// Requires initialization.
+ ///
+ public HandTrackingInputEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ public IMixedRealityController Controller { get; set; }
+
+ ///
+ /// This function is called to fill the HandTrackingIntputEventData object with information
+ ///
+ /// Reference to the HandTrackingInputSource that created the EventData
+ /// Reference to the IMixedRealityController that created the EventData
+ /// Handedness of the HandTrackingInputSource that created the EventData
+ /// Global position of the HandTrackingInputSource that created the EventData
+ public void Initialize(IMixedRealityInputSource inputSource, IMixedRealityController controller, Handedness sourceHandedness, Vector3 touchPoint)
+ {
+ Initialize(inputSource, sourceHandedness, MixedRealityInputAction.None, touchPoint);
+ Controller = controller;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/HandTrackingInputEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/HandTrackingInputEventData.cs.meta
new file mode 100644
index 0000000..ef05bd7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/HandTrackingInputEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: bd73f48e372ad0445a17977a0ea1b138
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/InputEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/InputEventData.cs
new file mode 100644
index 0000000..26aeba4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/InputEventData.cs
@@ -0,0 +1,55 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes an Input Event that has a source id.
+ ///
+ public class InputEventData : BaseInputEventData
+ {
+ ///
+ /// Handedness of the .
+ ///
+ public Handedness Handedness { get; private set; } = Handedness.None;
+
+ ///
+ public InputEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityInputSource inputSource, Handedness handedness, MixedRealityInputAction inputAction)
+ {
+ BaseInitialize(inputSource, inputAction);
+ Handedness = handedness;
+ }
+ }
+
+ ///
+ /// Describes and input event with a specific type.
+ ///
+ ///
+ public class InputEventData : InputEventData
+ {
+ ///
+ /// The input data of the event.
+ ///
+ public T InputData { get; private set; }
+
+ ///
+ public InputEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityInputSource inputSource, Handedness handedness, MixedRealityInputAction inputAction, T data)
+ {
+ Initialize(inputSource, handedness, inputAction);
+ InputData = data;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/InputEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/InputEventData.cs.meta
new file mode 100644
index 0000000..8264aa2
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/InputEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4c178e08073343b7be3f086a9e26376d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/MixedRealityPointerEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/MixedRealityPointerEventData.cs
new file mode 100644
index 0000000..1cc9cdb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/MixedRealityPointerEventData.cs
@@ -0,0 +1,55 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes an Input Event that involves a tap, click, or touch.
+ ///
+ public class MixedRealityPointerEventData : InputEventData
+ {
+ ///
+ /// Pointer for the Input Event
+ ///
+ public IMixedRealityPointer Pointer { get; private set; }
+
+ ///
+ /// Number of Clicks, Taps, or Presses that triggered the event.
+ ///
+ public int Count { get; private set; }
+
+ ///
+ public MixedRealityPointerEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityPointer pointer, MixedRealityInputAction inputAction, Handedness handedness = Handedness.None, IMixedRealityInputSource inputSource = null, int count = 0)
+ {
+ if (inputSource != null)
+ {
+ Initialize(inputSource, handedness, inputAction);
+ }
+ else
+ {
+ Initialize(pointer.InputSourceParent, handedness, inputAction);
+ }
+
+ Pointer = pointer;
+ Count = count;
+ }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityPointer pointer, Handedness handedness, MixedRealityInputAction inputAction, int count = 0)
+ {
+ Initialize(pointer.InputSourceParent, handedness, inputAction);
+ Pointer = pointer;
+ Count = count;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/MixedRealityPointerEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/MixedRealityPointerEventData.cs.meta
new file mode 100644
index 0000000..1376faa
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/MixedRealityPointerEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 76aed106be8a4a90ae82a99228cf33c8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourcePoseEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourcePoseEventData.cs
new file mode 100644
index 0000000..acd87df
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourcePoseEventData.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes a source change event.
+ ///
+ /// Source State events do not have an associated .
+ public class SourcePoseEventData : SourceStateEventData
+ {
+ ///
+ /// The new position of the input source.
+ ///
+ public T SourceData { get; private set; }
+
+ ///
+ public SourcePoseEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Populates the event with data.
+ ///
+ public void Initialize(IMixedRealityInputSource inputSource, IMixedRealityController controller, T data)
+ {
+ Initialize(inputSource, controller);
+ SourceData = data;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourcePoseEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourcePoseEventData.cs.meta
new file mode 100644
index 0000000..b1d6fa8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourcePoseEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f39af9085dab46278471d207ca5d0a61
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourceStateEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourceStateEventData.cs
new file mode 100644
index 0000000..d587c65
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourceStateEventData.cs
@@ -0,0 +1,29 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes an source state event that has a source id.
+ ///
+ /// Source State events do not have an associated .
+ public class SourceStateEventData : BaseInputEventData
+ {
+ public IMixedRealityController Controller { get; private set; }
+
+ ///
+ public SourceStateEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Populates the event with data.
+ ///
+ public void Initialize(IMixedRealityInputSource inputSource, IMixedRealityController controller)
+ {
+ // NOTE: Source State events do not have an associated Input Action.
+ BaseInitialize(inputSource, MixedRealityInputAction.None);
+ Controller = controller;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourceStateEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourceStateEventData.cs.meta
new file mode 100644
index 0000000..a0e4ca7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SourceStateEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 4e5c13da632146c3b92a6ba911e77ebb
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SpeechEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SpeechEventData.cs
new file mode 100644
index 0000000..84f2f69
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SpeechEventData.cs
@@ -0,0 +1,50 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ ///
+ /// Describes an input event that involves keyword recognition.
+ ///
+ public class SpeechEventData : BaseInputEventData
+ {
+ ///
+ /// The time it took for the phrase to be uttered.
+ ///
+ public TimeSpan PhraseDuration { get; private set; }
+
+ ///
+ /// The moment in UTC time when uttering of the phrase began.
+ ///
+ public DateTime PhraseStartTime { get; private set; }
+
+ ///
+ /// The text that was recognized.
+ ///
+ public SpeechCommands Command { get; private set; }
+
+ ///
+ /// A measure of correct recognition certainty.
+ ///
+ public RecognitionConfidenceLevel Confidence { get; private set; }
+
+ ///
+ public SpeechEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Populates the event with data.
+ ///
+ public void Initialize(IMixedRealityInputSource inputSource, RecognitionConfidenceLevel confidence, TimeSpan phraseDuration, DateTime phraseStartTime, SpeechCommands command)
+ {
+ BaseInitialize(inputSource, command.Action);
+ Confidence = confidence;
+ PhraseDuration = phraseDuration;
+ PhraseStartTime = phraseStartTime;
+ Command = command;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SpeechEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SpeechEventData.cs.meta
new file mode 100644
index 0000000..18ff8fe
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Input/SpeechEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d7c2e29b4f8642388b4f8bb33fa14ac8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/PlacementEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/PlacementEventData.cs
new file mode 100644
index 0000000..22167e3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/PlacementEventData.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Describes placement of objects events.
+ ///
+ public class PlacementEventData : GenericBaseEventData
+ {
+ ///
+ /// The game object that is being placed.
+ ///
+ public GameObject ObjectBeingPlaced { get; private set; }
+
+ ///
+ public PlacementEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Populates the event with data.
+ ///
+ public void Initialize(IMixedRealityEventSource eventSource, GameObject objectBeingPlaced)
+ {
+ BaseInitialize(eventSource);
+ ObjectBeingPlaced = objectBeingPlaced;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/PlacementEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/PlacementEventData.cs.meta
new file mode 100644
index 0000000..02db622
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/PlacementEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b5143cedf6204703871f3962938ba8b0
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/README.md b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/README.md
new file mode 100644
index 0000000..0c25b34
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/README.md
@@ -0,0 +1,5 @@
+# Mixed Reality Toolkit - EventDatum
+
+Data model classes for the inner workings of the Mixed Reality Toolkit and its supported Core systems.
+
+All data models required for system use within the MRTK should be recorded here.
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/README.md.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/README.md.meta
new file mode 100644
index 0000000..70d6072
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/README.md.meta
@@ -0,0 +1,7 @@
+fileFormatVersion: 2
+guid: 16afe9b62a93a864cb9071665b204d21
+TextScriptImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness.meta
new file mode 100644
index 0000000..ea0054a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 40032830d1c03bb4fb063fef3ad61dc9
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness/MixedRealitySpatialAwarenessEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness/MixedRealitySpatialAwarenessEventData.cs
new file mode 100644
index 0000000..0b8bee1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness/MixedRealitySpatialAwarenessEventData.cs
@@ -0,0 +1,61 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.SpatialAwareness
+{
+ ///
+ /// Data for spatial awareness events.
+ ///
+ public class MixedRealitySpatialAwarenessEventData : GenericBaseEventData
+ {
+ ///
+ /// Identifier of the object associated with this event.
+ ///
+ public int Id { get; private set; }
+
+ ///
+ /// Constructor.
+ ///
+ public MixedRealitySpatialAwarenessEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Initialize the event data.
+ ///
+ /// The that raised the event.
+ /// The identifier of the observed spatial object.
+ public void Initialize(IMixedRealitySpatialAwarenessObserver observer, int id)
+ {
+ BaseInitialize(observer);
+ Id = id;
+ }
+ }
+
+ ///
+ /// Data for spatial awareness events.
+ ///
+ /// The spatial object data type.
+ public class MixedRealitySpatialAwarenessEventData : MixedRealitySpatialAwarenessEventData
+ {
+ ///
+ /// The spatial object to which this event pertains.
+ ///
+ public T SpatialObject { get; private set; }
+
+ ///
+ public MixedRealitySpatialAwarenessEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Initialize the event data.
+ ///
+ /// The that raised the event.
+ /// The identifier of the observed spatial object.
+ /// The observed spatial object.
+ public void Initialize(IMixedRealitySpatialAwarenessObserver observer, int id, T spatialObject)
+ {
+ Initialize(observer, id);
+ SpatialObject = spatialObject;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness/MixedRealitySpatialAwarenessEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness/MixedRealitySpatialAwarenessEventData.cs.meta
new file mode 100644
index 0000000..2ec08b9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/SpatialAwareness/MixedRealitySpatialAwarenessEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 5b1027163c69ba64a9951f075fdd9818
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport.meta
new file mode 100644
index 0000000..35599fd
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 2863a477e4224b0b9ecd72139eee99df
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport/TeleportEventData.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport/TeleportEventData.cs
new file mode 100644
index 0000000..dcb29b1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport/TeleportEventData.cs
@@ -0,0 +1,40 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Input;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit.Teleport
+{
+ ///
+ /// Describes a Teleportation Event.
+ ///
+ public class TeleportEventData : GenericBaseEventData
+ {
+ ///
+ /// The pointer that raised the event.
+ ///
+ public IMixedRealityPointer Pointer { get; private set; }
+
+ ///
+ /// The teleport hot spot.
+ ///
+ public IMixedRealityTeleportHotspot Hotspot { get; private set; }
+
+ ///
+ /// Constructor.
+ ///
+ /// Typically will be EventSystem.current
+ public TeleportEventData(EventSystem eventSystem) : base(eventSystem) { }
+
+ ///
+ /// Used to initialize/reset the event and populate the data.
+ ///
+ public void Initialize(IMixedRealityPointer pointer, IMixedRealityTeleportHotspot target)
+ {
+ BaseInitialize(pointer.InputSourceParent);
+ Pointer = pointer;
+ Hotspot = target;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport/TeleportEventData.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport/TeleportEventData.cs.meta
new file mode 100644
index 0000000..eb51525
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/EventDatum/Teleport/TeleportEventData.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: dc54593d192e430081797a6bcd020886
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions.meta
new file mode 100644
index 0000000..cf7bbe4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: e43a7a74fd784eb08cf2dd49781bb60c
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AnimationCurveExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AnimationCurveExtensions.cs
new file mode 100644
index 0000000..a0efbbc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AnimationCurveExtensions.cs
@@ -0,0 +1,28 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's AnimationCurve class
+ ///
+ public static class AnimationCurveExtensions
+ {
+ ///
+ /// Returns the absolute duration of the curve from first to last key frame
+ ///
+ /// The animation curve to check duration of.
+ /// Returns 0 if the curve is null or has less than 1 frame, otherwise returns time difference between first and last frame.
+ public static float Duration(this AnimationCurve curve)
+ {
+ if (curve == null || curve.length <= 1)
+ {
+ return 0.0f;
+ }
+
+ return Mathf.Abs(curve[curve.length - 1].time - curve[0].time);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AnimationCurveExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AnimationCurveExtensions.cs.meta
new file mode 100644
index 0000000..cfe8fff
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AnimationCurveExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c896105445964076b2a7b48b1e86ba72
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ArrayExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ArrayExtensions.cs
new file mode 100644
index 0000000..252bfc3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ArrayExtensions.cs
@@ -0,0 +1,32 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// type method extensions.
+ ///
+ public static class ArrayExtensions
+ {
+ ///
+ /// Wraps the index around to the beginning of the array if the provided index is longer than the array.
+ ///
+ /// The array to wrap the index around.
+ /// The index to look for.
+ public static int WrapIndex(this Array array, int index)
+ {
+ int length = array.Length;
+ return ((index % length) + length) % length;
+ }
+
+ ///
+ /// Checks whether the given array is not null and has at least one entry
+ ///
+ public static bool IsValidArray(this Array array)
+ {
+ return array != null && array.Length > 0;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ArrayExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ArrayExtensions.cs.meta
new file mode 100644
index 0000000..edd9891
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ArrayExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 942332098e764f6a9a2c69d92dd0fc74
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AssemblyExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AssemblyExtensions.cs
new file mode 100644
index 0000000..905e5c0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AssemblyExtensions.cs
@@ -0,0 +1,28 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Reflection;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ public static class AssemblyExtensions
+ {
+ ///
+ /// Assembly.GetTypes() can throw in some cases. This extension will catch that exception and return only the types which were successfully loaded from the assembly.
+ ///
+ public static IEnumerable GetLoadableTypes(this Assembly @this)
+ {
+ try
+ {
+ return @this.GetTypes();
+ }
+ catch (ReflectionTypeLoadException e)
+ {
+ return e.Types.Where(t => t != null);
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AssemblyExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AssemblyExtensions.cs.meta
new file mode 100644
index 0000000..20c0772
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/AssemblyExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ca0b36fc217123c46a2fa92b0e83f619
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/BoundsExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/BoundsExtensions.cs
new file mode 100644
index 0000000..26969a3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/BoundsExtensions.cs
@@ -0,0 +1,776 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System.Collections.Generic;
+using UnityEngine;
+using UnityEngine.Assertions;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Bounds struct
+ ///
+ public static class BoundsExtensions
+ {
+ // Corners
+ public const int LBF = 0;
+ public const int LBB = 1;
+ public const int LTF = 2;
+ public const int LTB = 3;
+ public const int RBF = 4;
+ public const int RBB = 5;
+ public const int RTF = 6;
+ public const int RTB = 7;
+
+ // X axis
+ public const int LTF_RTF = 8;
+ public const int LBF_RBF = 9;
+ public const int RTB_LTB = 10;
+ public const int RBB_LBB = 11;
+
+ // Y axis
+ public const int LTF_LBF = 12;
+ public const int RTB_RBB = 13;
+ public const int LTB_LBB = 14;
+ public const int RTF_RBF = 15;
+
+ // Z axis
+ public const int RBF_RBB = 16;
+ public const int RTF_RTB = 17;
+ public const int LBF_LBB = 18;
+ public const int LTF_LTB = 19;
+
+ // 2D corners
+ public const int LT = 0;
+ public const int LB = 1;
+ public const int RT = 2;
+ public const int RB = 3;
+
+ // 2D midpoints
+ public const int LT_RT = 4;
+ public const int RT_RB = 5;
+ public const int RB_LB = 6;
+ public const int LB_LT = 7;
+
+ // Face points
+ public const int TOP = 0;
+ public const int BOT = 1;
+ public const int LFT = 2;
+ public const int RHT = 3;
+ public const int FWD = 4;
+ public const int BCK = 5;
+
+ // Axis of the capsule’s lengthwise orientation in the object’s local space
+ private const int CAPSULE_X_AXIS = 0;
+ private const int CAPSULE_Y_AXIS = 1;
+ private const int CAPSULE_Z_AXIS = 2;
+
+ // Edges used to render the bounds.
+ private static readonly int[] boundsEdges = new int[]
+ {
+ LBF, LBB,
+ LBB, LTB,
+ LTB, LTF,
+ LTF, LBF,
+ LBF, RTB,
+ RTB, RTF,
+ RTF, RBF,
+ RBF, RBB,
+ RBB, RTB,
+ RTF, LBB,
+ RBF, LTB,
+ RBB, LTF
+ };
+
+ public enum Axis
+ {
+ X,
+ Y,
+ Z
+ }
+
+ private static Vector3[] corners = null;
+
+ private static Vector3[] rectTransformCorners = new Vector3[4];
+
+ #region Public Static Functions
+ ///
+ /// Returns an instance of the 'Bounds' class which is invalid. An invalid 'Bounds' instance
+ /// is one which has its size vector set to 'float.MaxValue' for all 3 components. The center
+ /// of an invalid bounds instance is the zero vector.
+ ///
+ public static Bounds GetInvalidBoundsInstance()
+ {
+ return new Bounds(Vector3.zero, GetInvalidBoundsSize());
+ }
+
+ ///
+ /// Checks if the specified bounds instance is valid. A valid 'Bounds' instance is
+ /// one whose size vector does not have all 3 components set to 'float.MaxValue'.
+ ///
+ public static bool IsValid(this Bounds bounds)
+ {
+ return bounds.size != GetInvalidBoundsSize();
+ }
+
+ ///
+ /// Gets all the corner points of the bounds in world space by transforming input bounds using the given transform
+ ///
+ /// Local to world transform
+ /// Output corner positions
+ /// Input bounds, in local space
+ ///
+ /// Use BoxColliderExtensions.{Left|Right}{Bottom|Top}{Front|Back} consts to index into the output
+ /// corners array.
+ ///
+ public static void GetCornerPositions(this Bounds bounds, Transform transform, ref Vector3[] positions)
+ {
+ // Calculate the local points to transform.
+ Vector3 center = bounds.center;
+ Vector3 extents = bounds.extents;
+ float leftEdge = center.x - extents.x;
+ float rightEdge = center.x + extents.x;
+ float bottomEdge = center.y - extents.y;
+ float topEdge = center.y + extents.y;
+ float frontEdge = center.z - extents.z;
+ float backEdge = center.z + extents.z;
+
+ // Allocate the array if needed.
+ const int numPoints = 8;
+ if (positions == null || positions.Length != numPoints)
+ {
+ positions = new Vector3[numPoints];
+ }
+
+ // Transform all the local points to world space.
+ positions[LBF] = transform.TransformPoint(leftEdge, bottomEdge, frontEdge);
+ positions[LBB] = transform.TransformPoint(leftEdge, bottomEdge, backEdge);
+ positions[LTF] = transform.TransformPoint(leftEdge, topEdge, frontEdge);
+ positions[LTB] = transform.TransformPoint(leftEdge, topEdge, backEdge);
+ positions[RBF] = transform.TransformPoint(rightEdge, bottomEdge, frontEdge);
+ positions[RBB] = transform.TransformPoint(rightEdge, bottomEdge, backEdge);
+ positions[RTF] = transform.TransformPoint(rightEdge, topEdge, frontEdge);
+ positions[RTB] = transform.TransformPoint(rightEdge, topEdge, backEdge);
+ }
+
+ ///
+ /// Gets all the corner points of the bounds
+ ///
+ ///
+ /// Use BoxColliderExtensions.{Left|Right}{Bottom|Top}{Front|Back} consts to index into the output
+ /// corners array.
+ ///
+ public static void GetCornerPositions(this Bounds bounds, ref Vector3[] positions)
+ {
+ // Calculate the local points to transform.
+ Vector3 center = bounds.center;
+ Vector3 extents = bounds.extents;
+ float leftEdge = center.x - extents.x;
+ float rightEdge = center.x + extents.x;
+ float bottomEdge = center.y - extents.y;
+ float topEdge = center.y + extents.y;
+ float frontEdge = center.z - extents.z;
+ float backEdge = center.z + extents.z;
+
+ // Allocate the array if needed.
+ const int numPoints = 8;
+ if (positions == null || positions.Length != numPoints)
+ {
+ positions = new Vector3[numPoints];
+ }
+
+ // Transform all the local points to world space.
+ positions[LBF] = new Vector3(leftEdge, bottomEdge, frontEdge);
+ positions[LBB] = new Vector3(leftEdge, bottomEdge, backEdge);
+ positions[LTF] = new Vector3(leftEdge, topEdge, frontEdge);
+ positions[LTB] = new Vector3(leftEdge, topEdge, backEdge);
+ positions[RBF] = new Vector3(rightEdge, bottomEdge, frontEdge);
+ positions[RBB] = new Vector3(rightEdge, bottomEdge, backEdge);
+ positions[RTF] = new Vector3(rightEdge, topEdge, frontEdge);
+ positions[RTB] = new Vector3(rightEdge, topEdge, backEdge);
+ }
+
+ ///
+ /// Gets all the corner points from Renderer's Bounds
+ ///
+ public static void GetCornerPositionsFromRendererBounds(this Bounds bounds, ref Vector3[] positions)
+ {
+ Vector3 center = bounds.center;
+ Vector3 extents = bounds.extents;
+ float leftEdge = center.x - extents.x;
+ float rightEdge = center.x + extents.x;
+ float bottomEdge = center.y - extents.y;
+ float topEdge = center.y + extents.y;
+ float frontEdge = center.z - extents.z;
+ float backEdge = center.z + extents.z;
+
+ const int numPoints = 8;
+ if (positions == null || positions.Length != numPoints)
+ {
+ positions = new Vector3[numPoints];
+ }
+
+ positions[LBF] = new Vector3(leftEdge, bottomEdge, frontEdge);
+ positions[LBB] = new Vector3(leftEdge, bottomEdge, backEdge);
+ positions[LTF] = new Vector3(leftEdge, topEdge, frontEdge);
+ positions[LTB] = new Vector3(leftEdge, topEdge, backEdge);
+ positions[RBF] = new Vector3(rightEdge, bottomEdge, frontEdge);
+ positions[RBB] = new Vector3(rightEdge, bottomEdge, backEdge);
+ positions[RTF] = new Vector3(rightEdge, topEdge, frontEdge);
+ positions[RTB] = new Vector3(rightEdge, topEdge, backEdge);
+ }
+
+ public static void GetFacePositions(this Bounds bounds, Transform transform, ref Vector3[] positions)
+ {
+ Vector3 center = bounds.center;
+ Vector3 extents = bounds.extents;
+
+ const int numPoints = 6;
+ if (positions == null || positions.Length != numPoints)
+ {
+ positions = new Vector3[numPoints];
+ }
+
+ positions[TOP] = transform.TransformPoint(center + Vector3.up * extents.y);
+ positions[BOT] = transform.TransformPoint(center + Vector3.down * extents.y);
+ positions[LFT] = transform.TransformPoint(center + Vector3.left * extents.x);
+ positions[RHT] = transform.TransformPoint(center + Vector3.right * extents.x);
+ positions[FWD] = transform.TransformPoint(center + Vector3.forward * extents.z);
+ positions[BCK] = transform.TransformPoint(center + Vector3.back * extents.z);
+ }
+
+ ///
+ /// Gets all the corner points and mid points from Renderer's Bounds
+ ///
+ public static void GetCornerAndMidPointPositions(this Bounds bounds, Transform transform, ref Vector3[] positions)
+ {
+ // Calculate the local points to transform.
+ Vector3 center = bounds.center;
+ Vector3 extents = bounds.extents;
+ float leftEdge = center.x - extents.x;
+ float rightEdge = center.x + extents.x;
+ float bottomEdge = center.y - extents.y;
+ float topEdge = center.y + extents.y;
+ float frontEdge = center.z - extents.z;
+ float backEdge = center.z + extents.z;
+
+ // Allocate the array if needed.
+ const int numPoints = LTF_LTB + 1;
+ if (positions == null || positions.Length != numPoints)
+ {
+ positions = new Vector3[numPoints];
+ }
+
+ // Transform all the local points to world space.
+ positions[LBF] = transform.TransformPoint(leftEdge, bottomEdge, frontEdge);
+ positions[LBB] = transform.TransformPoint(leftEdge, bottomEdge, backEdge);
+ positions[LTF] = transform.TransformPoint(leftEdge, topEdge, frontEdge);
+ positions[LTB] = transform.TransformPoint(leftEdge, topEdge, backEdge);
+ positions[RBF] = transform.TransformPoint(rightEdge, bottomEdge, frontEdge);
+ positions[RBB] = transform.TransformPoint(rightEdge, bottomEdge, backEdge);
+ positions[RTF] = transform.TransformPoint(rightEdge, topEdge, frontEdge);
+ positions[RTB] = transform.TransformPoint(rightEdge, topEdge, backEdge);
+
+ positions[LTF_RTF] = Vector3.Lerp(positions[LTF], positions[RTF], 0.5f);
+ positions[LBF_RBF] = Vector3.Lerp(positions[LBF], positions[RBF], 0.5f);
+ positions[RTB_LTB] = Vector3.Lerp(positions[RTB], positions[LTB], 0.5f);
+ positions[RBB_LBB] = Vector3.Lerp(positions[RBB], positions[LBB], 0.5f);
+
+ positions[LTF_LBF] = Vector3.Lerp(positions[LTF], positions[LBF], 0.5f);
+ positions[RTB_RBB] = Vector3.Lerp(positions[RTB], positions[RBB], 0.5f);
+ positions[LTB_LBB] = Vector3.Lerp(positions[LTB], positions[LBB], 0.5f);
+ positions[RTF_RBF] = Vector3.Lerp(positions[RTF], positions[RBF], 0.5f);
+
+ positions[RBF_RBB] = Vector3.Lerp(positions[RBF], positions[RBB], 0.5f);
+ positions[RTF_RTB] = Vector3.Lerp(positions[RTF], positions[RTB], 0.5f);
+ positions[LBF_LBB] = Vector3.Lerp(positions[LBF], positions[LBB], 0.5f);
+ positions[LTF_LTB] = Vector3.Lerp(positions[LTF], positions[LTB], 0.5f);
+ }
+
+ ///
+ /// Gets all the corner points and mid points from Renderer's Bounds, ignoring the z axis
+ ///
+ public static void GetCornerAndMidPointPositions2D(this Bounds bounds, Transform transform, ref Vector3[] positions, Axis flattenAxis)
+ {
+ // Calculate the local points to transform.
+ Vector3 center = bounds.center;
+ Vector3 extents = bounds.extents;
+
+ float leftEdge = 0;
+ float rightEdge = 0;
+ float bottomEdge = 0;
+ float topEdge = 0;
+
+ // Allocate the array if needed.
+ const int numPoints = LB_LT + 1;
+ if (positions == null || positions.Length != numPoints)
+ {
+ positions = new Vector3[numPoints];
+ }
+
+ switch (flattenAxis)
+ {
+ case Axis.X:
+ default:
+ leftEdge = center.z - extents.z;
+ rightEdge = center.z + extents.z;
+ bottomEdge = center.y - extents.y;
+ topEdge = center.y + extents.y;
+ // Transform all the local points to world space.
+ positions[LT] = transform.TransformPoint(0, topEdge, leftEdge);
+ positions[LB] = transform.TransformPoint(0, bottomEdge, leftEdge);
+ positions[RT] = transform.TransformPoint(0, topEdge, rightEdge);
+ positions[RB] = transform.TransformPoint(0, bottomEdge, rightEdge);
+ break;
+
+ case Axis.Y:
+ leftEdge = center.z - extents.z;
+ rightEdge = center.z + extents.z;
+ bottomEdge = center.x - extents.x;
+ topEdge = center.x + extents.x;
+ // Transform all the local points to world space.
+ positions[LT] = transform.TransformPoint(topEdge, 0, leftEdge);
+ positions[LB] = transform.TransformPoint(bottomEdge, 0, leftEdge);
+ positions[RT] = transform.TransformPoint(topEdge, 0, rightEdge);
+ positions[RB] = transform.TransformPoint(bottomEdge, 0, rightEdge);
+ break;
+
+ case Axis.Z:
+ leftEdge = center.x - extents.x;
+ rightEdge = center.x + extents.x;
+ bottomEdge = center.y - extents.y;
+ topEdge = center.y + extents.y;
+ // Transform all the local points to world space.
+ positions[LT] = transform.TransformPoint(leftEdge, topEdge, 0);
+ positions[LB] = transform.TransformPoint(leftEdge, bottomEdge, 0);
+ positions[RT] = transform.TransformPoint(rightEdge, topEdge, 0);
+ positions[RB] = transform.TransformPoint(rightEdge, bottomEdge, 0);
+ break;
+ }
+
+ positions[LT_RT] = Vector3.Lerp(positions[LT], positions[RT], 0.5f);
+ positions[RT_RB] = Vector3.Lerp(positions[RT], positions[RB], 0.5f);
+ positions[RB_LB] = Vector3.Lerp(positions[RB], positions[LB], 0.5f);
+ positions[LB_LT] = Vector3.Lerp(positions[LB], positions[LT], 0.5f);
+ }
+
+ ///
+ /// Method to get bounds from a collection of points.
+ ///
+ /// The points to construct a bounds around.
+ /// An AABB in world space around all the points.
+ /// True if bounds were calculated, if zero points are present bounds will not be calculated.
+ public static bool GetPointsBounds(List points, out Bounds bounds)
+ {
+ if (points.Count != 0)
+ {
+ bounds = new Bounds(points[0], Vector3.zero);
+
+ for (var i = 1; i < points.Count; ++i)
+ {
+ bounds.Encapsulate(points[i]);
+ }
+
+ return true;
+ }
+
+ bounds = new Bounds();
+ return false;
+ }
+
+ ///
+ /// Method to get bounds using collider method.
+ ///
+ /// GameObject to generate the bounds around.
+ /// An AABB in world space around all the colliders in a gameObject hierarchy.
+ /// A LayerMask to restrict the colliders selected.
+ /// True if bounds were calculated, if zero colliders are present bounds will not be calculated.
+ public static bool GetColliderBounds(GameObject target, out Bounds bounds, LayerMask ignoreLayers)
+ {
+ var boundsPoints = new List();
+ GetColliderBoundsPoints(target, boundsPoints, ignoreLayers);
+
+ return GetPointsBounds(boundsPoints, out bounds);
+ }
+
+ ///
+ /// Calculates how much scale is required for this Bounds to match another Bounds.
+ ///
+ /// Object representation to be scaled to
+ /// padding multiplied into another bounds
+ /// Scale represented as a Vector3
+ public static Vector3 GetScaleToMatchBounds(this Bounds bounds, Bounds otherBounds, Vector3 padding = default(Vector3))
+ {
+ Vector3 szA = otherBounds.size + new Vector3(otherBounds.size.x * padding.x, otherBounds.size.y * padding.y, otherBounds.size.z * padding.z);
+ Vector3 szB = bounds.size;
+ Assert.IsTrue(szB.x != 0 && szB.y != 0 && szB.z != 0, "The bounds of the object must not be zero.");
+ return new Vector3(szA.x / szB.x, szA.y / szB.y, szA.z / szB.z);
+ }
+
+ ///
+ /// Calculates how much scale is required for this Bounds to fit inside another bounds without stretching.
+ ///
+ /// The bounds of the container we're trying to fit this object.
+ /// A single scale factor that can be applied to this object to fit inside the container.
+ public static float GetScaleToFitInside(this Bounds bounds, Bounds containerBounds)
+ {
+ var objectSize = bounds.size;
+ var containerSize = containerBounds.size;
+ Assert.IsTrue(objectSize.x != 0 && objectSize.y != 0 && objectSize.z != 0, "The bounds of the container must not be zero.");
+ return Mathf.Min(containerSize.x / objectSize.x, containerSize.y / objectSize.y, containerSize.z / objectSize.z);
+ }
+
+ ///
+ /// Method to get bounding box points using Collider method.
+ ///
+ /// gameObject that boundingBox bounds.
+ /// array reference that gets filled with points
+ /// layerMask to simplify search
+ /// compute bounds relative to this transform
+ public static void GetColliderBoundsPoints(GameObject target, List boundsPoints, LayerMask ignoreLayers, Transform relativeTo = null)
+ {
+ Collider[] colliders = target.GetComponentsInChildren();
+ for (int i = 0; i < colliders.Length; i++)
+ {
+ GetColliderBoundsPoints(colliders[i], boundsPoints, ignoreLayers, relativeTo);
+ }
+ }
+
+ private static void InverseTransformPoints(ref Vector3[] positions, Transform relativeTo)
+ {
+ if (relativeTo)
+ {
+ for (var i = 0; i < positions.Length; ++i)
+ {
+ positions[i] = relativeTo.InverseTransformPoint(positions[i]);
+ }
+ }
+ }
+
+
+ ///
+ /// Method to get bounds from a single Collider
+ ///
+ /// Target collider
+ /// array reference that gets filled with points
+ /// layerMask to simplify search
+ public static void GetColliderBoundsPoints(Collider collider, List boundsPoints, LayerMask ignoreLayers, Transform relativeTo = null)
+ {
+ if (ignoreLayers == (1 << collider.gameObject.layer | ignoreLayers)) { return; }
+
+ if (collider is SphereCollider)
+ {
+ SphereCollider sc = collider as SphereCollider;
+ Bounds sphereBounds = new Bounds(sc.center, Vector3.one * sc.radius * 2);
+ sphereBounds.GetFacePositions(sc.transform, ref corners);
+ InverseTransformPoints(ref corners, relativeTo);
+ boundsPoints.AddRange(corners);
+ }
+ else if (collider is BoxCollider)
+ {
+ BoxCollider bc = collider as BoxCollider;
+ Bounds boxBounds = new Bounds(bc.center, bc.size);
+ boxBounds.GetCornerPositions(bc.transform, ref corners);
+ InverseTransformPoints(ref corners, relativeTo);
+ boundsPoints.AddRange(corners);
+
+ }
+ else if (collider is MeshCollider)
+ {
+ MeshCollider mc = collider as MeshCollider;
+ Bounds meshBounds = mc.sharedMesh.bounds;
+ meshBounds.GetCornerPositions(mc.transform, ref corners);
+ InverseTransformPoints(ref corners, relativeTo);
+ boundsPoints.AddRange(corners);
+ }
+ else if (collider is CapsuleCollider)
+ {
+ CapsuleCollider cc = collider as CapsuleCollider;
+ Bounds capsuleBounds = new Bounds(cc.center, Vector3.zero);
+ switch (cc.direction)
+ {
+ case CAPSULE_X_AXIS:
+ capsuleBounds.size = new Vector3(cc.height, cc.radius * 2, cc.radius * 2);
+ break;
+
+ case CAPSULE_Y_AXIS:
+ capsuleBounds.size = new Vector3(cc.radius * 2, cc.height, cc.radius * 2);
+ break;
+
+ case CAPSULE_Z_AXIS:
+ capsuleBounds.size = new Vector3(cc.radius * 2, cc.radius * 2, cc.height);
+ break;
+ }
+ capsuleBounds.GetFacePositions(cc.transform, ref corners);
+ InverseTransformPoints(ref corners, relativeTo);
+ boundsPoints.AddRange(corners);
+ }
+ }
+
+ ///
+ /// Method to get bounds using renderer method.
+ ///
+ /// GameObject to generate the bounds around.
+ /// An AABB in world space around all the renderers in a gameObject hierarchy.
+ /// A LayerMask to restrict the colliders selected.
+ /// True if bounds were calculated, if zero renderers are present bounds will not be calculated.
+ public static bool GetRenderBounds(GameObject target, out Bounds bounds, LayerMask ignoreLayers)
+ {
+ var boundsPoints = new List();
+ GetRenderBoundsPoints(target, boundsPoints, ignoreLayers);
+
+ return GetPointsBounds(boundsPoints, out bounds);
+ }
+
+ ///
+ /// GetRenderBoundsPoints gets bounding box points using Render method.
+ ///
+ /// gameObject that boundingbox bounds
+ /// array reference that gets filled with points
+ /// layerMask to simplify search
+ public static void GetRenderBoundsPoints(GameObject target, List boundsPoints, LayerMask ignoreLayers)
+ {
+ Renderer[] renderers = target.GetComponentsInChildren();
+ for (int i = 0; i < renderers.Length; ++i)
+ {
+ Renderer rendererObj = renderers[i];
+ if (ignoreLayers == (1 << rendererObj.gameObject.layer | ignoreLayers))
+ {
+ continue;
+ }
+
+ rendererObj.bounds.GetCornerPositionsFromRendererBounds(ref corners);
+ boundsPoints.AddRange(corners);
+ }
+ }
+
+ ///
+ /// Method to get bounds using mesh filters method.
+ ///
+ /// GameObject to generate the bounds around.
+ /// An AABB in world space around all the mesh filters in a GameObject hierarchy.
+ /// A LayerMask to restrict the colliders selected.
+ /// True if bounds were calculated, if zero mesh filters are present bounds will not be calculated.
+ public static bool GetMeshFilterBounds(GameObject target, out Bounds bounds, LayerMask ignoreLayers)
+ {
+ var boundsPoints = new List();
+ GetMeshFilterBoundsPoints(target, boundsPoints, ignoreLayers);
+
+ return GetPointsBounds(boundsPoints, out bounds);
+ }
+
+ ///
+ /// GetMeshFilterBoundsPoints - gets bounding box points using MeshFilter method.
+ ///
+ /// gameObject that boundingbox bounds
+ /// array reference that gets filled with points
+ /// layerMask to simplify search
+ public static void GetMeshFilterBoundsPoints(GameObject target, List boundsPoints, LayerMask ignoreLayers)
+ {
+ MeshFilter[] meshFilters = target.GetComponentsInChildren();
+ for (int i = 0; i < meshFilters.Length; i++)
+ {
+ MeshFilter meshFilterObj = meshFilters[i];
+ if (ignoreLayers == (1 << meshFilterObj.gameObject.layer | ignoreLayers))
+ {
+ continue;
+ }
+
+ Bounds meshBounds = meshFilterObj.sharedMesh.bounds;
+ meshBounds.GetCornerPositions(meshFilterObj.transform, ref corners);
+ boundsPoints.AddRange(corners);
+ }
+ RectTransform[] rectTransforms = target.GetComponentsInChildren();
+ for (int i = 0; i < rectTransforms.Length; i++)
+ {
+ rectTransforms[i].GetWorldCorners(rectTransformCorners);
+ boundsPoints.AddRange(rectTransformCorners);
+ }
+ }
+
+ ///
+ /// Transforms 'bounds' using the specified transform matrix.
+ ///
+ ///
+ /// Transforming a 'Bounds' instance means that the function will construct a new 'Bounds'
+ /// instance which has its center translated using the translation information stored in
+ /// the specified matrix and its size adjusted to account for rotation and scale. The size
+ /// of the new 'Bounds' instance will be calculated in such a way that it will contain the
+ /// old 'Bounds'.
+ ///
+ ///
+ /// The 'Bounds' instance which must be transformed.
+ ///
+ ///
+ /// The specified 'Bounds' instance will be transformed using this transform matrix. The function
+ /// assumes that the matrix doesn't contain any projection or skew transformation.
+ ///
+ ///
+ /// The transformed 'Bounds' instance.
+ ///
+ public static Bounds Transform(this Bounds bounds, Matrix4x4 transformMatrix)
+ {
+ // We will need access to the right, up and look vector which are encoded inside the transform matrix
+ Vector3 rightAxis = transformMatrix.GetColumn(0);
+ Vector3 upAxis = transformMatrix.GetColumn(1);
+ Vector3 lookAxis = transformMatrix.GetColumn(2);
+
+ // We will 'imagine' that we want to rotate the bounds' extents vector using the rotation information
+ // stored inside the specified transform matrix. We will need these when calculating the new size if
+ // the transformed bounds.
+ Vector3 rotatedExtentsRight = rightAxis * bounds.extents.x;
+ Vector3 rotatedExtentsUp = upAxis * bounds.extents.y;
+ Vector3 rotatedExtentsLook = lookAxis * bounds.extents.z;
+
+ // Calculate the new bounds size along each axis. The size on each axis is calculated by summing up the
+ // corresponding vector component values of the rotated extents vectors. We multiply by 2 because we want
+ // to get a size and currently we are working with extents which represent half the size.
+ float newSizeX = (Mathf.Abs(rotatedExtentsRight.x) + Mathf.Abs(rotatedExtentsUp.x) + Mathf.Abs(rotatedExtentsLook.x)) * 2.0f;
+ float newSizeY = (Mathf.Abs(rotatedExtentsRight.y) + Mathf.Abs(rotatedExtentsUp.y) + Mathf.Abs(rotatedExtentsLook.y)) * 2.0f;
+ float newSizeZ = (Mathf.Abs(rotatedExtentsRight.z) + Mathf.Abs(rotatedExtentsUp.z) + Mathf.Abs(rotatedExtentsLook.z)) * 2.0f;
+
+ // Construct the transformed 'Bounds' instance
+ var transformedBounds = new Bounds();
+ transformedBounds.center = transformMatrix.MultiplyPoint(bounds.center);
+ transformedBounds.size = new Vector3(newSizeX, newSizeY, newSizeZ);
+
+ // Return the instance to the caller
+ return transformedBounds;
+ }
+
+ ///
+ /// Returns the screen space corner points of the specified 'Bounds' instance.
+ ///
+ ///
+ /// The camera used for rendering to the screen. This is needed to perform the
+ /// transformation to screen space.
+ ///
+ public static Vector2[] GetScreenSpaceCornerPoints(this Bounds bounds, Camera camera)
+ {
+ Vector3 aabbCenter = bounds.center;
+ Vector3 aabbExtents = bounds.extents;
+
+ // Return the screen space point array
+ return new Vector2[]
+ {
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x - aabbExtents.x, aabbCenter.y - aabbExtents.y, aabbCenter.z - aabbExtents.z)),
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x + aabbExtents.x, aabbCenter.y - aabbExtents.y, aabbCenter.z - aabbExtents.z)),
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x + aabbExtents.x, aabbCenter.y + aabbExtents.y, aabbCenter.z - aabbExtents.z)),
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x - aabbExtents.x, aabbCenter.y + aabbExtents.y, aabbCenter.z - aabbExtents.z)),
+
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x - aabbExtents.x, aabbCenter.y - aabbExtents.y, aabbCenter.z + aabbExtents.z)),
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x + aabbExtents.x, aabbCenter.y - aabbExtents.y, aabbCenter.z + aabbExtents.z)),
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x + aabbExtents.x, aabbCenter.y + aabbExtents.y, aabbCenter.z + aabbExtents.z)),
+ camera.WorldToScreenPoint(new Vector3(aabbCenter.x - aabbExtents.x, aabbCenter.y + aabbExtents.y, aabbCenter.z + aabbExtents.z))
+ };
+ }
+
+ ///
+ /// Returns the rectangle which encloses the specifies 'Bounds' instance in screen space.
+ ///
+ public static Rect GetScreenRectangle(this Bounds bounds, Camera camera)
+ {
+ // Retrieve the bounds' corner points in screen space
+ Vector2[] screenSpaceCornerPoints = bounds.GetScreenSpaceCornerPoints(camera);
+
+ // Identify the minimum and maximum points in the array
+ Vector3 minScreenPoint = screenSpaceCornerPoints[0], maxScreenPoint = screenSpaceCornerPoints[0];
+ for (int screenPointIndex = 1; screenPointIndex < screenSpaceCornerPoints.Length; ++screenPointIndex)
+ {
+ minScreenPoint = Vector3.Min(minScreenPoint, screenSpaceCornerPoints[screenPointIndex]);
+ maxScreenPoint = Vector3.Max(maxScreenPoint, screenSpaceCornerPoints[screenPointIndex]);
+ }
+
+ // Return the screen space rectangle
+ return new Rect(minScreenPoint.x, minScreenPoint.y, maxScreenPoint.x - minScreenPoint.x, maxScreenPoint.y - minScreenPoint.y);
+ }
+
+ ///
+ /// Returns the volume of the bounds.
+ ///
+ public static float Volume(this Bounds bounds)
+ {
+ return bounds.size.x * bounds.size.y * bounds.size.z;
+ }
+
+ ///
+ /// Returns bounds that contain both this bounds and the bounds passed in.
+ ///
+ public static Bounds ExpandToContain(this Bounds originalBounds, Bounds otherBounds)
+ {
+ Bounds tmpBounds = originalBounds;
+
+ tmpBounds.Encapsulate(otherBounds);
+
+ return tmpBounds;
+ }
+
+ ///
+ /// Checks to see if bounds contains the other bounds completely.
+ ///
+ public static bool ContainsBounds(this Bounds bounds, Bounds otherBounds)
+ {
+ return bounds.Contains(otherBounds.min) && bounds.Contains(otherBounds.max);
+ }
+
+ ///
+ /// Checks to see whether point is closer to bounds or otherBounds
+ ///
+ public static bool CloserToPoint(this Bounds bounds, Vector3 point, Bounds otherBounds)
+ {
+ Vector3 distToClosestPoint1 = bounds.ClosestPoint(point) - point;
+ Vector3 distToClosestPoint2 = otherBounds.ClosestPoint(point) - point;
+
+ if (distToClosestPoint1.magnitude == distToClosestPoint2.magnitude)
+ {
+ Vector3 toCenter1 = point - bounds.center;
+ Vector3 toCenter2 = point - otherBounds.center;
+ return (toCenter1.magnitude <= toCenter2.magnitude);
+
+ }
+
+ return (distToClosestPoint1.magnitude <= distToClosestPoint2.magnitude);
+ }
+
+ ///
+ /// Draws a wire frame Bounds object using Debug.DrawLine.
+ ///
+ /// The Bounds to draw.
+ /// Color of the line.
+ /// How long the line should be visible for in seconds.
+ /// Should the line be obscured by objects closer to the camera?
+ public static void DebugDraw(this Bounds bounds, Color color, float duration = 0.0f, bool depthTest = true)
+ {
+ var center = bounds.center;
+ var x = bounds.extents.x;
+ var y = bounds.extents.y;
+ var z = bounds.extents.z;
+ var a = new Vector3(-x, y, -z);
+ var b = new Vector3(x, -y, -z);
+ var c = new Vector3(x, y, -z);
+
+ var verticies = new Vector3[]
+ {
+ bounds.min, center + a, center + b, center + c,
+ bounds.max, center - a, center - b, center - c
+ };
+
+ for (var i = 0; i < boundsEdges.Length; i += 2)
+ {
+ Debug.DrawLine(verticies[boundsEdges[i]], verticies[boundsEdges[i + 1]], color, duration, depthTest);
+ }
+ }
+
+ #endregion
+
+ #region Private Static Functions
+ ///
+ /// Returns the vector which is used to represent and invalid bounds size.
+ ///
+ private static Vector3 GetInvalidBoundsSize()
+ {
+ return new Vector3(float.MaxValue, float.MaxValue, float.MaxValue);
+ }
+ #endregion
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/BoundsExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/BoundsExtensions.cs.meta
new file mode 100644
index 0000000..41ec2cb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/BoundsExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 8c8d13b5343245b79380f49b1f4359c0
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CameraExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CameraExtensions.cs
new file mode 100644
index 0000000..5014ecd
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CameraExtensions.cs
@@ -0,0 +1,90 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for the Unity's Camera class
+ ///
+ public static class CameraExtensions
+ {
+ ///
+ /// Get the horizontal FOV from the stereo camera in radians
+ ///
+ public static float GetHorizontalFieldOfViewRadians(this Camera camera)
+ {
+ return 2f * Mathf.Atan(Mathf.Tan(camera.fieldOfView * Mathf.Deg2Rad * 0.5f) * camera.aspect);
+ }
+
+ ///
+ /// Get the horizontal FOV from the stereo camera in degrees
+ ///
+ public static float GetHorizontalFieldOfViewDegrees(this Camera camera)
+ {
+ return camera.GetHorizontalFieldOfViewRadians() * Mathf.Rad2Deg;
+ }
+
+ ///
+ /// Returns if a point will be rendered on the screen in either eye
+ ///
+ /// The camera to check the point against
+ public static bool IsInFOV(this Camera camera, Vector3 position)
+ {
+ Vector3 screenPoint = camera.WorldToViewportPoint(position);
+
+ return screenPoint.z >= camera.nearClipPlane && screenPoint.z <= camera.farClipPlane
+ && screenPoint.x >= 0 && screenPoint.x <= 1
+ && screenPoint.y >= 0 && screenPoint.y <= 1;
+ }
+
+
+ ///
+ /// Returns true if a point is in the a cone inscribed into the Camera's frustum, false otherwise
+ /// The cone is inscribed to a radius equal to the vertical height of the camera's FOV.
+ /// By default, the cone's tip is "chopped off" by an amount defined by the camera's
+ /// far and near clip planes.
+ ///
+ /// Point to test
+ /// Degrees to expand the cone radius by.
+ public static bool IsInFOVCone(this Camera camera,
+ Vector3 point,
+ float coneAngleBufferDegrees = 0)
+ {
+ return MathUtilities.IsInFOVCone(camera.transform,
+ point,
+ camera.fieldOfView + coneAngleBufferDegrees,
+ camera.nearClipPlane,
+ camera.farClipPlane
+ );
+ }
+
+ ///
+ /// Gets the frustum size at a given distance from the camera.
+ ///
+ /// The camera to get the frustum size for
+ /// The distance from the camera to get the frustum size at
+ public static Vector2 GetFrustumSizeForDistance(this Camera camera, float distanceFromCamera)
+ {
+ Vector2 frustumSize = new Vector2
+ {
+ y = 2.0f * distanceFromCamera * Mathf.Tan(camera.fieldOfView * 0.5f * Mathf.Deg2Rad)
+ };
+ frustumSize.x = frustumSize.y * camera.aspect;
+
+ return frustumSize;
+ }
+
+ ///
+ /// Gets the distance to the camera that a specific frustum height would be at.
+ ///
+ /// The camera to get the distance from
+ /// The frustum height
+ public static float GetDistanceForFrustumHeight(this Camera camera, float frustumHeight)
+ {
+ return frustumHeight * 0.5f / Mathf.Max(0.00001f, Mathf.Tan(camera.fieldOfView * 0.5f * Mathf.Deg2Rad));
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CameraExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CameraExtensions.cs.meta
new file mode 100644
index 0000000..3688998
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CameraExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a82cef65e97445fe8ce8d90b15644cd7
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CanvasExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CanvasExtensions.cs
new file mode 100644
index 0000000..6e99775
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CanvasExtensions.cs
@@ -0,0 +1,208 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEngine;
+using UnityEngine.UI;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extensions for the Canvas class.
+ ///
+ public static class CanvasExtensions
+ {
+ ///
+ /// Convenience method for getting a plane for this canvas in world coordinates.
+ ///
+ /// The canvas to get the plane from.
+ /// A Plane for this canvas.
+ public static Plane GetPlane(this Canvas canvas)
+ {
+ Vector3[] corners = canvas.GetWorldCorners();
+
+ // Now set a plane from any of the 3 corners (clockwise) so that we can compute our gaze intersection
+ Plane plane = new Plane(corners[0], corners[1], corners[2]);
+
+ return plane;
+ }
+
+ ///
+ /// Convenience method for getting the corners of the canvas in world coordinates. Ordered clockwise from bottom-left.
+ ///
+ /// The canvas to get the world corners from.
+ /// An array of Vector3s that represent the corners of the canvas in world coordinates.
+ public static Vector3[] GetWorldCorners(this Canvas canvas)
+ {
+ Vector3[] worldCorners = new Vector3[4];
+ RectTransform rect = canvas.GetComponent();
+ rect.GetWorldCorners(worldCorners);
+ return worldCorners;
+ }
+
+ ///
+ /// Convenience method for getting the corners of the canvas in local coordinates. Ordered clockwise from bottom-left.
+ ///
+ /// The canvas to get the local corners from.
+ /// An array of Vector3s that represent the corners of the canvas in local coordinates.
+ public static Vector3[] GetLocalCorners(this Canvas canvas)
+ {
+ Vector3[] localCorners = new Vector3[4];
+ RectTransform rect = canvas.GetComponent();
+ rect.GetLocalCorners(localCorners);
+ return localCorners;
+ }
+
+ ///
+ /// Convenience method for getting the corners of the canvas in viewport coordinates. Note
+ /// that the points have the same ordering as the array returned in GetWorldCorners()
+ ///
+ /// The canvas to get the viewport corners from
+ /// An array of Vector3s that represent the corners of the canvas in viewport coordinates
+ public static Vector3[] GetViewportCorners(this Canvas canvas)
+ {
+ Vector3[] viewportCorners = new Vector3[4];
+
+ Vector3[] worldCorners = canvas.GetWorldCorners();
+
+ for (int i = 0; i < 4; i++)
+ {
+ viewportCorners[i] = CameraCache.Main.WorldToViewportPoint(worldCorners[i]);
+ }
+
+ return viewportCorners;
+ }
+
+ ///
+ /// Gets the position of the corners for a canvas in screen space.
+ /// 1 -- 2
+ /// | |
+ /// 0 -- 3
+ ///
+ /// The canvas to get the screen corners for.
+ public static Vector3[] GetScreenCorners(this Canvas canvas)
+ {
+ Vector3[] screenCorners = new Vector3[4];
+ Vector3[] worldCorners = canvas.GetWorldCorners();
+
+ for (int i = 0; i < 4; i++)
+ {
+ screenCorners[i] = CameraCache.Main.WorldToScreenPoint(worldCorners[i]);
+ }
+
+ return screenCorners;
+ }
+
+ ///
+ /// Returns a rectangle in screen coordinates that encompasses the bounds of the target canvas.
+ ///
+ /// The canvas the get the screen rect for
+ public static Rect GetScreenRect(this Canvas canvas)
+ {
+ Vector3[] screenCorners = canvas.GetScreenCorners();
+ float x = Mathf.Min(screenCorners[0].x, screenCorners[1].x);
+ float y = Mathf.Min(screenCorners[0].y, screenCorners[3].y);
+ float xMax = Mathf.Max(screenCorners[2].x, screenCorners[3].x);
+ float yMax = Mathf.Max(screenCorners[1].y, screenCorners[2].y);
+ return new Rect(x, y, xMax - x, yMax - y);
+ }
+
+ ///
+ /// Raycast against a canvas using a ray.
+ ///
+ /// The canvas to raycast against
+ /// The origin of the ray
+ /// The direction of the ray
+ /// The distance of the ray
+ /// The hitpoint of the ray
+ /// The child object that was hit or the canvas itself if it has no active children that were within the hit range.
+ public static bool Raycast(this Canvas canvas, Vector3 rayOrigin, Vector3 rayDirection, out float distance, out Vector3 hitPoint, out GameObject hitChildObject)
+ {
+ hitChildObject = null;
+ Plane plane = canvas.GetPlane();
+ Ray ray = new Ray(rayOrigin, rayDirection);
+
+ if (plane.Raycast(ray, out distance))
+ {
+ // See if the point lies within the local canvas rect of the plane
+ Vector3[] corners = canvas.GetLocalCorners();
+ hitPoint = rayOrigin + (rayDirection.normalized * distance);
+ Vector3 localHitPoint = canvas.transform.InverseTransformPoint(hitPoint);
+ if (localHitPoint.x >= corners[0].x
+ && localHitPoint.x <= corners[3].x
+ && localHitPoint.y <= corners[2].y
+ && localHitPoint.y >= corners[3].y)
+ {
+ hitChildObject = canvas.gameObject;
+
+ // look for the child object that was hit
+ RectTransform rectTransform = GetChildRectTransformAtPoint(canvas.GetComponent(), hitPoint, true, true, true);
+ if (rectTransform != null)
+ {
+ hitChildObject = rectTransform.gameObject;
+ }
+ else
+ {
+ hitChildObject = canvas.gameObject;
+ }
+
+ return true;
+ }
+ }
+
+ hitPoint = Vector3.zero;
+
+ return false;
+ }
+
+ ///
+ /// Gets a child rect transform for the given point and parameters.
+ ///
+ /// The rect transform to look for children that may contain the projected (orthogonal to the child's normal) world point
+ /// The world point
+ /// Indicates if the check should be done recursively
+ /// If true, will only check children that are active, otherwise it will check all children.
+ /// If true, will only check children that if they have a graphic and have its member raycastTarget set to true, otherwise will ignore the raycastTarget value. Will still allow children to be checked that do not have a graphic component.
+ public static RectTransform GetChildRectTransformAtPoint(this RectTransform rectTransformParent, Vector3 worldPoint, bool recursive, bool shouldReturnActive, bool shouldReturnRaycastable)
+ {
+ Vector3[] localCorners = new Vector3[4];
+ Vector3 childLocalPoint;
+ RectTransform rectTransform;
+ bool shouldRaycast = false;
+
+ for (int i = rectTransformParent.childCount - 1; i >= 0; i--)
+ {
+ rectTransform = rectTransformParent.GetChild(i).GetComponent();
+ Graphic graphic = rectTransform.GetComponent();
+ shouldRaycast = ((shouldReturnRaycastable && graphic != null && graphic.raycastTarget) || graphic == null || !shouldReturnRaycastable);
+
+ if (((shouldReturnActive && rectTransform.gameObject.activeSelf) || !shouldReturnActive))
+ {
+ rectTransform.GetLocalCorners(localCorners);
+ childLocalPoint = rectTransform.InverseTransformPoint(worldPoint);
+
+ if (recursive)
+ {
+ RectTransform childRect = GetChildRectTransformAtPoint(rectTransform, worldPoint, recursive, shouldReturnActive, shouldReturnRaycastable);
+
+ if (childRect != null)
+ {
+ return childRect;
+ }
+ }
+
+ if (shouldRaycast
+ && childLocalPoint.x >= localCorners[0].x
+ && childLocalPoint.x <= localCorners[3].x
+ && childLocalPoint.y <= localCorners[2].y
+ && childLocalPoint.y >= localCorners[3].y)
+ {
+ return rectTransform;
+ }
+ }
+ }
+
+ return null;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CanvasExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CanvasExtensions.cs.meta
new file mode 100644
index 0000000..bbafb92
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CanvasExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1f326861b6dc467459b57eab1db672d6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CollectionsExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CollectionsExtensions.cs
new file mode 100644
index 0000000..b71a03b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CollectionsExtensions.cs
@@ -0,0 +1,165 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Input;
+using System;
+using System.Collections.Generic;
+using System.Collections.ObjectModel;
+using System.Linq;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for .Net Collection objects, e.g. Lists, Dictionaries, Arrays
+ ///
+ public static class CollectionsExtensions
+ {
+ ///
+ /// Creates a read-only wrapper around an existing collection.
+ ///
+ /// The type of element in the collection.
+ /// The collection to be wrapped.
+ /// The new, read-only wrapper around .
+ public static ReadOnlyCollection AsReadOnly(this IList elements)
+ {
+ return new ReadOnlyCollection(elements);
+ }
+
+ ///
+ /// Creates a read-only copy of an existing collection.
+ ///
+ /// The type of element in the collection.
+ /// The collection to be copied.
+ /// The new, read-only copy of .
+ public static ReadOnlyCollection ToReadOnlyCollection(this IEnumerable elements)
+ {
+ return elements.ToArray().AsReadOnly();
+ }
+
+ ///
+ /// Inserts an item in its sorted position into an already sorted collection. This is useful if you need to consume the
+ /// collection in between insertions and need it to stay correctly sorted the whole time. If you just need to insert a
+ /// bunch of items and then consume the sorted collection at the end, it's faster to add all the elements and then use
+ /// at the end.
+ ///
+ /// The type of element in the collection.
+ /// The collection of sorted elements to be inserted into.
+ /// The element to insert.
+ /// The comparer to use when sorting or null to use .
+ public static int SortedInsert(this List elements, TElement toInsert, IComparer comparer = null)
+ {
+ var effectiveComparer = comparer ?? Comparer.Default;
+
+ if (Application.isEditor)
+ {
+ for (int iElement = 0; iElement < elements.Count - 1; iElement++)
+ {
+ var element = elements[iElement];
+ var nextElement = elements[iElement + 1];
+
+ if (effectiveComparer.Compare(element, nextElement) > 0)
+ {
+ Debug.LogWarning("Elements must already be sorted to call this method.");
+ break;
+ }
+ }
+ }
+
+ int searchResult = elements.BinarySearch(toInsert, effectiveComparer);
+
+ int insertionIndex = searchResult >= 0
+ ? searchResult
+ : ~searchResult;
+
+ elements.Insert(insertionIndex, toInsert);
+
+ return insertionIndex;
+ }
+
+ ///
+ /// Disposes of all non-null elements in a collection.
+ ///
+ /// The type of element in the collection.
+ /// The collection of elements to be disposed.
+ public static void DisposeElements(this IEnumerable elements)
+ where TElement : IDisposable
+ {
+ foreach (var element in elements)
+ {
+ if (element != null)
+ {
+ element.Dispose();
+ }
+ }
+ }
+
+ ///
+ /// Disposes of all non-null elements in a collection.
+ ///
+ /// The type of element in the collection.
+ /// The collection of elements to be disposed.
+ public static void DisposeElements(this IList elements)
+ where TElement : IDisposable
+ {
+ for (int iElement = 0; iElement < elements.Count; iElement++)
+ {
+ var element = elements[iElement];
+
+ if (element != null)
+ {
+ element.Dispose();
+ }
+ }
+ }
+
+ ///
+ /// Exports the values of a uint indexed Dictionary as an Array
+ ///
+ /// Type of data stored in the values of the Dictionary
+ /// Dictionary to be exported
+ /// array in the type of data stored in the Dictionary
+ public static T[] ExportDictionaryValuesAsArray(this Dictionary input)
+ {
+ T[] output = new T[input.Count];
+ input.Values.CopyTo(output, 0);
+ return output;
+ }
+
+ ///
+ /// Overload extension to enable getting of an InteractionDefinition of a specific type
+ ///
+ /// The InteractionDefinition array reference
+ /// The specific DeviceInputType value to query
+ public static MixedRealityInteractionMapping GetInteractionByType(this MixedRealityInteractionMapping[] input, DeviceInputType key)
+ {
+ for (int i = 0; i < input?.Length; i++)
+ {
+ if (input[i].InputType == key)
+ {
+ return input[i];
+ }
+ }
+
+ return default(MixedRealityInteractionMapping);
+ }
+
+ ///
+ /// Overload extension to enable getting of an InteractionDefinition of a specific type
+ ///
+ /// The InteractionDefinition array reference
+ /// The specific DeviceInputType value to query
+ public static bool SupportsInputType(this MixedRealityInteractionMapping[] input, DeviceInputType key)
+ {
+ for (int i = 0; i < input.Length; i++)
+ {
+ if (input[i].InputType == key)
+ {
+ return true;
+ }
+ }
+
+ return false;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CollectionsExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CollectionsExtensions.cs.meta
new file mode 100644
index 0000000..271b2f4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/CollectionsExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a2be8781cdb84a30811fcdbe0a3e0320
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Color32Extensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Color32Extensions.cs
new file mode 100644
index 0000000..e0799af
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Color32Extensions.cs
@@ -0,0 +1,65 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Globalization;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Color32 struct
+ ///
+ public static class Color32Extensions
+ {
+ public static Color PremultiplyAlpha(Color col)
+ {
+ col.r *= col.a;
+ col.g *= col.a;
+ col.b *= col.a;
+
+ return col;
+ }
+
+ public static Color32 PremultiplyAlpha(Color32 col)
+ {
+ Color floatCol = col;
+ return (Color32)PremultiplyAlpha(floatCol);
+ }
+
+ ///
+ /// Creates a Color from a hexcode string
+ ///
+ public static Color ParseHexcode(string hexstring)
+ {
+ if (hexstring.StartsWith("#"))
+ {
+ hexstring = hexstring.Substring(1);
+ }
+
+ if (hexstring.StartsWith("0x"))
+ {
+ hexstring = hexstring.Substring(2);
+ }
+
+ if (hexstring.Length == 6)
+ {
+ hexstring += "FF";
+ }
+
+ if (hexstring.Length != 8)
+ {
+ throw new ArgumentException(string.Format("{0} is not a valid color string.", hexstring));
+ }
+
+ byte r = byte.Parse(hexstring.Substring(0, 2), NumberStyles.HexNumber);
+ byte g = byte.Parse(hexstring.Substring(2, 2), NumberStyles.HexNumber);
+ byte b = byte.Parse(hexstring.Substring(4, 2), NumberStyles.HexNumber);
+ byte a = byte.Parse(hexstring.Substring(6, 2), NumberStyles.HexNumber);
+
+ const float maxRgbValue = 255;
+ Color c = new Color(r / maxRgbValue, g / maxRgbValue, b / maxRgbValue, a / maxRgbValue);
+ return c;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Color32Extensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Color32Extensions.cs.meta
new file mode 100644
index 0000000..a1ea0d1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Color32Extensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1df8bd8f5849456aadf4e518f9100a46
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComparerExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComparerExtensions.cs
new file mode 100644
index 0000000..34e02e8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComparerExtensions.cs
@@ -0,0 +1,42 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System.Collections.Generic;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for .Net Comparer's
+ ///
+ public static class ComparerExtensions
+ {
+ ///
+ /// Gets a comparer that sorts elements in the opposite order of the original comparer.
+ ///
+ /// The type of element the comparer compares.
+ /// The comparer whose order should be reversed.
+ /// A comparer that sorts elements in the opposite order of .
+ public static IComparer GetReversed(this IComparer originalComparer)
+ {
+ return new ReverseComparer(originalComparer);
+ }
+
+ private class ReverseComparer : IComparer
+ {
+ private readonly IComparer originalComparer;
+
+ public ReverseComparer(IComparer originalComparer)
+ {
+ Debug.Assert(originalComparer != null, "originalComparer cannot be null.");
+
+ this.originalComparer = originalComparer;
+ }
+
+ public int Compare(TElement left, TElement right)
+ {
+ return originalComparer.Compare(right, left);
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComparerExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComparerExtensions.cs.meta
new file mode 100644
index 0000000..305443e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComparerExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b4a220e9226b49efb855ca01a7f69ff4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComponentExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComponentExtensions.cs
new file mode 100644
index 0000000..32133dd
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComponentExtensions.cs
@@ -0,0 +1,67 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extensions methods for the Unity Component class.
+ /// This also includes some component-related extensions for the GameObject class.
+ ///
+ public static class ComponentExtensions
+ {
+ ///
+ /// Ensure that a component of type exists on the game object.
+ /// If it doesn't exist, creates it.
+ ///
+ /// Type of the component.
+ /// A component on the game object for which a component of type should exist.
+ /// The component that was retrieved or created.
+ public static T EnsureComponent(this Component component) where T : Component
+ {
+ return EnsureComponent(component.gameObject);
+ }
+
+ ///
+ /// Find the first component of type in the ancestors of the game object of the specified component.
+ ///
+ /// Type of component to find.
+ /// Component for which its game object's ancestors must be considered.
+ /// Indicates whether the specified game object should be included.
+ /// The component of type . Null if it none was found.
+ public static T FindAncestorComponent(this Component component, bool includeSelf = true) where T : Component
+ {
+ return component.transform.FindAncestorComponent(includeSelf);
+ }
+
+ ///
+ /// Ensure that a component of type exists on the game object.
+ /// If it doesn't exist, creates it.
+ ///
+ /// Type of the component.
+ /// Game object on which component should be.
+ /// The component that was retrieved or created.
+ ///
+ /// This extension has to remain in this class as it is required by the method
+ ///
+ public static T EnsureComponent(this GameObject gameObject) where T : Component
+ {
+ T foundComponent = gameObject.GetComponent();
+ return foundComponent == null ? gameObject.AddComponent() : foundComponent;
+ }
+
+ ///
+ /// Ensure that a component of type exists on the game object.
+ /// If it doesn't exist, creates it.
+ ///
+ /// A component on the game object for which a component of type should exist.
+ /// The component that was retrieved or created.
+ public static Component EnsureComponent(this GameObject gameObject, Type component)
+ {
+ var foundComponent = gameObject.GetComponent(component);
+ return foundComponent == null ? gameObject.AddComponent(component) : foundComponent;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComponentExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComponentExtensions.cs.meta
new file mode 100644
index 0000000..5458515
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ComponentExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f211201cc93f4cd4b5eff7d765a3a617
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DateTimeExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DateTimeExtensions.cs
new file mode 100644
index 0000000..0d9ebe6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DateTimeExtensions.cs
@@ -0,0 +1,41 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extensions.
+ ///
+ public static class DateTimeExtensions
+ {
+ ///
+ /// Gets string literal for relative time from now since the DateTime provided. String output is in most appropriate "x time units ago"
+ /// Example: If DateTime provided is 30 seconds before now, then result will be "30 seconds ago"
+ ///
+ /// DateTime in UTC to compare against DateTime.UtcNow
+ /// Encoded string.
+ public static string GetRelativeTime(this DateTime time)
+ {
+ var delta = new TimeSpan(DateTime.UtcNow.Ticks - time.Ticks);
+
+ if (Math.Abs(delta.TotalDays) > 1.0)
+ {
+ return (int)Math.Abs(delta.TotalDays) + " days ago";
+ }
+ else if (Math.Abs(delta.TotalHours) > 1.0)
+ {
+ return (int)Math.Abs(delta.TotalHours) + " hours ago";
+ }
+ else if (Math.Abs(delta.TotalMinutes) > 1.0)
+ {
+ return (int)Math.Abs(delta.TotalMinutes) + " minutes ago";
+ }
+ else
+ {
+ return (int)Math.Abs(delta.TotalSeconds) + " seconds ago";
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DateTimeExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DateTimeExtensions.cs.meta
new file mode 100644
index 0000000..5e919ae
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DateTimeExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ec124977f18b65143a87328ae508dc4d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DoubleExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DoubleExtensions.cs
new file mode 100644
index 0000000..613a47a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DoubleExtensions.cs
@@ -0,0 +1,26 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for the .Net Double struct
+ ///
+ public static class DoubleExtensions
+ {
+ ///
+ /// Checks if two numbers are approximately equal. Similar to Mathf.Approximately(float, float), but the tolerance
+ /// can be specified.
+ ///
+ /// One of the numbers to compare.
+ /// The other number to compare.
+ /// The amount of tolerance to allow while still considering the numbers approximately equal.
+ /// True if the difference between the numbers is less than or equal to the tolerance, false otherwise.
+ public static bool Approximately(this double number, double other, double tolerance)
+ {
+ return Math.Abs(number - other) <= tolerance;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DoubleExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DoubleExtensions.cs.meta
new file mode 100644
index 0000000..63eb275
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/DoubleExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ece6b9e6800a4a529a30b1fe7a13e742
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions.meta
new file mode 100644
index 0000000..ce7d65d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 6c5dce2e3e594811a9c5bcdc609848b3
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/AssemblyInfo.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/AssemblyInfo.cs
new file mode 100644
index 0000000..02ec68b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/AssemblyInfo.cs
@@ -0,0 +1,7 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+[assembly: System.Reflection.AssemblyVersion("2.8.3.0")]
+[assembly: System.Reflection.AssemblyFileVersion("2.8.3.0")]
+
+[assembly: System.Reflection.AssemblyProduct("Microsoft® Mixed Reality Toolkit")]
+[assembly: System.Reflection.AssemblyCopyright("Copyright © Microsoft Corporation")]
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/AssemblyInfo.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/AssemblyInfo.cs.meta
new file mode 100644
index 0000000..b56ddb9
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/AssemblyInfo.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d0b30485bf32be9408389c9d947b1d9a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/EditorLayerExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/EditorLayerExtensions.cs
new file mode 100644
index 0000000..83850ee
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/EditorLayerExtensions.cs
@@ -0,0 +1,88 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ public static class EditorLayerExtensions
+ {
+ private static SerializedProperty tagManagerLayers = null;
+
+ ///
+ /// The current layers defined in the Tag Manager.
+ ///
+ public static UnityEditor.SerializedProperty TagManagerLayers
+ {
+ get
+ {
+ if (tagManagerLayers == null)
+ {
+ InitializeTagManager();
+ }
+
+ return tagManagerLayers;
+ }
+ }
+
+ private static void InitializeTagManager()
+ {
+ Object[] tagAssets = UnityEditor.AssetDatabase.LoadAllAssetsAtPath("ProjectSettings/TagManager.asset");
+
+ if ((tagAssets == null) || (tagAssets.Length == 0))
+ {
+ Debug.LogError("Failed to load TagManager!");
+ return;
+ }
+
+ var tagsManager = new UnityEditor.SerializedObject(tagAssets);
+ tagManagerLayers = tagsManager.FindProperty("layers");
+
+ Debug.Assert(tagManagerLayers != null);
+ }
+
+ ///
+ /// Attempts to set the layer in Project Settings Tag Manager.
+ ///
+ /// The layer Id to attempt to set the layer on.
+ /// The layer name to attempt to set the layer on.
+ ///
+ /// True if the specified layerId was newly configured, false otherwise.
+ ///
+ public static bool SetupLayer(int layerId, string layerName)
+ {
+ SerializedProperty layer = TagManagerLayers.GetArrayElementAtIndex(layerId);
+
+ if (!string.IsNullOrEmpty(layer.stringValue))
+ {
+ // layer already set.
+ return false;
+ }
+
+ layer.stringValue = layerName;
+ layer.serializedObject.ApplyModifiedProperties();
+ AssetDatabase.SaveAssets();
+ return true;
+ }
+
+ ///
+ /// Attempts to remove the layer from the Project Settings Tag Manager.
+ ///
+ public static void RemoveLayer(string layerName)
+ {
+ for (int i = 0; i < TagManagerLayers.arraySize; i++)
+ {
+ var layer = TagManagerLayers.GetArrayElementAtIndex(i);
+
+ if (layer.stringValue == layerName)
+ {
+ layer.stringValue = string.Empty;
+ layer.serializedObject.ApplyModifiedProperties();
+ AssetDatabase.SaveAssets();
+ break;
+ }
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/EditorLayerExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/EditorLayerExtensions.cs.meta
new file mode 100644
index 0000000..76fd2ac
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/EditorLayerExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7bf7a3d18bba6c241b1d2fc4cff695fd
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/MRTK.Editor.ClassExtensions.asmdef b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/MRTK.Editor.ClassExtensions.asmdef
new file mode 100644
index 0000000..5558ebb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/MRTK.Editor.ClassExtensions.asmdef
@@ -0,0 +1,14 @@
+{
+ "name": "Microsoft.MixedReality.Toolkit.Editor.ClassExtensions",
+ "references": [],
+ "optionalUnityReferences": [],
+ "includePlatforms": [
+ "Editor"
+ ],
+ "excludePlatforms": [],
+ "allowUnsafeCode": false,
+ "overrideReferences": false,
+ "precompiledReferences": [],
+ "autoReferenced": true,
+ "defineConstraints": []
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/MRTK.Editor.ClassExtensions.asmdef.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/MRTK.Editor.ClassExtensions.asmdef.meta
new file mode 100644
index 0000000..facbf3d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/MRTK.Editor.ClassExtensions.asmdef.meta
@@ -0,0 +1,7 @@
+fileFormatVersion: 2
+guid: 7b81b416265c480884e087550884a3e8
+AssemblyDefinitionImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/ScriptableObjectExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/ScriptableObjectExtensions.cs
new file mode 100644
index 0000000..a2e935c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/ScriptableObjectExtensions.cs
@@ -0,0 +1,91 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.IO;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// Extensions for ScriptableObjects
+ ///
+ public static class ScriptableObjectExtensions
+ {
+ ///
+ /// Creates, saves, and then opens a new asset for the target ScriptableObject.
+ ///
+ /// ScriptableObject you want to create an asset file for.
+ /// Optional path for the new asset.
+ /// Optional filename for the new asset.
+ public static ScriptableObject CreateAsset(this ScriptableObject scriptableObject, string path = null, string fileName = null)
+ {
+ var name = string.IsNullOrEmpty(fileName) ? $"{scriptableObject.GetType().Name}" : fileName;
+
+ if (string.IsNullOrEmpty(path))
+ {
+ path = "Assets";
+ }
+
+ if (Path.GetExtension(path) != string.Empty)
+ {
+ var subtractedPath = path.Substring(path.LastIndexOf("/", StringComparison.Ordinal));
+ path = path.Replace(subtractedPath, string.Empty);
+ }
+
+ if (!Directory.Exists(Path.GetFullPath(path)))
+ {
+ Directory.CreateDirectory(Path.GetFullPath(path));
+ }
+
+ string assetPathAndName = AssetDatabase.GenerateUniqueAssetPath($"{path}/{name}.asset");
+
+ AssetDatabase.CreateAsset(scriptableObject, assetPathAndName);
+ AssetDatabase.SaveAssets();
+ AssetDatabase.Refresh();
+ EditorGUIUtility.PingObject(scriptableObject);
+ return scriptableObject;
+ }
+
+ ///
+ /// Gets all the scriptable object instances in the project.
+ ///
+ /// The Type of ScriptableObject you're wanting to find instances of.
+ /// An Array of instances for the type.
+ public static T[] GetAllInstances() where T : ScriptableObject
+ {
+ // FindAssets uses tags check documentation for more info
+ string[] guids = AssetDatabase.FindAssets($"t:{typeof(T).Name}");
+ var instances = new T[guids.Length];
+
+ for (int i = 0; i < guids.Length; i++)
+ {
+ string path = AssetDatabase.GUIDToAssetPath(guids[i]);
+ instances[i] = AssetDatabase.LoadAssetAtPath(path);
+ }
+
+ return instances;
+ }
+
+ ///
+ /// Gets all the scriptable object instances in the project.
+ ///
+ /// The Type of ScriptableObject you're wanting to find instances of.
+ /// An Array of instances for the type.
+ public static ScriptableObject[] GetAllInstances(Type assetType)
+ {
+ // FindAssets uses tags check documentation for more info
+ string[] guids = AssetDatabase.FindAssets($"t:{assetType.Name}");
+ var instances = new ScriptableObject[guids.Length];
+
+ for (int i = 0; i < guids.Length; i++)
+ {
+ string path = AssetDatabase.GUIDToAssetPath(guids[i]);
+ instances[i] = AssetDatabase.LoadAssetAtPath(path);
+ }
+
+ return instances;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/ScriptableObjectExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/ScriptableObjectExtensions.cs.meta
new file mode 100644
index 0000000..bd20bc6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EditorClassExtensions/ScriptableObjectExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 58fe43cbf27d453699a23c20159f459a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EnumerableExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EnumerableExtensions.cs
new file mode 100644
index 0000000..16c094b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EnumerableExtensions.cs
@@ -0,0 +1,42 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections.Generic;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for the .Net IEnumerable class
+ ///
+ public static class EnumerableExtensions
+ {
+ ///
+ /// Returns the max element based on the provided comparer or the default value when the list is empty
+ ///
+ /// Max or default value of T
+ public static T MaxOrDefault(this IEnumerable items, IComparer comparer = null)
+ {
+ if (items == null) { throw new ArgumentNullException("items"); }
+ comparer = comparer ?? Comparer.Default;
+
+ using (var enumerator = items.GetEnumerator())
+ {
+ if (!enumerator.MoveNext())
+ {
+ return default(T);
+ }
+
+ var max = enumerator.Current;
+ while (enumerator.MoveNext())
+ {
+ if (comparer.Compare(max, enumerator.Current) < 0)
+ {
+ max = enumerator.Current;
+ }
+ }
+ return max;
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EnumerableExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EnumerableExtensions.cs.meta
new file mode 100644
index 0000000..a62ded4
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EnumerableExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: fd03ae5f431b447ba64dbfba6b4b61f5
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EventSystemExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EventSystemExtensions.cs
new file mode 100644
index 0000000..52e3146
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EventSystemExtensions.cs
@@ -0,0 +1,100 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Physics;
+using System.Collections.Generic;
+using Unity.Profiling;
+using UnityEngine;
+using UnityEngine.EventSystems;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's EventSystem
+ ///
+ public static class EventSystemExtensions
+ {
+ private static readonly List RaycastResults = new List();
+ private static readonly RaycastResultComparer RaycastResultComparer = new RaycastResultComparer();
+
+ private static readonly ProfilerMarker RaycastPerfMarker = new ProfilerMarker("[MRTK] EventSystemExtensions.Raycast");
+
+ ///
+ /// Executes a raycast all and returns the closest element.
+ /// Fixes the current issue with Unity's raycast sorting which does not consider separate canvases.
+ ///
+ ///
+ /// Takes an optional RaycastResultComparer, which will be used to select the highest priority raycast result.
+ ///
+ /// RaycastResult if hit, or an empty RaycastResult if nothing was hit
+ public static RaycastResult Raycast(this EventSystem eventSystem, PointerEventData pointerEventData, LayerMask[] layerMasks, RaycastResultComparer raycastResultComparer = null)
+ {
+ using (RaycastPerfMarker.Auto())
+ {
+ eventSystem.RaycastAll(pointerEventData, RaycastResults);
+ return PrioritizeRaycastResult(layerMasks, raycastResultComparer);
+ }
+ }
+
+ private static readonly ProfilerMarker PrioritizeRaycastResultPerfMarker = new ProfilerMarker("[MRTK] EventSystemExtensions.PrioritizeRaycastResult");
+
+ ///
+ /// Sorts the available Raycasts in to a priority order for query.
+ ///
+ /// The layer mask priority.
+ ///
+ private static RaycastResult PrioritizeRaycastResult(LayerMask[] priority, RaycastResultComparer raycastResultComparer)
+ {
+ using (PrioritizeRaycastResultPerfMarker.Auto())
+ {
+ // If not specified, default to the in-box RaycastResultComparer.
+ if (raycastResultComparer == null)
+ {
+ raycastResultComparer = RaycastResultComparer;
+ }
+
+ ComparableRaycastResult maxResult = default(ComparableRaycastResult);
+
+ for (var i = 0; i < RaycastResults.Count; i++)
+ {
+ if (RaycastResults[i].gameObject == null) { continue; }
+
+ var layerMaskIndex = RaycastResults[i].gameObject.layer.FindLayerListIndex(priority);
+ if (layerMaskIndex == -1) { continue; }
+
+ var result = new ComparableRaycastResult(RaycastResults[i], layerMaskIndex);
+
+ if (maxResult.RaycastResult.module == null || raycastResultComparer.Compare(maxResult, result) < 0)
+ {
+ maxResult = result;
+ }
+ }
+
+ return maxResult.RaycastResult;
+ }
+ }
+
+ ///
+ /// Bubbles up an event to the parents of the root game object if the event data is not already used.
+ ///
+ /// The EventFunction type.
+ /// Events start executing on the parent of this game object.
+ /// Data associated with the Executing event.
+ /// Function to execute on the gameObject components.
+ /// GameObject that handled the event
+ public static GameObject ExecuteHierarchyUpward(GameObject root, BaseEventData eventData, ExecuteEvents.EventFunction callbackFunction) where T : IEventSystemHandler
+ {
+ if (!eventData.used && root != null)
+ {
+ var parent = root.transform.parent;
+
+ if (parent != null)
+ {
+ return ExecuteEvents.ExecuteHierarchy(parent.gameObject, eventData, callbackFunction);
+ }
+ }
+
+ return null;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EventSystemExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EventSystemExtensions.cs.meta
new file mode 100644
index 0000000..264e2c8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/EventSystemExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 246eece0943a48b2949d3fa2006e3b26
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/FloatExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/FloatExtensions.cs
new file mode 100644
index 0000000..d3a54f5
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/FloatExtensions.cs
@@ -0,0 +1,26 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for the .Net Float struct
+ ///
+ public static class FloatExtensions
+ {
+ ///
+ /// Checks if two numbers are approximately equal. Similar to Mathf.Approximately(float, float), but the tolerance
+ /// can be specified.
+ ///
+ /// One of the numbers to compare.
+ /// The other number to compare.
+ /// The amount of tolerance to allow while still considering the numbers approximately equal.
+ /// True if the difference between the numbers is less than or equal to the tolerance, false otherwise.
+ public static bool Approximately(this float number, float other, float tolerance)
+ {
+ return Mathf.Abs(number - other) <= tolerance;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/FloatExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/FloatExtensions.cs.meta
new file mode 100644
index 0000000..38dedc8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/FloatExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9735ed9f51f645b5be095dc6bdb70312
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/GameObjectExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/GameObjectExtensions.cs
new file mode 100644
index 0000000..0c8bbca
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/GameObjectExtensions.cs
@@ -0,0 +1,208 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Collections.Generic;
+using System.Threading.Tasks;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's GameObject class
+ ///
+ public static class GameObjectExtensions
+ {
+ ///
+ /// Export mesh data of current GameObject, and children if enabled, to file provided in OBJ format
+ ///
+ public static async Task ExportOBJAsync(this GameObject root, string filePath, bool includeChildren = true)
+ {
+ await OBJWriterUtility.ExportOBJAsync(root, filePath, includeChildren);
+ }
+
+ ///
+ /// Set all GameObject children active or inactive based on argument
+ ///
+ /// GameObject parent to traverse from
+ /// Indicates whether children GameObjects should be active or not
+ ///
+ /// Does not call SetActive on the top level GameObject, only its children
+ ///
+ public static void SetChildrenActive(this GameObject root, bool isActive)
+ {
+ for (int i = 0; i < root.transform.childCount; i++)
+ {
+ root.transform.GetChild(i).gameObject.SetActive(isActive);
+ }
+ }
+
+ ///
+ /// Set the layer to the given object and the full hierarchy below it.
+ ///
+ /// Start point of the traverse
+ /// The layer to apply
+ public static void SetLayerRecursively(this GameObject root, int layer)
+ {
+ if (root == null)
+ {
+ throw new ArgumentNullException(nameof(root), "Root transform can't be null.");
+ }
+
+ foreach (var child in root.transform.EnumerateHierarchy())
+ {
+ child.gameObject.layer = layer;
+ }
+ }
+
+ ///
+ /// Set the layer to the given object and the full hierarchy below it and cache the previous layers in the out parameter.
+ ///
+ /// Start point of the traverse
+ /// The layer to apply
+ /// The previously set layer for each object
+ public static void SetLayerRecursively(this GameObject root, int layer, out Dictionary cache)
+ {
+ if (root == null) { throw new ArgumentNullException(nameof(root)); }
+
+ cache = new Dictionary();
+
+ foreach (var child in root.transform.EnumerateHierarchy())
+ {
+ cache[child.gameObject] = child.gameObject.layer;
+ child.gameObject.layer = layer;
+ }
+ }
+
+ ///
+ /// Reapplies previously cached hierarchy layers
+ ///
+ /// Start point of the traverse
+ /// The previously set layer for each object
+ public static void ApplyLayerCacheRecursively(this GameObject root, Dictionary cache)
+ {
+ if (root == null) { throw new ArgumentNullException(nameof(root)); }
+ if (cache == null) { throw new ArgumentNullException(nameof(cache)); }
+
+ foreach (var child in root.transform.EnumerateHierarchy())
+ {
+ int layer;
+ if (!cache.TryGetValue(child.gameObject, out layer)) { continue; }
+ child.gameObject.layer = layer;
+ cache.Remove(child.gameObject);
+ }
+ }
+
+ ///
+ /// Determines whether or not a game object's layer is included in the specified layer mask.
+ ///
+ /// The game object whose layer to test.
+ /// The layer mask to test against.
+ /// True if 's layer is included in , false otherwise.
+ public static bool IsInLayerMask(this GameObject gameObject, LayerMask layerMask)
+ {
+ LayerMask gameObjectMask = 1 << gameObject.layer;
+ return (gameObjectMask & layerMask) == gameObjectMask;
+ }
+
+ ///
+ /// Apply the specified delegate to all objects in the hierarchy under a specified game object.
+ ///
+ /// Root game object of the hierarchy.
+ /// Delegate to apply.
+ public static void ApplyToHierarchy(this GameObject root, Action action)
+ {
+ action(root);
+ Transform[] items = root.GetComponentsInChildren();
+
+ for (var i = 0; i < items.Length; i++)
+ {
+ action(items[i].gameObject);
+ }
+ }
+
+ ///
+ /// Find the first component of type in the ancestors of the specified game object.
+ ///
+ /// Type of component to find.
+ /// Game object for which ancestors must be considered.
+ /// Indicates whether the specified game object should be included.
+ /// The component of type . Null if it none was found.
+ public static T FindAncestorComponent(this GameObject gameObject, bool includeSelf = true) where T : Component
+ {
+ return gameObject.transform.FindAncestorComponent(includeSelf);
+ }
+
+ ///
+ /// Perform an action on every component of type T that is on this GameObject
+ ///
+ /// Component Type
+ /// this gameObject
+ /// Action to perform.
+ public static void ForEachComponent(this GameObject gameObject, Action action) where T : Component
+ {
+ foreach (T i in gameObject.GetComponents())
+ {
+ action(i);
+ }
+ }
+
+ ///
+ /// Destroys GameObject appropriately depending if in edit or playmode
+ ///
+ /// GameObject to destroy
+ /// time in seconds at which to destroy GameObject if applicable
+ public static void DestroyGameObject(GameObject gameObject, float t = 0.0f)
+ {
+ UnityObjectExtensions.DestroyObject(gameObject, t);
+ }
+
+ ///
+ /// Checks if any MonoBehaviour on the given GameObject is using the RequireComponentAttribute requiring type T
+ ///
+ /// Only functions when called within a UNITY_EDITOR context. Outside of UNITY_EDITOR, always returns false
+ /// The potentially required component
+ /// the GameObject requiring the component
+ /// A list of types that do require the component in question
+ /// true if appears in any RequireComponentAttribute, otherwise false
+ public static bool IsComponentRequired(this GameObject gameObject, out List requiringTypes) where T : Component
+ {
+ var genericType = typeof(T);
+ requiringTypes = null;
+
+#if UNITY_EDITOR
+ foreach (var monoBehaviour in gameObject.GetComponents())
+ {
+ if (monoBehaviour == null)
+ {
+ continue;
+ }
+
+ var monoBehaviourType = monoBehaviour.GetType();
+ var attributes = Attribute.GetCustomAttributes(monoBehaviourType);
+
+ foreach (var attribute in attributes)
+ {
+ if (attribute is RequireComponent requireComponentAttribute)
+ {
+ if (requireComponentAttribute.m_Type0 == genericType ||
+ requireComponentAttribute.m_Type1 == genericType ||
+ requireComponentAttribute.m_Type2 == genericType)
+ {
+ if (requiringTypes == null)
+ {
+ requiringTypes = new List();
+ }
+
+ requiringTypes.Add(monoBehaviourType);
+ }
+ }
+ }
+ }
+#endif // UNITY_EDITOR
+
+ return requiringTypes != null;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/GameObjectExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/GameObjectExtensions.cs.meta
new file mode 100644
index 0000000..17ba414
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/GameObjectExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e51dffcf6ded45a48ba75d63d5bcb520
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/HandednessExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/HandednessExtensions.cs
new file mode 100644
index 0000000..c4f6762
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/HandednessExtensions.cs
@@ -0,0 +1,69 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// type method extensions.
+ ///
+ public static class HandednessExtensions
+ {
+ ///
+ /// Gets the opposite "hand" flag for the current Handedness value.
+ ///
+ ///
+ /// If current = Left, returns Right.
+ /// If current = Right, returns Left.
+ /// Otherwise, returns None
+ ///
+ public static Handedness GetOppositeHandedness(this Handedness current)
+ {
+ if (current == Handedness.Left)
+ {
+ return Handedness.Right;
+ }
+ else if (current == Handedness.Right)
+ {
+ return Handedness.Left;
+ }
+ else
+ {
+ return Handedness.None;
+ }
+ }
+
+ ///
+ /// Returns true if the current Handedness is the Right (i.e == Handedness.Right), false otherwise
+ ///
+ public static bool IsRight(this Handedness current)
+ {
+ return current == Handedness.Right;
+ }
+
+ ///
+ /// Returns true if the current Handedness is the Right (i.e == Handedness.Right), false otherwise
+ ///
+ public static bool IsLeft(this Handedness current)
+ {
+ return current == Handedness.Left;
+ }
+
+ ///
+ /// Returns true if the current Handedness is the Right (i.e == Handedness.Right), false otherwise
+ ///
+ public static bool IsNone(this Handedness current)
+ {
+ return current == Handedness.None;
+ }
+
+ ///
+ /// Returns true if the current Handedness flags are a match with the comparison Handedness flags, false otherwise
+ ///
+ public static bool IsMatch(this Handedness current, Handedness compare)
+ {
+ return (current & compare) != 0;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/HandednessExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/HandednessExtensions.cs.meta
new file mode 100644
index 0000000..52ff510
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/HandednessExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1fb5bfd4f80d7174c81d4778748ba924
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/LayerExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/LayerExtensions.cs
new file mode 100644
index 0000000..d71fa67
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/LayerExtensions.cs
@@ -0,0 +1,84 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's LayerMask struct
+ ///
+ public static class LayerExtensions
+ {
+ ///
+ /// The Invalid Layer Id.
+ ///
+ public const int InvalidLayerId = -1;
+
+ ///
+ /// Look through the layerMaskList and find the index in that list for which the supplied layer is part of
+ ///
+ /// Layer to search for
+ /// List of LayerMasks to search
+ /// LayerMaskList index, or -1 for not found
+ public static int FindLayerListIndex(this int layer, LayerMask[] layerMasks)
+ {
+ for (int j = 0; j < layerMasks.Length; j++)
+ {
+ if (layer.IsInLayerMask(layerMasks[j]))
+ {
+ return j;
+ }
+ }
+
+ return -1;
+ }
+
+ ///
+ /// Checks whether a layer is in a layer mask
+ ///
+ /// True if the layer mask contains the layer
+ public static bool IsInLayerMask(this int layer, int layerMask)
+ {
+ return ((1 << layer) & layerMask) != 0;
+ }
+
+ ///
+ /// Combines provided layers into a single layer mask.
+ ///
+ /// The combined layer mask
+ public static int Combine(this LayerMask[] layerMaskList)
+ {
+ int combinedLayerMask = 0;
+ for (int i = 0; i < layerMaskList.Length; i++)
+ {
+ combinedLayerMask |= layerMaskList[i].value;
+ }
+ return combinedLayerMask;
+ }
+
+ ///
+ /// Transform layer id to LayerMask
+ ///
+ public static LayerMask ToMask(int layerId)
+ {
+ return 1 << layerId;
+ }
+
+ ///
+ /// Gets a valid layer id using the layer name.
+ ///
+ /// The cached layer id.
+ /// The name of the layer to look for if the cache is unset.
+ /// The layer id.
+ public static int GetLayerId(ref int cache, string layerName)
+ {
+ if (cache == InvalidLayerId)
+ {
+ cache = LayerMask.NameToLayer(layerName);
+ }
+
+ return cache;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/LayerExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/LayerExtensions.cs.meta
new file mode 100644
index 0000000..502da78
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/LayerExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: fd6c695fdece4cc58512d654d8aade4a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/MathfExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/MathfExtensions.cs
new file mode 100644
index 0000000..f6e9728
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/MathfExtensions.cs
@@ -0,0 +1,70 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods and helper functions for various math data
+ ///
+ public static class MathExtensions
+ {
+ public static int MostSignificantBit(this int x)
+ {
+ x |= (x >> 1);
+ x |= (x >> 2);
+ x |= (x >> 4);
+ x |= (x >> 8);
+ x |= (x >> 16);
+
+ return x & ~(x >> 1);
+ }
+
+ public static int PowerOfTwoGreaterThanOrEqualTo(this int v)
+ {
+ return Mathf.IsPowerOfTwo(v) ? v : Mathf.NextPowerOfTwo(v);
+ }
+
+ public static Vector3Int PowerOfTwoGreaterThanOrEqualTo(this Vector3Int v)
+ {
+ return new Vector3Int(PowerOfTwoGreaterThanOrEqualTo(v.x),
+ PowerOfTwoGreaterThanOrEqualTo(v.y),
+ PowerOfTwoGreaterThanOrEqualTo(v.z));
+ }
+
+ public static int CubicToLinearIndex(Vector3Int ndx, Vector3Int size)
+ {
+ return (ndx.x) +
+ (ndx.y * size.x) +
+ (ndx.z * size.x * size.y);
+ }
+
+ public static Vector3Int LinearToCubicIndex(int linearIndex, Vector3Int size)
+ {
+ return new Vector3Int((linearIndex / 1) % size.x,
+ (linearIndex / size.x) % size.y,
+ (linearIndex / (size.x * size.y)) % size.z);
+ }
+
+ public static Vector3 ClampComponentWise(Vector3 value, Vector3 min, Vector3 max)
+ {
+ return new Vector3(Mathf.Clamp(value.x, min.x, max.x),
+ Mathf.Clamp(value.y, min.y, max.y),
+ Mathf.Clamp(value.z, min.z, max.z));
+ }
+
+ ///
+ /// Sets the value to zero if greater than the specified amount.
+ ///
+ public static int ResetIfGreaterThan(this int value, int amount)
+ {
+ if (value > amount)
+ {
+ value = 0;
+ }
+
+ return value;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/MathfExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/MathfExtensions.cs.meta
new file mode 100644
index 0000000..de5d56a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/MathfExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: dcbd7251a0c44e1d9a1a5e57457549d4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ProcessExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ProcessExtensions.cs
new file mode 100644
index 0000000..45ea06e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ProcessExtensions.cs
@@ -0,0 +1,153 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+#if UNITY_EDITOR || !UNITY_WSA
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.Collections.Generic;
+using System.Diagnostics;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Process Extension class.
+ ///
+ public static class ProcessExtensions
+ {
+ ///
+ /// Starts a process asynchronously.
+ ///
+ /// This Process.
+ /// The process executable to run.
+ /// The Process arguments.
+ /// Should output debug code to Editor Console?
+ ///
+ public static async Task StartProcessAsync(this Process process, string fileName, string args, bool showDebug = false, CancellationToken cancellationToken = default)
+ {
+ return await StartProcessAsync(process, new ProcessStartInfo
+ {
+ FileName = fileName,
+ CreateNoWindow = true,
+ UseShellExecute = false,
+ RedirectStandardOutput = true,
+ RedirectStandardError = true,
+ Arguments = args
+ }, showDebug, cancellationToken);
+ }
+
+ ///
+ /// Starts a process asynchronously.
+ ///
+ /// The provided Process Start Info must not use shell execution, and should redirect the standard output and errors.
+ /// This Process.
+ /// The Process start info.
+ /// Should output debug code to Editor Console?
+ ///
+ public static async Task StartProcessAsync(this Process process, ProcessStartInfo startInfo, bool showDebug = false, CancellationToken cancellationToken = default)
+ {
+ Debug.Assert(!startInfo.UseShellExecute, "Process Start Info must not use shell execution.");
+ Debug.Assert(startInfo.RedirectStandardOutput, "Process Start Info must redirect standard output.");
+ Debug.Assert(startInfo.RedirectStandardError, "Process Start Info must redirect standard errors.");
+
+ process.StartInfo = startInfo;
+ process.EnableRaisingEvents = true;
+
+ var processResult = new TaskCompletionSource();
+ var errorCodeResult = new TaskCompletionSource();
+ var errorList = new List();
+ var outputCodeResult = new TaskCompletionSource();
+ var outputList = new List();
+
+ process.Exited += OnProcessExited;
+ process.ErrorDataReceived += OnErrorDataReceived;
+ process.OutputDataReceived += OnOutputDataReceived;
+
+ async void OnProcessExited(object sender, EventArgs args)
+ {
+ processResult.TrySetResult(new ProcessResult(process.ExitCode, await errorCodeResult.Task, await outputCodeResult.Task));
+ process.Close();
+ process.Dispose();
+ }
+
+ void OnErrorDataReceived(object sender, DataReceivedEventArgs args)
+ {
+ if (args.Data != null)
+ {
+ errorList.Add(args.Data);
+
+ if (!showDebug)
+ {
+ return;
+ }
+
+ UnityEngine.Debug.LogError(args.Data);
+ }
+ else
+ {
+ errorCodeResult.TrySetResult(errorList.ToArray());
+ }
+ }
+
+ void OnOutputDataReceived(object sender, DataReceivedEventArgs args)
+ {
+ if (args.Data != null)
+ {
+ outputList.Add(args.Data);
+
+ if (!showDebug)
+ {
+ return;
+ }
+
+ UnityEngine.Debug.Log(args.Data);
+ }
+ else
+ {
+ outputCodeResult.TrySetResult(outputList.ToArray());
+ }
+ }
+
+ if (!process.Start())
+ {
+ if (showDebug)
+ {
+ UnityEngine.Debug.LogError("Failed to start process!");
+ }
+
+ processResult.TrySetResult(new ProcessResult(process.ExitCode, new[] { "Failed to start process!" }, null));
+ }
+ else
+ {
+ process.BeginOutputReadLine();
+ process.BeginErrorReadLine();
+ CancellationWatcher(process);
+ }
+
+ async void CancellationWatcher(Process _process)
+ {
+ await Task.Run(() =>
+ {
+ try
+ {
+ while (!_process.HasExited)
+ {
+ if (cancellationToken.IsCancellationRequested)
+ {
+ _process.Kill();
+ }
+ }
+ }
+ catch
+ {
+ // ignored
+ }
+ });
+ }
+
+ return await processResult.Task;
+ }
+ }
+}
+#endif
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ProcessExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ProcessExtensions.cs.meta
new file mode 100644
index 0000000..a298fcd
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ProcessExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 74b749c932e241b3bde767959c28da46
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/QuaternionExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/QuaternionExtensions.cs
new file mode 100644
index 0000000..0bd35e1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/QuaternionExtensions.cs
@@ -0,0 +1,32 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Quaternion struct.
+ ///
+ public static class QuaternionExtensions
+ {
+ public static bool IsValidRotation(this Quaternion rotation)
+ {
+ return !float.IsNaN(rotation.x) && !float.IsNaN(rotation.y) && !float.IsNaN(rotation.z) && !float.IsNaN(rotation.w) &&
+ !float.IsInfinity(rotation.x) && !float.IsInfinity(rotation.y) && !float.IsInfinity(rotation.z) && !float.IsInfinity(rotation.w);
+ }
+
+ ///
+ /// Determines if the angle between two quaternions is within a given tolerance.
+ ///
+ /// The first quaternion.
+ /// The second quaternion.
+ /// The maximum angle that will cause this to return true.
+ /// True if the quaternions are aligned within the tolerance, false otherwise.
+ public static bool AlignedEnough(Quaternion q1, Quaternion q2, float angleTolerance)
+ {
+ return Mathf.Abs(Quaternion.Angle(q1, q2)) < angleTolerance;
+ }
+
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/QuaternionExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/QuaternionExtensions.cs.meta
new file mode 100644
index 0000000..6b1fdf6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/QuaternionExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 7601214696b04442b86ae79754fdb182
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/RayExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/RayExtensions.cs
new file mode 100644
index 0000000..7482e3c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/RayExtensions.cs
@@ -0,0 +1,23 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Ray struct
+ ///
+ public static class RayExtensions
+ {
+ ///
+ /// Determines whether or not a ray is valid.
+ ///
+ /// The ray being tested.
+ /// True if the ray is valid, false otherwise.
+ public static bool IsValid(this Ray ray)
+ {
+ return ray.direction != Vector3.zero;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/RayExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/RayExtensions.cs.meta
new file mode 100644
index 0000000..5e1b499
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/RayExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 06ac0b53630a473581d7e70505975010
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ReflectionExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ReflectionExtensions.cs
new file mode 100644
index 0000000..ba32539
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ReflectionExtensions.cs
@@ -0,0 +1,199 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+#if WINDOWS_UWP && !ENABLE_IL2CPP
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Reflection;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for .Net reflection functions
+ ///
+ public static class ReflectionExtensions
+ {
+ public static EventInfo GetEvent(this Type type, string eventName)
+ {
+ return type.GetRuntimeEvent(eventName);
+ }
+
+ public static MethodInfo GetMethod(this Type type, string methodName)
+ {
+ return GetMethod(type, methodName, (BindingFlags)0x0);
+ }
+
+ public static MethodInfo GetMethod(this Type type, string methodName, BindingFlags flags)
+ {
+ var result = type.GetTypeInfo().GetDeclaredMethod(methodName);
+ if (((flags & BindingFlags.FlattenHierarchy) != 0) && result == null)
+ {
+ var baseType = type.GetTypeInfo().BaseType;
+ if (baseType != null)
+ {
+ return GetMethod(baseType, methodName, flags);
+ }
+ }
+
+ return result;
+ }
+
+ public static MethodInfo GetMethod(this Type type, string methodName, BindingFlags bindingAttr, Object binder, Type[] parameters, Object[] modifiers)
+ {
+ var result = type.GetTypeInfo().GetDeclaredMethod(methodName);
+ if (result == null)
+ {
+ var baseType = type.GetTypeInfo().BaseType;
+ if (baseType != null)
+ {
+ return GetMethod(baseType, methodName, bindingAttr, binder, parameters, modifiers);
+ }
+ }
+
+ return result;
+ }
+
+ public static MethodInfo GetMethod(this Type type, string methodName, Type[] parameters)
+ {
+ return GetMethods(type).Where(m => m.Name == methodName).FirstOrDefault(
+ m =>
+ {
+ var types = m.GetParameters().Select(p => p.ParameterType).ToArray();
+ if (types.Length == parameters.Length)
+ {
+ for (int idx = 0; idx < types.Length; idx++)
+ {
+ if (types[idx] != parameters[idx])
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+ else
+ {
+ return false;
+ }
+ }
+ );
+ }
+
+ public static IEnumerable GetMethods(this Type type)
+ {
+ return GetMethods(type, (BindingFlags)0x0);
+ }
+
+ public static IEnumerable GetMethods(this Type type, BindingFlags flags)
+ {
+ return type.GetTypeInfo().GetMethods(flags);
+ }
+
+ public static IEnumerable GetMethods(this TypeInfo type)
+ {
+ return GetMethods(type, (BindingFlags)0x0);
+ }
+
+ public static IEnumerable GetMethods(this TypeInfo type, BindingFlags flags)
+ {
+ return type.DeclaredMethods;
+ }
+
+ public static IEnumerable GetFields(this Type type)
+ {
+ return GetFields(type, (BindingFlags)0x0);
+ }
+
+ public static IEnumerable GetFields(this Type type, BindingFlags flags)
+ {
+ return type.GetTypeInfo().DeclaredFields;
+ }
+
+ public static FieldInfo GetField(this Type type, string fieldName)
+ {
+ return type.GetRuntimeField(fieldName);
+ }
+
+ public static IEnumerable GetProperties(this Type type, BindingFlags flags)
+ {
+ return type.GetTypeInfo().DeclaredProperties;
+ }
+
+ public static PropertyInfo GetProperty(this Type type, string propertyName)
+ {
+ return GetProperty(type, propertyName, (BindingFlags)0x0);
+ }
+
+ public static PropertyInfo GetProperty(this Type type, string propertyName, BindingFlags flags)
+ {
+ return type.GetRuntimeProperty(propertyName);
+ }
+
+ public static PropertyInfo GetProperty(this Type type, string propertyName, Type returnType)
+ {
+ return type.GetRuntimeProperty(propertyName);
+ }
+
+ public static IEnumerable GetTypeInfos(this Assembly assembly)
+ {
+ return assembly.DefinedTypes;
+ }
+
+ public static Type[] GetInterfaces(this Type type)
+ {
+ return type.GetTypeInfo().ImplementedInterfaces.ToArray();
+ }
+
+ public static bool IsClass(this Type type)
+ {
+ return type.GetTypeInfo().IsClass;
+ }
+
+ public static bool IsInterface(this Type type)
+ {
+ return type.GetTypeInfo().IsInterface;
+ }
+
+ public static bool IsAbstract(this Type type)
+ {
+ return type.GetTypeInfo().IsAbstract;
+ }
+
+ public static bool IsSubclassOf(this Type type, Type c)
+ {
+ return type.GetTypeInfo().IsSubclassOf(c);
+ }
+
+ public static bool IsAssignableFrom(this Type type, Type c)
+ {
+ return type.IsAssignableFrom(c.GetTypeInfo());
+ }
+
+ public static bool IsEnum(this Type type)
+ {
+ return type.GetTypeInfo().IsEnum;
+ }
+
+ public static bool IsValueType(this Type type)
+ {
+ return type.GetTypeInfo().IsValueType;
+ }
+
+ public static bool IsAssignableFrom(this Type type, TypeInfo typeInfo)
+ {
+ return type.GetTypeInfo().IsAssignableFrom(typeInfo);
+ }
+
+ public static object[] GetCustomAttributes(this Type type, bool inherit)
+ {
+ return type.GetTypeInfo().GetCustomAttributes(inherit).ToArray();
+ }
+
+ public static object[] GetCustomAttributes(this Type type, Type attributeType, bool inherit)
+ {
+ return type.GetTypeInfo().GetCustomAttributes(attributeType, inherit).ToArray();
+ }
+ }
+}
+#endif // WINDOWS_UWP && !ENABLE_IL2CPP
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ReflectionExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ReflectionExtensions.cs.meta
new file mode 100644
index 0000000..4f32c8f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/ReflectionExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d8682816b9a14e8e9e9a777af1e25df5
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringBuilderExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringBuilderExtensions.cs
new file mode 100644
index 0000000..f101364
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringBuilderExtensions.cs
@@ -0,0 +1,23 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Text;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extensions.
+ ///
+ public static class StringBuilderExtensions
+ {
+ ///
+ /// Append new line for current Environment to this StringBuilder buffer
+ ///
+ public static StringBuilder AppendNewLine(this StringBuilder sb)
+ {
+ sb.Append(Environment.NewLine);
+ return sb;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringBuilderExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringBuilderExtensions.cs.meta
new file mode 100644
index 0000000..da889ea
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringBuilderExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 00d64e9c83401b94eb4def98c5bada6a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringExtensions.cs
new file mode 100644
index 0000000..c52fd18
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringExtensions.cs
@@ -0,0 +1,88 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.IO;
+using System.Text;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extensions.
+ ///
+ public static class StringExtensions
+ {
+ ///
+ /// Encodes the string to base 64 ASCII.
+ ///
+ /// String to encode.
+ /// Encoded string.
+ public static string EncodeTo64(this string toEncode)
+ {
+ byte[] toEncodeAsBytes = Encoding.ASCII.GetBytes(toEncode);
+ return Convert.ToBase64String(toEncodeAsBytes);
+ }
+
+ ///
+ /// Decodes string from base 64 ASCII.
+ ///
+ /// String to decode.
+ /// Decoded string.
+ public static string DecodeFrom64(this string encodedData)
+ {
+ byte[] encodedDataAsBytes = Convert.FromBase64String(encodedData);
+ return Encoding.ASCII.GetString(encodedDataAsBytes);
+ }
+
+ ///
+ /// Capitalize the first character and add a space before
+ /// each capitalized letter (except the first character).
+ ///
+ public static string ToProperCase(this string value)
+ {
+ // If there are 0 or 1 characters, just return the string.
+ if (value == null) { return value; }
+ if (value.Length < 2) { return value.ToUpper(); }
+ // If there's already spaces in the string, return.
+ if (value.Contains(" ")) { return value; }
+
+ // Start with the first character.
+ string result = value.Substring(0, 1).ToUpper();
+
+ // Add the remaining characters.
+ for (int i = 1; i < value.Length; i++)
+ {
+ if (char.IsLetter(value[i]) &&
+ char.IsUpper(value[i]))
+ {
+ // Add a space if the previous character is not upper-case.
+ // e.g. "LeftHand" -> "Left Hand"
+ if (i != 1 && // First character is upper-case in result.
+ (!char.IsLetter(value[i - 1]) || char.IsLower(value[i - 1])))
+ {
+ result += " ";
+ }
+ // If previous character is upper-case, only add space if the next
+ // character is lower-case. Otherwise assume this character to be inside
+ // an acronym.
+ // e.g. "OpenVRLeftHand" -> "Open VR Left Hand"
+ else if (i < value.Length - 1 &&
+ char.IsLetter(value[i + 1]) && char.IsLower(value[i + 1]))
+ {
+ result += " ";
+ }
+ }
+
+ result += value[i];
+ }
+
+ return result;
+ }
+
+ ///
+ /// Ensures directory separator chars in provided string are platform independent. Given path might use \ or / but not all platforms support both.
+ ///
+ public static string NormalizeSeparators(this string path)
+ => path?.Replace('\\', Path.DirectorySeparatorChar).Replace('/', Path.DirectorySeparatorChar);
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringExtensions.cs.meta
new file mode 100644
index 0000000..f75231f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/StringExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 42f4271ffab3451aa81517d121c43957
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/SystemNumericsExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/SystemNumericsExtensions.cs
new file mode 100644
index 0000000..e3a1325
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/SystemNumericsExtensions.cs
@@ -0,0 +1,61 @@
+// Copyright(c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ public static class SystemNumericsExtensions
+ {
+ ///
+ /// Converts this System.Numerics Vector3 to a UnityEngine Vector3.
+ ///
+ /// The vector to convert.
+ /// The converted vector.
+ public static UnityEngine.Vector3 ToUnityVector3(this System.Numerics.Vector3 vector)
+ {
+ return new UnityEngine.Vector3(vector.X, vector.Y, -vector.Z);
+ }
+
+ ///
+ /// Converts this System.Numerics Vector3 to a UnityEngine Vector3 format, storing values directly in referenced parameter
+ ///
+ public static void ConvertToUnityVector3(this System.Numerics.Vector3 source, ref UnityEngine.Vector3 target)
+ {
+ target.x = source.X;
+ target.y = source.Y;
+ target.z = -source.Z;
+ }
+
+ ///
+ /// Converts this System.Numerics Quaternion to a UnityEngine Quaternion.
+ ///
+ /// The quaternion to convert.
+ /// The converted quaternion.
+ public static UnityEngine.Quaternion ToUnityQuaternion(this System.Numerics.Quaternion quaternion)
+ {
+ return new UnityEngine.Quaternion(-quaternion.X, -quaternion.Y, quaternion.Z, quaternion.W);
+ }
+
+ public static UnityEngine.Matrix4x4 ToUnity(this System.Numerics.Matrix4x4 m) => new UnityEngine.Matrix4x4(
+ new UnityEngine.Vector4(m.M11, m.M12, -m.M13, m.M14),
+ new UnityEngine.Vector4(m.M21, m.M22, -m.M23, m.M24),
+ new UnityEngine.Vector4(-m.M31, -m.M32, m.M33, -m.M34),
+ new UnityEngine.Vector4(m.M41, m.M42, -m.M43, m.M44));
+
+ public static System.Numerics.Matrix4x4 ToSystemNumerics(this UnityEngine.Matrix4x4 m) => new System.Numerics.Matrix4x4(
+ m.m00, m.m10, -m.m20, m.m30,
+ m.m01, m.m11, -m.m21, m.m31,
+ -m.m02, -m.m12, m.m22, -m.m32,
+ m.m03, m.m13, -m.m23, m.m33);
+
+ ///
+ /// Converts this System.Numerics Quaternion to a UnityEngine Quaternion, storing values directly in referenced parameter
+ ///
+ public static void ConvertToUnityQuaternion(this System.Numerics.Quaternion source, ref UnityEngine.Quaternion target)
+ {
+ target.x = -source.X;
+ target.y = -source.Y;
+ target.z = source.Z;
+ target.w = source.W;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/SystemNumericsExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/SystemNumericsExtensions.cs.meta
new file mode 100644
index 0000000..e56f4ef
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/SystemNumericsExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f9f9eb9aede721d4f9092acc4e938be5
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Texture2DExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Texture2DExtensions.cs
new file mode 100644
index 0000000..5dbc398
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Texture2DExtensions.cs
@@ -0,0 +1,86 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// A collection of helper functions for Texture2D
+ ///
+ public static class Texture2DExtensions
+ {
+ ///
+ /// Fills the pixels. You will need to call Apply on the texture in order for changes to take place.
+ ///
+ /// The Texture2D.
+ /// The row to start filling at.
+ /// The column to start filling at.
+ /// The width to fill.
+ /// The height to fill.
+ /// Color of the fill.
+ /// This function considers row 0 to be left and col 0 to be top.
+ public static void FillPixels(this Texture2D texture2D, int row, int col, int width, int height, Color fillColor)
+ {
+ if (col + width > texture2D.width || row + height > texture2D.height)
+ {
+ throw new IndexOutOfRangeException();
+ }
+
+ Color32 toColor = fillColor; // Implicit cast
+ Color32[] colors = new Color32[width * height];
+ for (int i = 0; i < colors.Length; i++)
+ {
+ colors[i] = toColor;
+ }
+
+ texture2D.SetPixels32(col, (texture2D.height - row) - height, width, height, colors);
+ }
+
+ ///
+ /// Fills the texture with a single color.
+ ///
+ /// The Texture2D.
+ /// Color of the fill.
+ public static void FillPixels(this Texture2D texture2D, Color fillColor)
+ {
+ texture2D.FillPixels(0, 0, texture2D.width, texture2D.height, fillColor);
+ }
+
+ ///
+ /// Captures a region of the screen and returns it as a texture.
+ ///
+ /// x position of the screen to capture from (bottom-left)
+ /// y position of the screen to capture from (bottom-left)
+ /// width of the screen area to capture
+ /// height of the screen area to capture
+ /// A Texture2D containing pixels from the region of the screen specified
+ /// You should call this in OnPostRender.
+ public static Texture2D CaptureScreenRegion(int x, int y, int width, int height)
+ {
+ Texture2D tex = new Texture2D(width, height);
+ tex.ReadPixels(new Rect(x, y, Screen.width, Screen.height), 0, 0);
+ tex.Apply();
+ return tex;
+ }
+
+ ///
+ /// Creates a texture from a defined region.
+ ///
+ /// The Texture2D.
+ /// x position of this texture to get the texture from
+ /// y position of this texture to get the texture from
+ /// width of the region to capture
+ /// height of the region to capture
+ public static Texture2D CreateTextureFromRegion(this Texture2D texture2D, int x, int y, int width, int height)
+ {
+ Color[] pixels = texture2D.GetPixels(x, y, width, height);
+
+ Texture2D destTex = new Texture2D(width, height);
+ destTex.SetPixels(pixels);
+ destTex.Apply();
+
+ return destTex;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Texture2DExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Texture2DExtensions.cs.meta
new file mode 100644
index 0000000..c296d4e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/Texture2DExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 500ec3454fba61948860b4f6e1a5e4eb
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TransformExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TransformExtensions.cs
new file mode 100644
index 0000000..0ef1ac0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TransformExtensions.cs
@@ -0,0 +1,274 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections.Generic;
+using System.Text;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Transform class
+ ///
+ public static class TransformExtensions
+ {
+ ///
+ /// An extension method that will get you the full path to an object.
+ ///
+ /// The transform you wish a full path to.
+ /// The delimiter with which each object is delimited in the string.
+ /// Prefix with which the full path to the object should start.
+ /// A delimited string that is the full path to the game object in the hierarchy.
+ public static string GetFullPath(this Transform transform, string delimiter = ".", string prefix = "/")
+ {
+ var stringBuilder = new StringBuilder();
+ GetFullPath(stringBuilder, transform, delimiter, prefix);
+ return stringBuilder.ToString();
+ }
+
+ private static void GetFullPath(StringBuilder stringBuilder, Transform transform, string delimiter, string prefix)
+ {
+ if (transform.parent == null)
+ {
+ stringBuilder.Append(prefix);
+ }
+ else
+ {
+ GetFullPath(stringBuilder, transform.parent, delimiter, prefix);
+ stringBuilder.Append(delimiter);
+ }
+
+ stringBuilder.Append(transform.name);
+ }
+
+ ///
+ /// Enumerates all children in the hierarchy starting at the root object.
+ ///
+ /// Start point of the traversion set
+ public static IEnumerable EnumerateHierarchy(this Transform root)
+ {
+ if (root == null) { throw new ArgumentNullException("root"); }
+ return root.EnumerateHierarchyCore(new List(0));
+ }
+
+ ///
+ /// Enumerates all children in the hierarchy starting at the root object except for the branches in ignore.
+ ///
+ /// Start point of the traversion set
+ /// Transforms and all its children to be ignored
+ public static IEnumerable EnumerateHierarchy(this Transform root, ICollection ignore)
+ {
+ if (root == null) { throw new ArgumentNullException("root"); }
+ if (ignore == null)
+ {
+ throw new ArgumentNullException("ignore", "Ignore collection can't be null, use EnumerateHierarchy(root) instead.");
+ }
+ return root.EnumerateHierarchyCore(ignore);
+ }
+
+ ///
+ /// Enumerates all children in the hierarchy starting at the root object except for the branches in ignore.
+ ///
+ /// Start point of the traversion set
+ /// Transforms and all its children to be ignored
+ private static IEnumerable EnumerateHierarchyCore(this Transform root, ICollection ignore)
+ {
+ var transformQueue = new Queue();
+ transformQueue.Enqueue(root);
+
+ while (transformQueue.Count > 0)
+ {
+ var parentTransform = transformQueue.Dequeue();
+
+ if (!parentTransform || ignore.Contains(parentTransform)) { continue; }
+
+ for (var i = 0; i < parentTransform.childCount; i++)
+ {
+ transformQueue.Enqueue(parentTransform.GetChild(i));
+ }
+
+ yield return parentTransform;
+ }
+ }
+
+ ///
+ /// Calculates the bounds of all the colliders attached to this GameObject and all its children
+ ///
+ /// Transform of root GameObject the colliders are attached to
+ /// The total bounds of all colliders attached to this GameObject.
+ /// If no colliders attached, returns a bounds of center and extents 0
+ public static Bounds GetColliderBounds(this Transform transform)
+ {
+ Collider[] colliders = transform.GetComponentsInChildren();
+ if (colliders.Length == 0) { return new Bounds(); }
+
+ Bounds bounds = colliders[0].bounds;
+ for (int i = 1; i < colliders.Length; i++)
+ {
+ bounds.Encapsulate(colliders[i].bounds);
+ }
+ return bounds;
+ }
+
+ ///
+ /// Checks if the provided transforms are child/parent related.
+ ///
+ /// True if either transform is the parent of the other or if they are the same
+ public static bool IsParentOrChildOf(this Transform transform1, Transform transform2)
+ {
+ return transform1.IsChildOf(transform2) || transform2.IsChildOf(transform1);
+ }
+
+ ///
+ /// Find the first component of type in the ancestors of the specified transform.
+ ///
+ /// Type of component to find.
+ /// Transform for which ancestors must be considered.
+ /// Indicates whether the specified transform should be included.
+ /// The component of type . Null if it none was found.
+ public static T FindAncestorComponent(this Transform startTransform, bool includeSelf = true) where T : Component
+ {
+ foreach (Transform transform in startTransform.EnumerateAncestors(includeSelf))
+ {
+ T component = transform.GetComponent();
+ if (component != null)
+ {
+ return component;
+ }
+ }
+
+ return null;
+ }
+
+ ///
+ /// Enumerates the ancestors of the specified transform.
+ ///
+ /// Transform for which ancestors must be returned.
+ /// Indicates whether the specified transform should be included.
+ /// An enumeration of all ancestor transforms of the specified start transform.
+ public static IEnumerable EnumerateAncestors(this Transform startTransform, bool includeSelf)
+ {
+ if (!includeSelf)
+ {
+ startTransform = startTransform.parent;
+ }
+
+ for (Transform transform = startTransform; transform != null; transform = transform.parent)
+ {
+ yield return transform;
+ }
+ }
+
+ ///
+ /// Transforms the size from local to world.
+ ///
+ /// The transform.
+ /// The local size.
+ /// World size.
+ public static Vector3 TransformSize(this Transform transform, Vector3 localSize)
+ {
+ Vector3 transformedSize = new Vector3(localSize.x, localSize.y, localSize.z);
+
+ Transform t = transform;
+ do
+ {
+ transformedSize.x *= t.localScale.x;
+ transformedSize.y *= t.localScale.y;
+ transformedSize.z *= t.localScale.z;
+ t = t.parent;
+ }
+ while (t != null);
+
+ return transformedSize;
+ }
+
+ ///
+ /// Transforms the size from world to local.
+ ///
+ /// The transform.
+ /// The world size
+ /// World size.
+ public static Vector3 InverseTransformSize(this Transform transform, Vector3 worldSize)
+ {
+ Vector3 transformedSize = new Vector3(worldSize.x, worldSize.y, worldSize.z);
+
+ Transform t = transform;
+ do
+ {
+ transformedSize.x /= t.localScale.x;
+ transformedSize.y /= t.localScale.y;
+ transformedSize.z /= t.localScale.z;
+ t = t.parent;
+ }
+ while (t != null);
+
+ return transformedSize;
+ }
+
+ ///
+ /// Gets the hierarchical depth of the Transform from its root. Returns -1 if the transform is the root.
+ ///
+ /// The transform to get the depth for.
+ public static int GetDepth(this Transform t)
+ {
+ int depth = -1;
+
+ Transform root = t.transform.root;
+ if (root == t.transform)
+ {
+ return depth;
+ }
+
+ TryGetDepth(t, root, ref depth);
+
+ return depth;
+ }
+
+ ///
+ /// Tries to get the hierarchical depth of the Transform from the specified parent. This method is recursive.
+ ///
+ /// The transform to get the depth for
+ /// The starting transform to look for the target transform in
+ /// The depth of the target transform
+ /// 'true' if the depth could be retrieved, or 'false' because the transform is a root transform.
+ public static bool TryGetDepth(Transform target, Transform parent, ref int depth)
+ {
+ foreach (Transform child in parent)
+ {
+ depth++;
+ if (child == target.transform || TryGetDepth(target, child, ref depth))
+ {
+ return true;
+ }
+ }
+
+ return false;
+ }
+
+ ///
+ /// Walk hierarchy looking for named transform
+ ///
+ /// root transform to start searching from
+ /// name to look for
+ /// returns found transform or null if none found
+ public static Transform GetChildRecursive(Transform t, string name)
+ {
+ int numChildren = t.childCount;
+ for (int ii = 0; ii < numChildren; ++ii)
+ {
+ Transform child = t.GetChild(ii);
+ if (child.name == name)
+ {
+ return child;
+ }
+ Transform foundIt = GetChildRecursive(child, name);
+ if (foundIt != null)
+ {
+ return foundIt;
+ }
+ }
+ return null;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TransformExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TransformExtensions.cs.meta
new file mode 100644
index 0000000..6b7d634
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TransformExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2024b7a0610742e3b3f95864622c59e4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TypeExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TypeExtensions.cs
new file mode 100644
index 0000000..ca2f0f3
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TypeExtensions.cs
@@ -0,0 +1,43 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections.Generic;
+using System.Reflection;
+using System.Threading.Tasks;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ public static class TypeExtensions
+ {
+#if !NETFX_CORE
+ ///
+ /// Returns a list of types for all classes that extend from the current type and are not abstract
+ ///
+ /// The class type from which to search for inherited classes
+ /// List of assemblies to search through for types. If null, default is to grab all assemblies in current app domain
+ /// Null if rootType is not a class, otherwise returns list of types for sub-classes of rootType
+ public static List GetAllSubClassesOf(this Type rootType, Assembly[] searchAssemblies = null)
+ {
+ if (!rootType.IsClass) return null;
+
+ if (searchAssemblies == null) { searchAssemblies = AppDomain.CurrentDomain.GetAssemblies(); }
+
+ var results = new List();
+
+ Parallel.ForEach(searchAssemblies, (assembly) =>
+ {
+ Parallel.ForEach(assembly.GetLoadableTypes(), (type) =>
+ {
+ if (type != null && type.IsClass && !type.IsAbstract && type.IsSubclassOf(rootType))
+ {
+ results.Add(type);
+ }
+ });
+ });
+
+ return results;
+ }
+#endif
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TypeExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TypeExtensions.cs.meta
new file mode 100644
index 0000000..6c96271
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/TypeExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 531777f8944c3094b8f9815ff8e6aa8e
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/UnityObjectExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/UnityObjectExtensions.cs
new file mode 100644
index 0000000..5cb80d0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/UnityObjectExtensions.cs
@@ -0,0 +1,73 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEngine;
+
+using Object = UnityEngine.Object;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Object class
+ ///
+ public static class UnityObjectExtensions
+ {
+ ///
+ /// Enable Unity objects to skip "DontDestroyOnLoad" when editor isn't playing so test runner passes.
+ ///
+ public static void DontDestroyOnLoad(this Object target)
+ {
+#if UNITY_EDITOR
+ if (UnityEditor.EditorApplication.isPlaying)
+#endif
+ Object.DontDestroyOnLoad(target);
+ }
+
+ ///
+ /// Destroys a Unity object appropriately depending if running in edit or play mode.
+ ///
+ /// Unity object to destroy
+ /// Time in seconds at which to destroy the object, if applicable.
+ public static void DestroyObject(Object obj, float t = 0.0f)
+ {
+ if (Application.isPlaying)
+ {
+ Object.Destroy(obj, t);
+ }
+ else
+ {
+#if UNITY_EDITOR
+ // Must use DestroyImmediate in edit mode but it is not allowed when called from
+ // trigger/contact, animation event callbacks or OnValidate. Must use Destroy instead.
+ // Delay call to counter this issue in editor
+ UnityEditor.EditorApplication.delayCall += () =>
+ {
+ Object.DestroyImmediate(obj);
+ };
+#else
+ Object.DestroyImmediate(obj);
+#endif
+ }
+ }
+
+ ///
+ /// Tests if an interface is null, taking potential UnityEngine.Object derived class implementers into account
+ /// which require their overridden operators to be called.
+ ///
+ /// True if either the managed or native object is null, false otherwise.
+ public static bool IsNull(this T @interface) where T : class => @interface == null || @interface.Equals(null);
+
+ ///
+ /// Tests if an interface is null, taking potential UnityEngine.Object derived class implementers into account
+ /// which require their overridden operators to be called.
+ ///
+ /// Falseif either the managed or native object is null, true otherwise.
+ public static bool IsNotNull(this T @interface) where T : class => !@interface.IsNull();
+
+ ///
+ /// Properly checks an interface for null and returns the MonoBehaviour implementing it.
+ ///
+ /// True if the implementer of the interface is a MonoBehaviour and the MonoBehaviour is not null.
+ public static bool TryGetMonoBehaviour(this T @interface, out MonoBehaviour monoBehaviour) where T : class => (monoBehaviour = @interface as MonoBehaviour) != null;
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/UnityObjectExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/UnityObjectExtensions.cs.meta
new file mode 100644
index 0000000..70f4f0f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/UnityObjectExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 0524b94852744b79866e72b300fc98db
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/VectorExtensions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/VectorExtensions.cs
new file mode 100644
index 0000000..0e13d7a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/VectorExtensions.cs
@@ -0,0 +1,255 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System.Collections.Generic;
+using System.Linq;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit
+{
+ ///
+ /// Extension methods for Unity's Vector struct
+ ///
+ public static class VectorExtensions
+ {
+ public static Vector2 Mul(this Vector2 value, Vector2 scale)
+ {
+ return new Vector2(value.x * scale.x, value.y * scale.y);
+ }
+
+ public static Vector2 Div(this Vector2 value, Vector2 scale)
+ {
+ return new Vector2(value.x / scale.x, value.y / scale.y);
+ }
+
+ public static Vector3 Mul(this Vector3 value, Vector3 scale)
+ {
+ return new Vector3(value.x * scale.x, value.y * scale.y, value.z * scale.z);
+ }
+
+ public static Vector3 Div(this Vector3 value, Vector3 scale)
+ {
+ return new Vector3(value.x / scale.x, value.y / scale.y, value.z / scale.z);
+ }
+
+ public static Vector3 RotateAround(this Vector3 point, Vector3 pivot, Quaternion rotation)
+ {
+ return rotation * (point - pivot) + pivot;
+ }
+
+ public static Vector3 RotateAround(this Vector3 point, Vector3 pivot, Vector3 eulerAngles)
+ {
+ return RotateAround(point, pivot, Quaternion.Euler(eulerAngles));
+ }
+
+ public static Vector3 TransformPoint(this Vector3 point, Vector3 translation, Quaternion rotation, Vector3 lossyScale)
+ {
+ return rotation * Vector3.Scale(lossyScale, point) + translation;
+ }
+
+ public static Vector3 InverseTransformPoint(this Vector3 point, Vector3 translation, Quaternion rotation, Vector3 lossyScale)
+ {
+ var scaleInv = new Vector3(1 / lossyScale.x, 1 / lossyScale.y, 1 / lossyScale.z);
+ return Vector3.Scale(scaleInv, (Quaternion.Inverse(rotation) * (point - translation)));
+ }
+
+ public static Vector2 Average(this IEnumerable vectors)
+ {
+ float x = 0f;
+ float y = 0f;
+ int count = 0;
+
+ foreach (var pos in vectors)
+ {
+ x += pos.x;
+ y += pos.y;
+ count++;
+ }
+
+ return new Vector2(x / count, y / count);
+ }
+
+ public static Vector3 Average(this IEnumerable vectors)
+ {
+ float x = 0f;
+ float y = 0f;
+ float z = 0f;
+ int count = 0;
+
+ foreach (var pos in vectors)
+ {
+ x += pos.x;
+ y += pos.y;
+ z += pos.z;
+ count++;
+ }
+
+ return new Vector3(x / count, y / count, z / count);
+ }
+
+ public static Vector2 Average(this ICollection vectors)
+ {
+ int count = vectors.Count;
+ if (count == 0)
+ {
+ return Vector2.zero;
+ }
+
+ float x = 0f;
+ float y = 0f;
+
+ foreach (var pos in vectors)
+ {
+ x += pos.x;
+ y += pos.y;
+ }
+
+ return new Vector2(x / count, y / count);
+ }
+
+ public static Vector3 Average(this ICollection vectors)
+ {
+ int count = vectors.Count;
+
+ if (count == 0)
+ {
+ return Vector3.zero;
+ }
+
+ float x = 0f;
+ float y = 0f;
+ float z = 0f;
+
+ foreach (var pos in vectors)
+ {
+ x += pos.x;
+ y += pos.y;
+ z += pos.z;
+ }
+
+ return new Vector3(x / count, y / count, z / count);
+ }
+
+ public static Vector2 Median(this IEnumerable vectors)
+ {
+ var enumerable = vectors as Vector2[] ?? vectors.ToArray();
+ int count = enumerable.Length;
+ return count == 0 ? Vector2.zero : enumerable.OrderBy(v => v.sqrMagnitude).ElementAt(count / 2);
+ }
+
+ public static Vector3 Median(this IEnumerable vectors)
+ {
+ var enumerable = vectors as Vector3[] ?? vectors.ToArray();
+ int count = enumerable.Length;
+ return count == 0 ? Vector3.zero : enumerable.OrderBy(v => v.sqrMagnitude).ElementAt(count / 2);
+ }
+
+ public static Vector2 Median(this ICollection vectors)
+ {
+ int count = vectors.Count;
+ return count == 0 ? Vector2.zero : vectors.OrderBy(v => v.sqrMagnitude).ElementAt(count / 2);
+ }
+
+ public static Vector3 Median(this ICollection vectors)
+ {
+ int count = vectors.Count;
+ return count == 0 ? Vector3.zero : vectors.OrderBy(v => v.sqrMagnitude).ElementAt(count / 2);
+ }
+
+ public static bool IsValidVector(this Vector3 vector)
+ {
+ return !float.IsNaN(vector.x) && !float.IsNaN(vector.y) && !float.IsNaN(vector.z) &&
+ !float.IsInfinity(vector.x) && !float.IsInfinity(vector.y) && !float.IsInfinity(vector.z);
+ }
+
+ ///
+ /// Determines if the distance between two vectors is within a given tolerance.
+ ///
+ /// The first vector.
+ /// The second vector.
+ /// The maximum distance that will cause this to return true.
+ /// True if the distance between the two vectors is within the tolerance, false otherwise.
+ public static bool CloseEnough(Vector3 v1, Vector3 v2, float distanceTolerance)
+ {
+ return Mathf.Abs(Vector3.Distance(v1, v2)) < distanceTolerance;
+ }
+
+ ///
+ /// Get the relative mapping based on a source Vec3 and a radius for spherical mapping.
+ ///
+ /// The source Vector3 to be mapped to sphere
+ /// This is a for the radius of the sphere
+ public static Vector3 SphericalMapping(Vector3 source, float radius)
+ {
+ float circ = 2f * Mathf.PI * radius;
+
+ float xAngle = (source.x / circ) * 360f;
+ float yAngle = -(source.y / circ) * 360f;
+
+ source.Set(0.0f, 0.0f, radius);
+
+ Quaternion rot = Quaternion.Euler(yAngle, xAngle, 0.0f);
+ source = rot * source;
+
+ return source;
+ }
+
+ ///
+ /// Get the relative mapping based on a source Vec3 and a radius for cylinder mapping.
+ ///
+ /// The source Vector3 to be mapped to cylinder
+ /// This is a for the radius of the cylinder
+ public static Vector3 CylindricalMapping(Vector3 source, float radius)
+ {
+ float circ = 2f * Mathf.PI * radius;
+
+ float xAngle = (source.x / circ) * 360f;
+
+ source.Set(0.0f, source.y, radius);
+
+ Quaternion rot = Quaternion.Euler(0.0f, xAngle, 0.0f);
+ source = rot * source;
+
+ return source;
+ }
+
+ ///
+ /// Get the relative mapping based on a source Vec3 and a radius for radial mapping.
+ ///
+ /// The source Vector3 to be mapped to cylinder
+ /// The total range of the radial in degrees as a
+ /// This is a for the radius of the radial
+ /// The current row as a for the radial calculation
+ /// The total rows as a for the radial calculation
+ /// The current column as a for the radial calculation
+ /// The total columns as a for the radial calculation
+ public static Vector3 RadialMapping(Vector3 source, float radialRange, float radius, int row, int totalRows, int column, int totalColumns)
+ {
+ float radialCellAngle = radialRange / totalColumns;
+
+ source.x = 0f;
+ source.y = 0f;
+ source.z = (radius / totalRows) * row;
+
+ float yAngle = radialCellAngle * (column - (totalColumns * 0.5f)) + (radialCellAngle * .5f);
+
+ Quaternion rot = Quaternion.Euler(0.0f, yAngle, 0.0f);
+ source = rot * source;
+
+ return source;
+ }
+
+ ///
+ /// Randomized mapping based on a source Vec3 and a radius for randomization distance.
+ ///
+ /// The source Vector3 to be mapped to cylinder
+ /// This is a for the radius of the cylinder
+ public static Vector3 ScatterMapping(Vector3 source, float radius)
+ {
+ source.x = UnityEngine.Random.Range(-radius, radius);
+ source.y = UnityEngine.Random.Range(-radius, radius);
+ return source;
+ }
+
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/VectorExtensions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/VectorExtensions.cs.meta
new file mode 100644
index 0000000..ff36a27
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Extensions/VectorExtensions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: fdc018100e504884bc9c70e9d1ccff08
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors.meta
new file mode 100644
index 0000000..cee9878
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: bb9514f032be43fb8dd3853ddd73a772
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/AssemblyInfo.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/AssemblyInfo.cs
new file mode 100644
index 0000000..02ec68b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/AssemblyInfo.cs
@@ -0,0 +1,7 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+[assembly: System.Reflection.AssemblyVersion("2.8.3.0")]
+[assembly: System.Reflection.AssemblyFileVersion("2.8.3.0")]
+
+[assembly: System.Reflection.AssemblyProduct("Microsoft® Mixed Reality Toolkit")]
+[assembly: System.Reflection.AssemblyCopyright("Copyright © Microsoft Corporation")]
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/AssemblyInfo.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/AssemblyInfo.cs.meta
new file mode 100644
index 0000000..da2a230
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/AssemblyInfo.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 995d0f2184e449345b82692bd6d4d1b4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingBoxInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingBoxInspector.cs
new file mode 100644
index 0000000..7ddaa15
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingBoxInspector.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom editor for the ClippingBox to allow for specification of the framing bounds.
+ ///
+ [CustomEditor(typeof(ClippingBox))]
+ [CanEditMultipleObjects]
+ public class ClippingBoxEditor : ClippingPrimitiveEditor
+ {
+ ///
+ protected override bool HasFrameBounds()
+ {
+ return true;
+ }
+
+ ///
+ protected override Bounds OnGetFrameBounds()
+ {
+ var primitive = target as ClippingBox;
+ Debug.Assert(primitive != null);
+ return new Bounds(primitive.transform.position, primitive.transform.lossyScale);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingBoxInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingBoxInspector.cs.meta
new file mode 100644
index 0000000..e0af38e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingBoxInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: bbc3da09f5e3ebc4b8a030465a8eeb3d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPlaneInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPlaneInspector.cs
new file mode 100644
index 0000000..66e6208
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPlaneInspector.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom editor for the ClippingPlaneEditor to allow for specification of the framing bounds.
+ ///
+ [CustomEditor(typeof(ClippingPlane))]
+ [CanEditMultipleObjects]
+ public class ClippingPlaneEditor : ClippingPrimitiveEditor
+ {
+ ///
+ protected override bool HasFrameBounds()
+ {
+ return true;
+ }
+
+ ///
+ protected override Bounds OnGetFrameBounds()
+ {
+ var primitive = target as ClippingPlane;
+ Debug.Assert(primitive != null);
+ return new Bounds(primitive.transform.position, Vector3.one);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPlaneInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPlaneInspector.cs.meta
new file mode 100644
index 0000000..f1f9e21
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPlaneInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ea66b1225d554b10a1c1b6ccb169cca9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPrimitiveInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPrimitiveInspector.cs
new file mode 100644
index 0000000..71dbd85
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPrimitiveInspector.cs
@@ -0,0 +1,69 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System.Linq;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// An abstract editor component to improve the editor experience with ClippingPrimitives.
+ ///
+ [CustomEditor(typeof(ClippingPrimitive))]
+ [CanEditMultipleObjects]
+ public abstract class ClippingPrimitiveEditor : UnityEditor.Editor
+ {
+ ///
+ /// Notifies the Unity editor if this object has custom frame bounds.
+ ///
+ /// True if custom frame bounds can be used from OnGetFrameBounds.
+ protected abstract bool HasFrameBounds();
+
+ ///
+ /// Returns the bounds the editor should focus on.
+ ///
+ /// The bounds of the clipping primitive.
+ protected abstract Bounds OnGetFrameBounds();
+
+ private ClippingPrimitive clippingPrimitive;
+
+ private void OnEnable()
+ {
+ clippingPrimitive = (ClippingPrimitive)target;
+ }
+
+ ///
+ /// Looks for changes to the list of renderers and gracefully adds and removes them.
+ ///
+ public override void OnInspectorGUI()
+ {
+ var previousRenderers = clippingPrimitive.GetRenderersCopy();
+
+ using (var check = new EditorGUI.ChangeCheckScope())
+ {
+ DrawDefaultInspector();
+
+ if (check.changed)
+ {
+ // Flagging changes other than renderers
+ clippingPrimitive.IsDirty = true;
+ }
+ }
+
+ var currentRenderers = clippingPrimitive.GetRenderersCopy();
+
+ // Add or remove and renderers that were added or removed via the inspector.
+ foreach (var renderer in previousRenderers.Except(currentRenderers))
+ {
+ clippingPrimitive.RemoveRenderer(renderer);
+ }
+
+ foreach (var renderer in currentRenderers.Except(previousRenderers))
+ {
+ clippingPrimitive.AddRenderer(renderer);
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPrimitiveInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPrimitiveInspector.cs.meta
new file mode 100644
index 0000000..ee26c61
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingPrimitiveInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 00b3a9914942baf42b886607218f3ee6
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingSphereInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingSphereInspector.cs
new file mode 100644
index 0000000..938990b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingSphereInspector.cs
@@ -0,0 +1,31 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom editor for the ClippingSphere to allow for specification of the framing bounds.
+ ///
+ [CustomEditor(typeof(ClippingSphere))]
+ [CanEditMultipleObjects]
+ public class ClippingSphereEditor : ClippingPrimitiveEditor
+ {
+ ///
+ protected override bool HasFrameBounds()
+ {
+ return true;
+ }
+
+ ///
+ protected override Bounds OnGetFrameBounds()
+ {
+ var primitive = target as ClippingSphere;
+ Debug.Assert(primitive != null);
+ return new Bounds(primitive.transform.position, primitive.Radii);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingSphereInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingSphereInspector.cs.meta
new file mode 100644
index 0000000..1ae9b35
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ClippingSphereInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 3ceb6d1e47c68c04f85ff5b33426a013
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ControllerPopupWindow.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ControllerPopupWindow.cs
new file mode 100644
index 0000000..5e0e65d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ControllerPopupWindow.cs
@@ -0,0 +1,793 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Input;
+using Microsoft.MixedReality.Toolkit.Input.Editor;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ public class ControllerPopupWindow : EditorWindow
+ {
+ // Inspectors/Data/EditorWindowOptions.json
+ private const string EditorWindowOptionsGuid = "28091d5ea9b5739419a221a06fa1ec89";
+ private const float InputActionLabelWidth = 128f;
+
+ ///
+ /// Used to enable editing the input axis label positions on controllers.
+ ///
+ private static readonly bool EnableWysiwyg = false;
+
+ private static readonly GUIContent InteractionAddButtonContent = new GUIContent("+ Add a New Interaction Mapping");
+ private static readonly GUIContent InteractionMinusButtonContent = new GUIContent("-", "Remove Interaction Mapping");
+ private static readonly GUIContent AxisTypeContent = new GUIContent("Axis Type", "The axis type of the button, e.g. Analogue, Digital, etc.");
+ private static readonly GUIContent ControllerInputTypeContent = new GUIContent("Input Type", "The primary action of the input as defined by the controller SDK.");
+ private static readonly GUIContent ActionContent = new GUIContent("Action", "Action to be raised to the Input Manager when the input data has changed.");
+ private static readonly GUIContent KeyCodeContent = new GUIContent("KeyCode", "Unity Input KeyCode id to listen for.");
+ private static readonly GUIContent XAxisContent = new GUIContent("X Axis", "Horizontal Axis to listen for.");
+ private static readonly GUIContent YAxisContent = new GUIContent("Y Axis", "Vertical Axis to listen for.");
+ private static readonly GUIContent InvertContent = new GUIContent("Invert", "Should an Axis be inverted?");
+ private static readonly GUIContent[] InvertAxisContent =
+ {
+ new GUIContent("None"),
+ new GUIContent("X"),
+ new GUIContent("Y"),
+ new GUIContent("Both")
+ };
+
+ private static readonly int[] InvertAxisValues = { 0, 1, 2, 3 };
+
+ private static readonly Vector2 InputActionLabelPosition = new Vector2(256f, 0f);
+ private static readonly Vector2 InputActionDropdownPosition = new Vector2(88f, 0f);
+ private static readonly Vector2 InputActionFlipTogglePosition = new Vector2(-24f, 0f);
+ private static readonly Vector2 HorizontalSpace = new Vector2(8f, 0f);
+
+ private static readonly Rect ControllerRectPosition = new Rect(new Vector2(128f, 0f), new Vector2(512f, 512f));
+
+ private static ControllerPopupWindow window;
+ private static ControllerInputActionOptions controllerInputActionOptions;
+
+ private static GUIContent[] axisLabels;
+ private static int[] actionIds;
+ private static GUIContent[] actionLabels;
+ private static int[] rawActionIds;
+ private static GUIContent[] rawActionLabels;
+ private static int[] digitalActionIds;
+ private static GUIContent[] digitalActionLabels;
+ private static int[] singleAxisActionIds;
+ private static GUIContent[] singleAxisActionLabels;
+ private static int[] dualAxisActionIds;
+ private static GUIContent[] dualAxisActionLabels;
+ private static int[] threeDofPositionActionIds;
+ private static GUIContent[] threeDofPositionActionLabels;
+ private static int[] threeDofRotationActionIds;
+ private static GUIContent[] threeDofRotationActionLabels;
+ private static int[] sixDofActionIds;
+ private static GUIContent[] sixDofActionLabels;
+ private static bool[] isMouseInRects;
+
+ private static bool editInputActionPositions;
+
+ private static float defaultLabelWidth;
+ private static float defaultFieldWidth;
+
+ private static Vector2 horizontalScrollPosition;
+
+ private SerializedProperty currentInteractionList;
+ private List mappedControllerList = new List();
+
+ private ControllerPopupWindow thisWindow;
+
+ private MixedRealityControllerMapping currentControllerMapping;
+
+ private Vector2 mouseDragOffset;
+ private GUIStyle flippedLabelStyle;
+ private Texture2D currentControllerTexture;
+ private ControllerInputActionOption currentControllerOption;
+
+ private void OnFocus()
+ {
+ currentControllerTexture = ControllerMappingLibrary.GetControllerTexture(currentControllerMapping.ControllerType, currentControllerMapping.Handedness);
+
+ #region Interaction Constraint Setup
+
+ actionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ axisLabels = ControllerMappingLibrary.UnityInputManagerAxes
+ .Select(axis => new GUIContent(axis.Name))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ actionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.None)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ actionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.None)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ rawActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.Raw)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ rawActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.Raw)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ digitalActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.Digital)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ digitalActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.Digital)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ singleAxisActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.SingleAxis)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ singleAxisActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.SingleAxis)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ dualAxisActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.DualAxis)
+ .Select(action => (int)action.Id).Prepend(0).ToArray();
+
+ dualAxisActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.DualAxis)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ threeDofPositionActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.ThreeDofPosition)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ threeDofPositionActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.ThreeDofPosition)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ threeDofRotationActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.ThreeDofRotation)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ threeDofRotationActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.ThreeDofRotation)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ sixDofActionIds = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.SixDof)
+ .Select(action => (int)action.Id)
+ .Prepend(0).ToArray();
+
+ sixDofActionLabels = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions
+ .Where(inputAction => inputAction.AxisConstraint == AxisType.SixDof)
+ .Select(inputAction => new GUIContent(inputAction.Description))
+ .Prepend(new GUIContent("None")).ToArray();
+
+ #endregion Interaction Constraint Setup
+ }
+
+ ///
+ /// Displays the controller mapping window for the specified controller mapping.
+ ///
+ /// The controller mapping being modified.
+ /// The underlying serialized property being modified.
+ /// The handedness of the controller.
+ /// The list of controller types affected by this mapping.
+ public static void Show(MixedRealityControllerMapping controllerMapping, SerializedProperty interactionsList, Handedness handedness = Handedness.None, List mappedControllers = null)
+ {
+ if (window != null)
+ {
+ window.Close();
+ }
+
+ window = null;
+
+ if (!MixedRealityToolkit.IsInitialized)
+ {
+ throw new InvalidOperationException("Mixed Reality Toolkit hasn't been initialized yet! Open a scene with a Mixed Reality Toolkit to initialize it before editing the controller mappings.");
+ }
+
+ window = CreateInstance();
+ window.thisWindow = window;
+ window.titleContent = new GUIContent($"{controllerMapping.Description} - Input Action Assignment");
+ window.mappedControllerList = mappedControllers;
+ window.currentControllerMapping = controllerMapping;
+ window.currentInteractionList = interactionsList;
+ isMouseInRects = new bool[interactionsList.arraySize];
+
+ string editorWindowOptionsPath = ResolveEditorWindowOptionsPath();
+ if (!File.Exists(editorWindowOptionsPath))
+ {
+ var empty = new ControllerInputActionOptions
+ {
+ Controllers = new List
+ {
+ new ControllerInputActionOption
+ {
+ Controller = 0,
+ Handedness = Handedness.None,
+ InputLabelPositions = new[] {new Vector2(0, 0)},
+ IsLabelFlipped = new []{false}
+ }
+ }
+ };
+
+ File.WriteAllText(editorWindowOptionsPath, JsonUtility.ToJson(empty, true));
+ AssetDatabase.Refresh(ImportAssetOptions.ForceUpdate);
+ }
+ else
+ {
+ controllerInputActionOptions = JsonUtility.FromJson(File.ReadAllText(editorWindowOptionsPath));
+
+ if (controllerInputActionOptions.Controllers.Any(option => option.Controller == controllerMapping.SupportedControllerType && option.Handedness == handedness))
+ {
+ window.currentControllerOption = controllerInputActionOptions.Controllers.FirstOrDefault(option => option.Controller == controllerMapping.SupportedControllerType && option.Handedness == handedness);
+
+ if (window.currentControllerOption != null && window.currentControllerOption.IsLabelFlipped == null)
+ {
+ window.currentControllerOption.IsLabelFlipped = new bool[interactionsList.arraySize];
+ }
+ }
+ }
+
+ var windowSize = new Vector2(controllerMapping.HasCustomInteractionMappings ? 896f : 768f, 512f);
+ window.maxSize = windowSize;
+ window.minSize = windowSize;
+ window.CenterOnMainWin();
+ window.ShowUtility();
+
+ defaultLabelWidth = EditorGUIUtility.labelWidth;
+ defaultFieldWidth = EditorGUIUtility.fieldWidth;
+ }
+
+ ///
+ /// Use this to repaint the pop-up window.
+ ///
+ public static void RepaintWindow()
+ {
+ if (window != null && window.thisWindow != null)
+ {
+ window.thisWindow.Repaint();
+ }
+ }
+
+ private void Update()
+ {
+ if (editInputActionPositions)
+ {
+ Repaint();
+ }
+ }
+
+ private void OnGUI()
+ {
+ if (flippedLabelStyle == null)
+ {
+ flippedLabelStyle = new GUIStyle("Label")
+ {
+ alignment = TextAnchor.UpperRight,
+ stretchWidth = true
+ };
+ }
+
+ if (!currentControllerMapping.HasCustomInteractionMappings && currentControllerTexture != null)
+ {
+ GUILayout.BeginHorizontal();
+ GUI.DrawTexture(ControllerRectPosition, currentControllerTexture);
+ GUILayout.EndHorizontal();
+ }
+
+ try
+ {
+ RenderInteractionList(currentInteractionList, currentControllerMapping.HasCustomInteractionMappings);
+ RenderMappingList(mappedControllerList);
+ }
+ catch (Exception)
+ {
+ thisWindow.Close();
+ }
+ }
+
+ private void RenderMappingList(List controllerList)
+ {
+ if (controllerList == null)
+ {
+ return;
+ }
+
+ using (new EditorGUILayout.VerticalScope())
+ {
+ GUILayout.FlexibleSpace();
+ EditorGUILayout.LabelField("Controllers affected by this mapping", EditorStyles.boldLabel);
+ for (int i = 0; i < controllerList.Count; i++)
+ {
+ EditorGUILayout.LabelField(controllerList[i]);
+ }
+ }
+ }
+
+ private void RenderInteractionList(SerializedProperty interactionList, bool useCustomInteractionMapping)
+ {
+ if (interactionList == null) { throw new Exception(); }
+
+ bool noInteractions = interactionList.arraySize == 0;
+
+ if (currentControllerOption != null && (currentControllerOption.IsLabelFlipped == null || currentControllerOption.IsLabelFlipped.Length != interactionList.arraySize))
+ {
+ currentControllerOption.IsLabelFlipped = new bool[interactionList.arraySize];
+ }
+
+ GUILayout.BeginVertical();
+
+ if (useCustomInteractionMapping)
+ {
+ horizontalScrollPosition = EditorGUILayout.BeginScrollView(horizontalScrollPosition, false, false, GUILayout.ExpandWidth(true), GUILayout.ExpandWidth(true));
+ }
+
+ if (useCustomInteractionMapping)
+ {
+ if (GUILayout.Button(InteractionAddButtonContent))
+ {
+ interactionList.arraySize += 1;
+ var interaction = interactionList.GetArrayElementAtIndex(interactionList.arraySize - 1);
+ var axisType = interaction.FindPropertyRelative("axisType");
+ axisType.enumValueIndex = 0;
+ var inputType = interaction.FindPropertyRelative("inputType");
+ inputType.enumValueIndex = 0;
+ var action = interaction.FindPropertyRelative("inputAction");
+ var actionId = action.FindPropertyRelative("id");
+ var actionDescription = action.FindPropertyRelative("description");
+ actionDescription.stringValue = "None";
+ actionId.intValue = 0;
+ }
+
+ if (noInteractions)
+ {
+ EditorGUILayout.HelpBox("Create an Interaction Mapping.", MessageType.Warning);
+ EditorGUILayout.EndScrollView();
+ GUILayout.EndVertical();
+ return;
+ }
+ }
+ else if (EnableWysiwyg)
+ {
+ EditorGUI.BeginChangeCheck();
+ editInputActionPositions = EditorGUILayout.Toggle("Edit Input Action Positions", editInputActionPositions);
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ string editorWindowOptionsPath = ResolveEditorWindowOptionsPath();
+ if (!editInputActionPositions)
+ {
+ File.WriteAllText(editorWindowOptionsPath, JsonUtility.ToJson(controllerInputActionOptions, true));
+ }
+ else
+ {
+ if (!controllerInputActionOptions.Controllers.Any(
+ option => option.Controller == currentControllerMapping.SupportedControllerType && option.Handedness == currentControllerMapping.Handedness))
+ {
+ currentControllerOption = new ControllerInputActionOption
+ {
+ Controller = currentControllerMapping.SupportedControllerType,
+ Handedness = currentControllerMapping.Handedness,
+ InputLabelPositions = new Vector2[currentInteractionList.arraySize],
+ IsLabelFlipped = new bool[currentInteractionList.arraySize]
+ };
+
+ controllerInputActionOptions.Controllers.Add(currentControllerOption);
+ isMouseInRects = new bool[currentInteractionList.arraySize];
+
+ if (controllerInputActionOptions.Controllers.Any(option => option.Controller == 0))
+ {
+ controllerInputActionOptions.Controllers.Remove(
+ controllerInputActionOptions.Controllers.Find(option =>
+ option.Controller == 0));
+ }
+
+ File.WriteAllText(editorWindowOptionsPath, JsonUtility.ToJson(controllerInputActionOptions, true));
+ }
+ }
+ }
+ }
+
+ GUILayout.BeginHorizontal();
+
+ if (useCustomInteractionMapping)
+ {
+ EditorGUILayout.LabelField("Id", GUILayout.Width(32f));
+ EditorGUIUtility.labelWidth = 24f;
+ EditorGUIUtility.fieldWidth = 24f;
+ EditorGUILayout.LabelField(ControllerInputTypeContent, GUILayout.Width(InputActionLabelWidth));
+ EditorGUILayout.LabelField(AxisTypeContent, GUILayout.Width(InputActionLabelWidth));
+ EditorGUILayout.LabelField(ActionContent, GUILayout.Width(InputActionLabelWidth));
+ EditorGUILayout.LabelField(KeyCodeContent, GUILayout.Width(InputActionLabelWidth));
+ EditorGUILayout.LabelField(XAxisContent, GUILayout.Width(InputActionLabelWidth));
+ EditorGUILayout.LabelField(YAxisContent, GUILayout.Width(InputActionLabelWidth));
+ GUILayout.FlexibleSpace();
+ EditorGUILayout.LabelField(string.Empty, GUILayout.Width(24f));
+
+ EditorGUIUtility.labelWidth = defaultLabelWidth;
+ EditorGUIUtility.fieldWidth = defaultFieldWidth;
+ }
+
+ GUILayout.EndHorizontal();
+
+ for (int i = 0; i < interactionList.arraySize; i++)
+ {
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ SerializedProperty interaction = interactionList.GetArrayElementAtIndex(i);
+
+ if (useCustomInteractionMapping)
+ {
+ EditorGUILayout.LabelField($"{i + 1}", GUILayout.Width(32f));
+ var inputType = interaction.FindPropertyRelative("inputType");
+ EditorGUILayout.PropertyField(inputType, GUIContent.none, GUILayout.Width(InputActionLabelWidth));
+ var axisType = interaction.FindPropertyRelative("axisType");
+ EditorGUILayout.PropertyField(axisType, GUIContent.none, GUILayout.Width(InputActionLabelWidth));
+ var invertXAxis = interaction.FindPropertyRelative("invertXAxis");
+ var invertYAxis = interaction.FindPropertyRelative("invertYAxis");
+ var interactionAxisConstraint = interaction.FindPropertyRelative("axisType");
+
+ var action = interaction.FindPropertyRelative("inputAction");
+ var actionId = action.FindPropertyRelative("id");
+ var actionDescription = action.FindPropertyRelative("description");
+ var actionConstraint = action.FindPropertyRelative("axisConstraint");
+
+ GUIContent[] labels;
+ int[] ids;
+
+ switch ((AxisType)interactionAxisConstraint.intValue)
+ {
+ default:
+ case AxisType.None:
+ labels = actionLabels;
+ ids = actionIds;
+ break;
+ case AxisType.Raw:
+ labels = rawActionLabels;
+ ids = rawActionIds;
+ break;
+ case AxisType.Digital:
+ labels = digitalActionLabels;
+ ids = digitalActionIds;
+ break;
+ case AxisType.SingleAxis:
+ labels = singleAxisActionLabels;
+ ids = singleAxisActionIds;
+ break;
+ case AxisType.DualAxis:
+ labels = dualAxisActionLabels;
+ ids = dualAxisActionIds;
+ break;
+ case AxisType.ThreeDofPosition:
+ labels = threeDofPositionActionLabels;
+ ids = threeDofPositionActionIds;
+ break;
+ case AxisType.ThreeDofRotation:
+ labels = threeDofRotationActionLabels;
+ ids = threeDofRotationActionIds;
+ break;
+ case AxisType.SixDof:
+ labels = sixDofActionLabels;
+ ids = sixDofActionIds;
+ break;
+ }
+
+ EditorGUI.BeginChangeCheck();
+ actionId.intValue = EditorGUILayout.IntPopup(GUIContent.none, actionId.intValue, labels, ids, GUILayout.Width(InputActionLabelWidth));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ var inputAction = actionId.intValue == 0 ? MixedRealityInputAction.None : MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions[actionId.intValue - 1];
+ actionDescription.stringValue = inputAction.Description;
+ actionConstraint.intValue = (int)inputAction.AxisConstraint;
+ }
+
+ if ((AxisType)axisType.intValue == AxisType.Digital)
+ {
+ var keyCode = interaction.FindPropertyRelative("keyCode");
+ EditorGUILayout.PropertyField(keyCode, GUIContent.none, GUILayout.Width(InputActionLabelWidth));
+ }
+ else
+ {
+ if ((AxisType)axisType.intValue == AxisType.DualAxis)
+ {
+ EditorGUIUtility.labelWidth = InputActionLabelWidth * 0.5f;
+ EditorGUIUtility.fieldWidth = InputActionLabelWidth * 0.5f;
+
+ int currentAxisSetting = 0;
+
+ if (invertXAxis.boolValue)
+ {
+ currentAxisSetting += 1;
+ }
+
+ if (invertYAxis.boolValue)
+ {
+ currentAxisSetting += 2;
+ }
+
+ EditorGUI.BeginChangeCheck();
+ currentAxisSetting = EditorGUILayout.IntPopup(InvertContent, currentAxisSetting, InvertAxisContent, InvertAxisValues, GUILayout.Width(InputActionLabelWidth));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ switch (currentAxisSetting)
+ {
+ case 0:
+ invertXAxis.boolValue = false;
+ invertYAxis.boolValue = false;
+ break;
+ case 1:
+ invertXAxis.boolValue = true;
+ invertYAxis.boolValue = false;
+ break;
+ case 2:
+ invertXAxis.boolValue = false;
+ invertYAxis.boolValue = true;
+ break;
+ case 3:
+ invertXAxis.boolValue = true;
+ invertYAxis.boolValue = true;
+ break;
+ }
+ }
+
+ EditorGUIUtility.labelWidth = defaultLabelWidth;
+ EditorGUIUtility.fieldWidth = defaultFieldWidth;
+ }
+ else if ((AxisType)axisType.intValue == AxisType.SingleAxis)
+ {
+ invertXAxis.boolValue = EditorGUILayout.ToggleLeft("Invert X", invertXAxis.boolValue, GUILayout.Width(InputActionLabelWidth));
+ EditorGUIUtility.labelWidth = defaultLabelWidth;
+ }
+ else
+ {
+ EditorGUILayout.LabelField(GUIContent.none, GUILayout.Width(InputActionLabelWidth));
+ }
+ }
+
+ if ((AxisType)axisType.intValue == AxisType.SingleAxis ||
+ (AxisType)axisType.intValue == AxisType.DualAxis)
+ {
+ var axisCodeX = interaction.FindPropertyRelative("axisCodeX");
+ RenderAxisPopup(axisCodeX, InputActionLabelWidth);
+ }
+ else
+ {
+ EditorGUILayout.LabelField(GUIContent.none, GUILayout.Width(InputActionLabelWidth));
+ }
+
+ if ((AxisType)axisType.intValue == AxisType.DualAxis)
+ {
+ var axisCodeY = interaction.FindPropertyRelative("axisCodeY");
+ RenderAxisPopup(axisCodeY, InputActionLabelWidth);
+ }
+ else
+ {
+ EditorGUILayout.LabelField(GUIContent.none, GUILayout.Width(InputActionLabelWidth));
+ }
+
+ if (GUILayout.Button(InteractionMinusButtonContent, EditorStyles.miniButtonRight, GUILayout.ExpandWidth(true)))
+ {
+ interactionList.DeleteArrayElementAtIndex(i);
+ }
+ }
+ else
+ {
+ var interactionDescription = interaction.FindPropertyRelative("description");
+ var interactionAxisConstraint = interaction.FindPropertyRelative("axisType");
+ var action = interaction.FindPropertyRelative("inputAction");
+ var actionId = action.FindPropertyRelative("id");
+ var actionDescription = action.FindPropertyRelative("description");
+ var actionConstraint = action.FindPropertyRelative("axisConstraint");
+
+ GUIContent[] labels;
+ int[] ids;
+
+ switch ((AxisType)interactionAxisConstraint.intValue)
+ {
+ default:
+ case AxisType.None:
+ labels = actionLabels;
+ ids = actionIds;
+ break;
+ case AxisType.Raw:
+ labels = rawActionLabels;
+ ids = rawActionIds;
+ break;
+ case AxisType.Digital:
+ labels = digitalActionLabels;
+ ids = digitalActionIds;
+ break;
+ case AxisType.SingleAxis:
+ labels = singleAxisActionLabels;
+ ids = singleAxisActionIds;
+ break;
+ case AxisType.DualAxis:
+ labels = dualAxisActionLabels;
+ ids = dualAxisActionIds;
+ break;
+ case AxisType.ThreeDofPosition:
+ labels = threeDofPositionActionLabels;
+ ids = threeDofPositionActionIds;
+ break;
+ case AxisType.ThreeDofRotation:
+ labels = threeDofRotationActionLabels;
+ ids = threeDofRotationActionIds;
+ break;
+ case AxisType.SixDof:
+ labels = sixDofActionLabels;
+ ids = sixDofActionIds;
+ break;
+ }
+
+ EditorGUI.BeginChangeCheck();
+
+ if (currentControllerOption == null || currentControllerTexture == null)
+ {
+ bool skip = false;
+ var description = interactionDescription.stringValue;
+ if (currentControllerMapping.SupportedControllerType == SupportedControllerType.GGVHand
+ && currentControllerMapping.Handedness == Handedness.None)
+ {
+ if (description != "Select")
+ {
+ skip = true;
+ }
+ }
+
+ if (!skip)
+ {
+ actionId.intValue = EditorGUILayout.IntPopup(GUIContent.none, actionId.intValue, labels, ids, GUILayout.Width(80f));
+ EditorGUILayout.LabelField(description, GUILayout.ExpandWidth(true));
+ }
+ }
+ else
+ {
+ var rectPosition = currentControllerOption.InputLabelPositions[i];
+ var rectSize = InputActionLabelPosition + InputActionDropdownPosition + new Vector2(currentControllerOption.IsLabelFlipped[i] ? 0f : 8f, EditorGUIUtility.singleLineHeight);
+ GUI.Box(new Rect(rectPosition, rectSize), GUIContent.none, EditorGUIUtility.isProSkin ? "ObjectPickerBackground" : "ObjectPickerResultsEven");
+ var offset = currentControllerOption.IsLabelFlipped[i] ? InputActionLabelPosition : Vector2.zero;
+ var popupRect = new Rect(rectPosition + offset, new Vector2(InputActionDropdownPosition.x, EditorGUIUtility.singleLineHeight));
+
+ actionId.intValue = EditorGUI.IntPopup(popupRect, actionId.intValue, labels, ids);
+ offset = currentControllerOption.IsLabelFlipped[i] ? Vector2.zero : InputActionDropdownPosition;
+ var labelRect = new Rect(rectPosition + offset, new Vector2(InputActionLabelPosition.x, EditorGUIUtility.singleLineHeight));
+ EditorGUI.LabelField(labelRect, interactionDescription.stringValue, currentControllerOption.IsLabelFlipped[i] ? flippedLabelStyle : EditorStyles.label);
+
+ if (editInputActionPositions)
+ {
+ offset = currentControllerOption.IsLabelFlipped[i] ? InputActionLabelPosition + InputActionDropdownPosition + HorizontalSpace : InputActionFlipTogglePosition;
+ var toggleRect = new Rect(rectPosition + offset, new Vector2(-InputActionFlipTogglePosition.x, EditorGUIUtility.singleLineHeight));
+
+ EditorGUI.BeginChangeCheck();
+ currentControllerOption.IsLabelFlipped[i] = EditorGUI.Toggle(toggleRect, currentControllerOption.IsLabelFlipped[i]);
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (currentControllerOption.IsLabelFlipped[i])
+ {
+ currentControllerOption.InputLabelPositions[i] -= InputActionLabelPosition;
+ }
+ else
+ {
+ currentControllerOption.InputLabelPositions[i] += InputActionLabelPosition;
+ }
+ }
+
+ if (!isMouseInRects.Any(value => value) || isMouseInRects[i])
+ {
+ if (Event.current.type == EventType.MouseDrag && labelRect.Contains(Event.current.mousePosition) && !isMouseInRects[i])
+ {
+ isMouseInRects[i] = true;
+ mouseDragOffset = Event.current.mousePosition - currentControllerOption.InputLabelPositions[i];
+ }
+ else if (Event.current.type == EventType.Repaint && isMouseInRects[i])
+ {
+ currentControllerOption.InputLabelPositions[i] = Event.current.mousePosition - mouseDragOffset;
+ }
+ else if (Event.current.type == EventType.DragUpdated && isMouseInRects[i])
+ {
+ currentControllerOption.InputLabelPositions[i] = Event.current.mousePosition - mouseDragOffset;
+ }
+ else if (Event.current.type == EventType.MouseUp && isMouseInRects[i])
+ {
+ currentControllerOption.InputLabelPositions[i] = Event.current.mousePosition - mouseDragOffset;
+ mouseDragOffset = Vector2.zero;
+ isMouseInRects[i] = false;
+ }
+ }
+ }
+ }
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ MixedRealityInputAction inputAction = actionId.intValue == 0 ?
+ MixedRealityInputAction.None :
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions[actionId.intValue - 1];
+ actionId.intValue = (int)inputAction.Id;
+ actionDescription.stringValue = inputAction.Description;
+ actionConstraint.intValue = (int)inputAction.AxisConstraint;
+ interactionList.serializedObject.ApplyModifiedProperties();
+ }
+ }
+ }
+ }
+
+ if (useCustomInteractionMapping)
+ {
+ EditorGUILayout.EndScrollView();
+ interactionList.serializedObject.ApplyModifiedProperties();
+ }
+
+ GUILayout.EndVertical();
+ }
+
+ private static void RenderAxisPopup(SerializedProperty axisCode, float customLabelWidth)
+ {
+ var axisId = -1;
+
+ for (int j = 0; j < ControllerMappingLibrary.UnityInputManagerAxes.Length; j++)
+ {
+ if (ControllerMappingLibrary.UnityInputManagerAxes[j].Name == axisCode.stringValue)
+ {
+ axisId = j + 1;
+ break;
+ }
+ }
+
+ EditorGUI.BeginChangeCheck();
+ axisId = EditorGUILayout.IntPopup(GUIContent.none, axisId, axisLabels, null, GUILayout.Width(customLabelWidth));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (axisId == 0)
+ {
+ axisCode.stringValue = string.Empty;
+ axisCode.serializedObject.ApplyModifiedProperties();
+ }
+ else
+ {
+ for (int j = 0; j < ControllerMappingLibrary.UnityInputManagerAxes.Length; j++)
+ {
+ if (axisId - 1 == j)
+ {
+ axisCode.stringValue = ControllerMappingLibrary.UnityInputManagerAxes[j].Name;
+ axisCode.serializedObject.ApplyModifiedProperties();
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ private static string ResolveEditorWindowOptionsPath()
+ {
+ return Path.GetFullPath(AssetDatabase.GUIDToAssetPath(EditorWindowOptionsGuid));
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ControllerPopupWindow.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ControllerPopupWindow.cs.meta
new file mode 100644
index 0000000..be3705e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/ControllerPopupWindow.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 66928ed4a53ad00429dd8597ce18a858
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data.meta
new file mode 100644
index 0000000..0f83819
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 01081ff15c3de9a45834acf3cbf52c00
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOption.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOption.cs
new file mode 100644
index 0000000..586bb26
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOption.cs
@@ -0,0 +1,21 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ ///
+ /// Used to aid in layout of Controller Input Actions.
+ ///
+ [Serializable]
+ public class ControllerInputActionOption
+ {
+ public SupportedControllerType Controller;
+ public Handedness Handedness;
+ public Vector2[] InputLabelPositions;
+ public bool[] IsLabelFlipped;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOption.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOption.cs.meta
new file mode 100644
index 0000000..8268ffc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOption.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 3a1857d39eeeb034fb61982a68634e34
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOptions.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOptions.cs
new file mode 100644
index 0000000..3ac408f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOptions.cs
@@ -0,0 +1,17 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using System.Collections.Generic;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ ///
+ /// Used to aid in layout of Controller Input Actions.
+ ///
+ [Serializable]
+ public class ControllerInputActionOptions
+ {
+ public List Controllers;
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOptions.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOptions.cs.meta
new file mode 100644
index 0000000..1747e89
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/ControllerInputActionOptions.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9517fcc8e5ebbbb48a9f9b15d57752c9
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/EditorWindowOptions.json b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/EditorWindowOptions.json
new file mode 100644
index 0000000..0c9a00a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/EditorWindowOptions.json
@@ -0,0 +1,538 @@
+{
+ "Controllers": [
+ {
+ "Controller": 32,
+ "Handedness": 2,
+ "InputLabelPositions": [
+ {
+ "x": 291.0,
+ "y": 22.0
+ },
+ {
+ "x": 382.0,
+ "y": 348.0
+ },
+ {
+ "x": -48.0,
+ "y": 283.0
+ },
+ {
+ "x": 486.0,
+ "y": 144.0
+ },
+ {
+ "x": -84.0,
+ "y": 158.0
+ },
+ {
+ "x": -82.0,
+ "y": 136.0
+ },
+ {
+ "x": 484.0,
+ "y": 190.0
+ },
+ {
+ "x": 484.0,
+ "y": 211.0
+ },
+ {
+ "x": 484.0,
+ "y": 231.0
+ },
+ {
+ "x": -70.0,
+ "y": 241.0
+ },
+ {
+ "x": -77.0,
+ "y": 191.0
+ },
+ {
+ "x": -78.0,
+ "y": 211.0
+ }
+ ],
+ "IsLabelFlipped": [
+ false,
+ false,
+ true,
+ false,
+ true,
+ true,
+ false,
+ false,
+ false,
+ true,
+ true,
+ true
+ ]
+ },
+ {
+ "Controller": 32,
+ "Handedness": 1,
+ "InputLabelPositions": [
+ {
+ "x": 121.0,
+ "y": 21.0
+ },
+ {
+ "x": 404.0,
+ "y": 355.0
+ },
+ {
+ "x": 469.0,
+ "y": 283.0
+ },
+ {
+ "x": -67.0,
+ "y": 145.0
+ },
+ {
+ "x": 500.0,
+ "y": 158.0
+ },
+ {
+ "x": 501.0,
+ "y": 137.0
+ },
+ {
+ "x": -68.0,
+ "y": 182.0
+ },
+ {
+ "x": -69.0,
+ "y": 205.0
+ },
+ {
+ "x": -69.0,
+ "y": 226.0
+ },
+ {
+ "x": 488.0,
+ "y": 238.0
+ },
+ {
+ "x": 499.0,
+ "y": 190.0
+ },
+ {
+ "x": 499.0,
+ "y": 211.0
+ }
+ ],
+ "IsLabelFlipped": [
+ true,
+ false,
+ false,
+ true,
+ false,
+ false,
+ true,
+ true,
+ true,
+ false,
+ false,
+ false
+ ]
+ },
+ {
+ "Controller": 128,
+ "Handedness": 0,
+ "InputLabelPositions": [
+ {
+ "x": 15.0,
+ "y": 398.0
+ },
+ {
+ "x": 14.0,
+ "y": 419.0
+ },
+ {
+ "x": 418.0,
+ "y": 399.0
+ },
+ {
+ "x": 418.0,
+ "y": 420.0
+ },
+ {
+ "x": -188.0,
+ "y": 338.0
+ },
+ {
+ "x": 298.0,
+ "y": 54.0
+ },
+ {
+ "x": -132.0,
+ "y": 134.0
+ },
+ {
+ "x": 551.0,
+ "y": 126.0
+ },
+ {
+ "x": 14.0,
+ "y": 101.0
+ },
+ {
+ "x": 409.0,
+ "y": 101.0
+ },
+ {
+ "x": -159.0,
+ "y": 182.0
+ },
+ {
+ "x": 577.0,
+ "y": 173.0
+ },
+ {
+ "x": 628.0,
+ "y": 310.0
+ },
+ {
+ "x": 622.0,
+ "y": 282.0
+ },
+ {
+ "x": 616.0,
+ "y": 253.0
+ },
+ {
+ "x": 607.0,
+ "y": 225.0
+ }
+ ],
+ "IsLabelFlipped": [
+ true,
+ true,
+ false,
+ false,
+ true,
+ false,
+ true,
+ false,
+ true,
+ false,
+ true,
+ false,
+ false,
+ false,
+ false,
+ false
+ ]
+ },
+ {
+ "Controller": 8,
+ "Handedness": 2,
+ "InputLabelPositions": [
+ {
+ "x": -43.0,
+ "y": 165.0
+ },
+ {
+ "x": 5.0,
+ "y": 289.0
+ },
+ {
+ "x": 7.0,
+ "y": 330.0
+ },
+ {
+ "x": 8.0,
+ "y": 351.0
+ },
+ {
+ "x": 6.0,
+ "y": 309.0
+ },
+ {
+ "x": 436.0,
+ "y": 272.0
+ },
+ {
+ "x": 267.0,
+ "y": 30.0
+ },
+ {
+ "x": 267.0,
+ "y": 68.0
+ },
+ {
+ "x": 268.0,
+ "y": 88.0
+ },
+ {
+ "x": 267.0,
+ "y": 49.0
+ },
+ {
+ "x": 418.0,
+ "y": 163.0
+ },
+ {
+ "x": -43.0,
+ "y": 204.0
+ },
+ {
+ "x": 418.0,
+ "y": 182.0
+ },
+ {
+ "x": -42.0,
+ "y": 184.0
+ },
+ {
+ "x": 420.0,
+ "y": 202.0
+ },
+ {
+ "x": 419.0,
+ "y": 222.0
+ }
+ ],
+ "IsLabelFlipped": [
+ true,
+ true,
+ true,
+ true,
+ true,
+ false,
+ false,
+ false,
+ false,
+ false,
+ false,
+ true,
+ false,
+ true,
+ false,
+ false
+ ]
+ },
+ {
+ "Controller": 8,
+ "Handedness": 1,
+ "InputLabelPositions": [
+ {
+ "x": 482.0,
+ "y": 158.0
+ },
+ {
+ "x": 433.0,
+ "y": 295.0
+ },
+ {
+ "x": 433.0,
+ "y": 334.0
+ },
+ {
+ "x": 433.0,
+ "y": 355.0
+ },
+ {
+ "x": 433.0,
+ "y": 315.0
+ },
+ {
+ "x": -14.0,
+ "y": 268.0
+ },
+ {
+ "x": 171.0,
+ "y": 28.0
+ },
+ {
+ "x": 171.0,
+ "y": 67.0
+ },
+ {
+ "x": 171.0,
+ "y": 87.0
+ },
+ {
+ "x": 172.0,
+ "y": 47.0
+ },
+ {
+ "x": 15.0,
+ "y": 168.0
+ },
+ {
+ "x": 480.0,
+ "y": 179.0
+ },
+ {
+ "x": 16.0,
+ "y": 141.0
+ },
+ {
+ "x": 15.0,
+ "y": 188.0
+ },
+ {
+ "x": 479.0,
+ "y": 198.0
+ },
+ {
+ "x": 15.0,
+ "y": 207.0
+ },
+ {
+ "x": 14.0,
+ "y": 225.0
+ }
+ ],
+ "IsLabelFlipped": [
+ false,
+ false,
+ false,
+ false,
+ false,
+ true,
+ true,
+ true,
+ true,
+ true,
+ true,
+ false,
+ true,
+ true,
+ false,
+ true,
+ true
+ ]
+ },
+ {
+ "Controller": 2,
+ "Handedness": 1,
+ "InputLabelPositions": [
+ {
+ "x": 198.0,
+ "y": 49.0
+ },
+ {
+ "x": 545.0,
+ "y": 215.0
+ },
+ {
+ "x": 543.0,
+ "y": 233.0
+ },
+ {
+ "x": 543.0,
+ "y": 251.0
+ },
+ {
+ "x": 542.0,
+ "y": 298.0
+ },
+ {
+ "x": -131.0,
+ "y": 216.0
+ },
+ {
+ "x": -131.0,
+ "y": 256.0
+ },
+ {
+ "x": -131.0,
+ "y": 235.0
+ },
+ {
+ "x": -125.0,
+ "y": 173.0
+ }
+ ],
+ "IsLabelFlipped": [
+ false,
+ false,
+ false,
+ false,
+ false,
+ true,
+ true,
+ true,
+ true
+ ]
+ },
+ {
+ "Controller": 2,
+ "Handedness": 2,
+ "InputLabelPositions": [
+ {
+ "x": 418.0,
+ "y": 43.0
+ },
+ {
+ "x": -118.0,
+ "y": 208.0
+ },
+ {
+ "x": -117.0,
+ "y": 228.0
+ },
+ {
+ "x": -117.0,
+ "y": 248.0
+ },
+ {
+ "x": -117.0,
+ "y": 285.0
+ },
+ {
+ "x": 563.0,
+ "y": 212.0
+ },
+ {
+ "x": 565.0,
+ "y": 252.0
+ },
+ {
+ "x": 563.0,
+ "y": 232.0
+ },
+ {
+ "x": 568.0,
+ "y": 168.0
+ }
+ ],
+ "IsLabelFlipped": [
+ false,
+ true,
+ true,
+ true,
+ true,
+ false,
+ false,
+ false,
+ false
+ ]
+ },
+ {
+ "Controller": 16,
+ "Handedness": 0,
+ "InputLabelPositions": [
+ {
+ "x": 301.0,
+ "y": 63.0
+ },
+ {
+ "x": -64.0,
+ "y": 147.0
+ },
+ {
+ "x": 490.0,
+ "y": 226.0
+ }
+ ],
+ "IsLabelFlipped": [
+ false,
+ true,
+ false
+ ]
+ }
+ ]
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/EditorWindowOptions.json.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/EditorWindowOptions.json.meta
new file mode 100644
index 0000000..6206498
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Data/EditorWindowOptions.json.meta
@@ -0,0 +1,7 @@
+fileFormatVersion: 2
+guid: 28091d5ea9b5739419a221a06fa1ec89
+TextScriptImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/HoverLightInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/HoverLightInspector.cs
new file mode 100644
index 0000000..57f4c48
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/HoverLightInspector.cs
@@ -0,0 +1,36 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ [CustomEditor(typeof(HoverLight))]
+ public class HoverLightInspector : UnityEditor.Editor
+ {
+ private bool HasFrameBounds() { return true; }
+
+ private Bounds OnGetFrameBounds()
+ {
+ var light = target as HoverLight;
+ Debug.Assert(light != null);
+ return new Bounds(light.transform.position, Vector3.one * light.Radius);
+ }
+
+ [MenuItem("GameObject/Light/Hover Light")]
+ private static void CreateHoverLight(MenuCommand menuCommand)
+ {
+ GameObject hoverLight = new GameObject("Hover Light", typeof(HoverLight));
+
+ // Ensure the light gets re-parented to the active context.
+ GameObjectUtility.SetParentAndAlign(hoverLight, menuCommand.context as GameObject);
+
+ // Register the creation in the undo system.
+ Undo.RegisterCreatedObjectUndo(hoverLight, "Create " + hoverLight.name);
+
+ Selection.activeObject = hoverLight;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/HoverLightInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/HoverLightInspector.cs.meta
new file mode 100644
index 0000000..3f4a170
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/HoverLightInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f912602b961f4064be7f83f4ee817167
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MRTK.Inspectors.asmdef b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MRTK.Inspectors.asmdef
new file mode 100644
index 0000000..a4e0972
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MRTK.Inspectors.asmdef
@@ -0,0 +1,24 @@
+{
+ "name": "Microsoft.MixedReality.Toolkit.Editor.Inspectors",
+ "references": [
+ "Microsoft.MixedReality.Toolkit",
+ "Microsoft.MixedReality.Toolkit.Async",
+ "Microsoft.MixedReality.Toolkit.Editor.BuildAndDeploy",
+ "Microsoft.MixedReality.Toolkit.Editor.ClassExtensions",
+ "Microsoft.MixedReality.Toolkit.Editor.Utilities",
+ "Unity.TextMeshPro.Editor",
+ "Unity.TextMeshPro"
+ ],
+ "optionalUnityReferences": [],
+ "includePlatforms": [
+ "Editor"
+ ],
+ "excludePlatforms": [],
+ "allowUnsafeCode": false,
+ "overrideReferences": false,
+ "precompiledReferences": [],
+ "autoReferenced": true,
+ "defineConstraints": [],
+ "versionDefines": [],
+ "noEngineReferences": false
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MRTK.Inspectors.asmdef.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MRTK.Inspectors.asmdef.meta
new file mode 100644
index 0000000..5cfcd04
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MRTK.Inspectors.asmdef.meta
@@ -0,0 +1,7 @@
+fileFormatVersion: 2
+guid: 3fd8365dd9cf49cc9886d651945cdaae
+AssemblyDefinitionImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityShaderGUI.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityShaderGUI.cs
new file mode 100644
index 0000000..211fde1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityShaderGUI.cs
@@ -0,0 +1,418 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System;
+using UnityEditor;
+using UnityEngine;
+using UnityEngine.Rendering;
+using Object = UnityEngine.Object;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom base shader inspector for Mixed Reality Toolkit shaders.
+ ///
+ public abstract class MixedRealityShaderGUI : ShaderGUI
+ {
+ protected enum RenderingMode
+ {
+ Opaque = 0,
+ Cutout = 1,
+ Fade = 2,
+ Transparent = 3,
+ Additive = 4,
+ Custom = 5
+ }
+
+ protected enum CustomRenderingMode
+ {
+ Opaque = 0,
+ Cutout = 1,
+ Fade = 2
+ }
+
+ protected enum DepthWrite
+ {
+ Off = 0,
+ On = 1
+ }
+
+ protected static class BaseStyles
+ {
+ public static string renderingOptionsTitle = "Rendering Options";
+ public static string advancedOptionsTitle = "Advanced Options";
+ public static string renderTypeName = "RenderType";
+ public static string renderingModeName = "_Mode";
+ public static string customRenderingModeName = "_CustomMode";
+ public static string sourceBlendName = "_SrcBlend";
+ public static string destinationBlendName = "_DstBlend";
+ public static string blendOperationName = "_BlendOp";
+ public static string depthTestName = "_ZTest";
+ public static string depthWriteName = "_ZWrite";
+ public static string depthOffsetFactorName = "_ZOffsetFactor";
+ public static string depthOffsetUnitsName = "_ZOffsetUnits";
+ public static string colorWriteMaskName = "_ColorWriteMask";
+
+ public static string cullModeName = "_CullMode";
+ public static string renderQueueOverrideName = "_RenderQueueOverride";
+
+ public static string alphaTestOnName = "_ALPHATEST_ON";
+ public static string alphaBlendOnName = "_ALPHABLEND_ON";
+
+ public static readonly string[] renderingModeNames = Enum.GetNames(typeof(RenderingMode));
+ public static readonly string[] customRenderingModeNames = Enum.GetNames(typeof(CustomRenderingMode));
+ public static readonly string[] depthWriteNames = Enum.GetNames(typeof(DepthWrite));
+ public static GUIContent sourceBlend = new GUIContent("Source Blend", "Blend Mode of Newly Calculated Color");
+ public static GUIContent destinationBlend = new GUIContent("Destination Blend", "Blend Mode of Existing Color");
+ public static GUIContent blendOperation = new GUIContent("Blend Operation", "Operation for Blending New Color With Existing Color");
+ public static GUIContent depthTest = new GUIContent("Depth Test", "How Should Depth Testing Be Performed.");
+ public static GUIContent depthWrite = new GUIContent("Depth Write", "Controls Whether Pixels From This Material Are Written to the Depth Buffer");
+ public static GUIContent depthOffsetFactor = new GUIContent("Depth Offset Factor", "Scales the Maximum Z Slope, with Respect to X or Y of the Polygon");
+ public static GUIContent depthOffsetUnits = new GUIContent("Depth Offset Units", "Scales the Minimum Resolvable Depth Buffer Value");
+ public static GUIContent colorWriteMask = new GUIContent("Color Write Mask", "Color Channel Writing Mask");
+ public static GUIContent cullMode = new GUIContent("Cull Mode", "Triangle Culling Mode");
+ public static GUIContent renderQueueOverride = new GUIContent("Render Queue Override", "Manually Override the Render Queue");
+ }
+
+ protected bool initialized;
+
+ protected MaterialProperty renderingMode;
+ protected MaterialProperty customRenderingMode;
+ protected MaterialProperty sourceBlend;
+ protected MaterialProperty destinationBlend;
+ protected MaterialProperty blendOperation;
+ protected MaterialProperty depthTest;
+ protected MaterialProperty depthWrite;
+ protected MaterialProperty depthOffsetFactor;
+ protected MaterialProperty depthOffsetUnits;
+ protected MaterialProperty colorWriteMask;
+ protected MaterialProperty cullMode;
+ protected MaterialProperty renderQueueOverride;
+
+ protected const string LegacyShadersPath = "Legacy Shaders/";
+ protected const string TransparentShadersPath = "/Transparent/";
+ protected const string TransparentCutoutShadersPath = "/Transparent/Cutout/";
+
+ public override void OnGUI(MaterialEditor materialEditor, MaterialProperty[] props)
+ {
+ Material material = (Material)materialEditor.target;
+
+ FindProperties(props);
+ Initialize(material);
+
+ RenderingModeOptions(materialEditor);
+ }
+
+ protected virtual void FindProperties(MaterialProperty[] props)
+ {
+ renderingMode = FindProperty(BaseStyles.renderingModeName, props);
+ customRenderingMode = FindProperty(BaseStyles.customRenderingModeName, props);
+ sourceBlend = FindProperty(BaseStyles.sourceBlendName, props);
+ destinationBlend = FindProperty(BaseStyles.destinationBlendName, props);
+ blendOperation = FindProperty(BaseStyles.blendOperationName, props);
+ depthTest = FindProperty(BaseStyles.depthTestName, props);
+ depthWrite = FindProperty(BaseStyles.depthWriteName, props);
+ depthOffsetFactor = FindProperty(BaseStyles.depthOffsetFactorName, props);
+ depthOffsetUnits = FindProperty(BaseStyles.depthOffsetUnitsName, props);
+ colorWriteMask = FindProperty(BaseStyles.colorWriteMaskName, props);
+
+ cullMode = FindProperty(BaseStyles.cullModeName, props);
+ renderQueueOverride = FindProperty(BaseStyles.renderQueueOverrideName, props);
+ }
+
+ protected void Initialize(Material material)
+ {
+ if (!initialized)
+ {
+ MaterialChanged(material);
+ initialized = true;
+ }
+ }
+
+ protected virtual void MaterialChanged(Material material)
+ {
+ SetupMaterialWithRenderingMode(material,
+ (RenderingMode)renderingMode.floatValue,
+ (CustomRenderingMode)customRenderingMode.floatValue,
+ (int)renderQueueOverride.floatValue);
+ }
+
+ protected void RenderingModeOptions(MaterialEditor materialEditor)
+ {
+ EditorGUI.BeginChangeCheck();
+
+ EditorGUI.showMixedValue = renderingMode.hasMixedValue;
+ RenderingMode mode = (RenderingMode)renderingMode.floatValue;
+ EditorGUI.BeginChangeCheck();
+ mode = (RenderingMode)EditorGUILayout.Popup(renderingMode.displayName, (int)mode, BaseStyles.renderingModeNames);
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ materialEditor.RegisterPropertyChangeUndo(renderingMode.displayName);
+ renderingMode.floatValue = (float)mode;
+
+ Object[] targets = renderingMode.targets;
+
+ foreach (Object target in targets)
+ {
+ MaterialChanged((Material)target);
+ }
+ }
+
+ EditorGUI.showMixedValue = false;
+
+ if ((RenderingMode)renderingMode.floatValue == RenderingMode.Custom)
+ {
+ EditorGUI.indentLevel += 2;
+ customRenderingMode.floatValue = EditorGUILayout.Popup(customRenderingMode.displayName, (int)customRenderingMode.floatValue, BaseStyles.customRenderingModeNames);
+ materialEditor.ShaderProperty(sourceBlend, BaseStyles.sourceBlend);
+ materialEditor.ShaderProperty(destinationBlend, BaseStyles.destinationBlend);
+ materialEditor.ShaderProperty(blendOperation, BaseStyles.blendOperation);
+ materialEditor.ShaderProperty(depthTest, BaseStyles.depthTest);
+ depthWrite.floatValue = EditorGUILayout.Popup(depthWrite.displayName, (int)depthWrite.floatValue, BaseStyles.depthWriteNames);
+ materialEditor.ShaderProperty(depthOffsetFactor, BaseStyles.depthOffsetFactor);
+ materialEditor.ShaderProperty(depthOffsetUnits, BaseStyles.depthOffsetUnits);
+ materialEditor.ShaderProperty(colorWriteMask, BaseStyles.colorWriteMask);
+ EditorGUI.indentLevel -= 2;
+ }
+
+ if (!PropertyEnabled(depthWrite))
+ {
+ if (MixedRealityToolkitShaderGUIUtilities.DisplayDepthWriteWarning(materialEditor))
+ {
+ renderingMode.floatValue = (float)RenderingMode.Custom;
+ depthWrite.floatValue = (float)DepthWrite.On;
+ }
+ }
+
+ materialEditor.ShaderProperty(cullMode, BaseStyles.cullMode);
+ }
+
+ protected static void SetupMaterialWithRenderingMode(Material material, RenderingMode mode, CustomRenderingMode customMode, int renderQueueOverride)
+ {
+ // If we aren't switching to Custom, then set default values for all RenderingMode types. Otherwise keep whatever user had before
+ if (mode != RenderingMode.Custom)
+ {
+ material.SetInt(BaseStyles.blendOperationName, (int)BlendOp.Add);
+ material.SetInt(BaseStyles.depthTestName, (int)CompareFunction.LessEqual);
+ material.SetFloat(BaseStyles.depthOffsetFactorName, 0.0f);
+ material.SetFloat(BaseStyles.depthOffsetUnitsName, 0.0f);
+ material.SetInt(BaseStyles.colorWriteMaskName, (int)ColorWriteMask.All);
+ }
+
+ switch (mode)
+ {
+ case RenderingMode.Opaque:
+ {
+ material.SetOverrideTag(BaseStyles.renderTypeName, BaseStyles.renderingModeNames[(int)RenderingMode.Opaque]);
+ material.SetInt(BaseStyles.customRenderingModeName, (int)CustomRenderingMode.Opaque);
+ material.SetInt(BaseStyles.sourceBlendName, (int)BlendMode.One);
+ material.SetInt(BaseStyles.destinationBlendName, (int)BlendMode.Zero);
+ material.SetInt(BaseStyles.depthWriteName, (int)DepthWrite.On);
+ material.DisableKeyword(BaseStyles.alphaTestOnName);
+ material.DisableKeyword(BaseStyles.alphaBlendOnName);
+ material.renderQueue = (renderQueueOverride >= 0) ? renderQueueOverride : (int)RenderQueue.Geometry;
+ break;
+ }
+
+ case RenderingMode.Cutout:
+ {
+ material.SetOverrideTag(BaseStyles.renderTypeName, BaseStyles.renderingModeNames[(int)RenderingMode.Cutout]);
+ material.SetInt(BaseStyles.customRenderingModeName, (int)CustomRenderingMode.Cutout);
+ material.SetInt(BaseStyles.sourceBlendName, (int)BlendMode.One);
+ material.SetInt(BaseStyles.destinationBlendName, (int)BlendMode.Zero);
+ material.SetInt(BaseStyles.depthWriteName, (int)DepthWrite.On);
+ material.EnableKeyword(BaseStyles.alphaTestOnName);
+ material.DisableKeyword(BaseStyles.alphaBlendOnName);
+ material.renderQueue = (renderQueueOverride >= 0) ? renderQueueOverride : (int)RenderQueue.AlphaTest;
+ break;
+ }
+
+ case RenderingMode.Fade:
+ {
+ material.SetOverrideTag(BaseStyles.renderTypeName, BaseStyles.renderingModeNames[(int)RenderingMode.Fade]);
+ material.SetInt(BaseStyles.customRenderingModeName, (int)CustomRenderingMode.Fade);
+ material.SetInt(BaseStyles.sourceBlendName, (int)BlendMode.SrcAlpha);
+ material.SetInt(BaseStyles.destinationBlendName, (int)BlendMode.OneMinusSrcAlpha);
+ material.SetInt(BaseStyles.depthWriteName, (int)DepthWrite.Off);
+ material.DisableKeyword(BaseStyles.alphaTestOnName);
+ material.EnableKeyword(BaseStyles.alphaBlendOnName);
+ material.renderQueue = (renderQueueOverride >= 0) ? renderQueueOverride : (int)RenderQueue.Transparent;
+ break;
+ }
+
+ case RenderingMode.Transparent:
+ {
+ material.SetOverrideTag(BaseStyles.renderTypeName, BaseStyles.renderingModeNames[(int)RenderingMode.Fade]);
+ material.SetInt(BaseStyles.customRenderingModeName, (int)CustomRenderingMode.Fade);
+ material.SetInt(BaseStyles.sourceBlendName, (int)BlendMode.One);
+ material.SetInt(BaseStyles.destinationBlendName, (int)BlendMode.OneMinusSrcAlpha);
+ material.SetInt(BaseStyles.depthWriteName, (int)DepthWrite.Off);
+ material.DisableKeyword(BaseStyles.alphaTestOnName);
+ material.EnableKeyword(BaseStyles.alphaBlendOnName);
+ material.renderQueue = (renderQueueOverride >= 0) ? renderQueueOverride : (int)RenderQueue.Transparent;
+ break;
+ }
+
+ case RenderingMode.Additive:
+ {
+ material.SetOverrideTag(BaseStyles.renderTypeName, BaseStyles.renderingModeNames[(int)RenderingMode.Fade]);
+ material.SetInt(BaseStyles.customRenderingModeName, (int)CustomRenderingMode.Fade);
+ material.SetInt(BaseStyles.sourceBlendName, (int)BlendMode.One);
+ material.SetInt(BaseStyles.destinationBlendName, (int)BlendMode.One);
+ material.SetInt(BaseStyles.depthWriteName, (int)DepthWrite.Off);
+ material.DisableKeyword(BaseStyles.alphaTestOnName);
+ material.EnableKeyword(BaseStyles.alphaBlendOnName);
+ material.renderQueue = (renderQueueOverride >= 0) ? renderQueueOverride : (int)RenderQueue.Transparent;
+ break;
+ }
+
+ case RenderingMode.Custom:
+ {
+ material.SetOverrideTag(BaseStyles.renderTypeName, BaseStyles.customRenderingModeNames[(int)customMode]);
+ // _SrcBlend, _DstBlend, _BlendOp, _ZTest, _ZWrite, _ColorWriteMask are controlled by UI.
+
+ switch (customMode)
+ {
+ case CustomRenderingMode.Opaque:
+ {
+ material.DisableKeyword(BaseStyles.alphaTestOnName);
+ material.DisableKeyword(BaseStyles.alphaBlendOnName);
+ break;
+ }
+
+ case CustomRenderingMode.Cutout:
+ {
+ material.EnableKeyword(BaseStyles.alphaTestOnName);
+ material.DisableKeyword(BaseStyles.alphaBlendOnName);
+ break;
+ }
+
+ case CustomRenderingMode.Fade:
+ {
+ material.DisableKeyword(BaseStyles.alphaTestOnName);
+ material.EnableKeyword(BaseStyles.alphaBlendOnName);
+ break;
+ }
+ }
+
+ material.renderQueue = (renderQueueOverride >= 0) ? renderQueueOverride : material.renderQueue;
+ break;
+ }
+ }
+ }
+
+ ///
+ /// Check whether shader feature is enabled
+ ///
+ /// float property to check against
+ /// false if 0.0f, true otherwise
+ protected static bool PropertyEnabled(MaterialProperty property)
+ {
+ return !property.floatValue.Equals(0.0f);
+ }
+
+ ///
+ /// Get the value of a given float property for a material
+ ///
+ /// material to check
+ /// name of property against material
+ /// if has property, then value of that property for current material, null otherwise
+ protected static float? GetFloatProperty(Material material, string propertyName)
+ {
+ if (material.HasProperty(propertyName))
+ {
+ return material.GetFloat(propertyName);
+ }
+
+ return null;
+ }
+
+ ///
+ /// Get the value of a given vector property for a material
+ ///
+ /// material to check
+ /// name of property against material
+ /// if has property, then value of that property for current material, null otherwise
+ protected static Vector4? GetVectorProperty(Material material, string propertyName)
+ {
+ if (material.HasProperty(propertyName))
+ {
+ return material.GetVector(propertyName);
+ }
+
+ return null;
+ }
+
+ ///
+ /// Get the value of a given color property for a material
+ ///
+ /// material to check
+ /// name of property against material
+ /// if has property, then value of that property for current material, null otherwise
+ protected static Color? GetColorProperty(Material material, string propertyName)
+ {
+ if (material.HasProperty(propertyName))
+ {
+ return material.GetColor(propertyName);
+ }
+
+ return null;
+ }
+
+ ///
+ /// Sets the shader feature controlled by keyword and property name parameters active or inactive
+ ///
+ /// Material to modify
+ /// Keyword of shader feature
+ /// Associated property name for shader feature
+ /// float to be treated as a boolean flag for setting shader feature active or inactive
+ protected static void SetShaderFeatureActive(Material material, string keywordName, string propertyName, float? propertyValue)
+ {
+ if (propertyValue.HasValue)
+ {
+ if (keywordName != null)
+ {
+ if (!propertyValue.Value.Equals(0.0f))
+ {
+ material.EnableKeyword(keywordName);
+ }
+ else
+ {
+ material.DisableKeyword(keywordName);
+ }
+ }
+
+ material.SetFloat(propertyName, propertyValue.Value);
+ }
+ }
+
+ ///
+ /// Sets vector property against associated material
+ ///
+ /// material to control
+ /// name of property to set
+ /// value of property to set
+ protected static void SetVectorProperty(Material material, string propertyName, Vector4? propertyValue)
+ {
+ if (propertyValue.HasValue)
+ {
+ material.SetVector(propertyName, propertyValue.Value);
+ }
+ }
+
+ ///
+ /// Set color property against associated material
+ ///
+ /// material to control
+ /// name of property to set
+ /// value of property to set
+ protected static void SetColorProperty(Material material, string propertyName, Color? propertyValue)
+ {
+ if (propertyValue.HasValue)
+ {
+ material.SetColor(propertyName, propertyValue.Value);
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityShaderGUI.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityShaderGUI.cs.meta
new file mode 100644
index 0000000..81d46c7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityShaderGUI.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 5c455a0029df0144a8d8bd9b27f781eb
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityStandardShaderGUI.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityStandardShaderGUI.cs
new file mode 100644
index 0000000..6877d25
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityStandardShaderGUI.cs
@@ -0,0 +1,853 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using System;
+using System.IO;
+using UnityEditor;
+using UnityEngine;
+using UnityEngine.Rendering;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom shader inspector for the "Mixed Reality Toolkit/Standard" shader.
+ ///
+ public class MixedRealityStandardShaderGUI : MixedRealityShaderGUI
+ {
+ protected enum AlbedoAlphaMode
+ {
+ Transparency,
+ Metallic,
+ Smoothness
+ }
+
+ protected static class Styles
+ {
+ public static string primaryMapsTitle = "Main Maps";
+ public static string renderingOptionsTitle = "Rendering Options";
+ public static string advancedOptionsTitle = "Advanced Options";
+ public static string fluentOptionsTitle = "Fluent Options";
+ public static string stencilComparisonName = "_StencilComparison";
+ public static string stencilOperationName = "_StencilOperation";
+ public static string disableAlbedoMapName = "_DISABLE_ALBEDO_MAP";
+ public static string albedoMapAlphaMetallicName = "_METALLIC_TEXTURE_ALBEDO_CHANNEL_A";
+ public static string albedoMapAlphaSmoothnessName = "_SMOOTHNESS_TEXTURE_ALBEDO_CHANNEL_A";
+ public static string propertiesComponentHelp = "Use the {0} component(s) to control {1} properties.";
+ public static readonly string[] albedoAlphaModeNames = Enum.GetNames(typeof(AlbedoAlphaMode));
+ public static GUIContent albedo = new GUIContent("Albedo", "Albedo (RGB) and Transparency (Alpha)");
+ public static GUIContent albedoAssignedAtRuntime = new GUIContent("Assigned at Runtime", "As an optimization albedo operations are disabled when no albedo texture is specified. If a albedo texture will be specified at runtime enable this option.");
+ public static GUIContent alphaCutoff = new GUIContent("Alpha Cutoff", "Threshold for Alpha Cutoff");
+ public static GUIContent metallic = new GUIContent("Metallic", "Metallic Value");
+ public static GUIContent smoothness = new GUIContent("Smoothness", "Smoothness Value");
+ public static GUIContent enableChannelMap = new GUIContent("Channel Map", "Enable Channel Map, a Channel Packing Texture That Follows Unity's Standard Channel Setup");
+ public static GUIContent channelMap = new GUIContent("Channel Map", "Metallic (Red), Occlusion (Green), Emission (Blue), Smoothness (Alpha)");
+ public static GUIContent enableNormalMap = new GUIContent("Normal Map", "Enable Normal Map");
+ public static GUIContent normalMap = new GUIContent("Normal Map");
+ public static GUIContent normalMapScale = new GUIContent("Scale", "Scales the Normal Map Normal");
+ public static GUIContent enableEmission = new GUIContent("Emission", "Enable Emission");
+ public static GUIContent emissiveColor = new GUIContent("Color");
+ public static GUIContent enableTriplanarMapping = new GUIContent("Triplanar Mapping", "Enable Triplanar Mapping, a technique which programmatically generates UV coordinates");
+ public static GUIContent enableSSAA = new GUIContent("Super Sample Anti-Aliasing", "Enable Super Sample Anti-Aliasing, a technique improves texture clarity at long distances");
+ public static GUIContent mipmapBias = new GUIContent("Mipmap Bias", "Degree to bias the mip map. A larger negative value reduces aliasing and improves clarity, but may decrease performance");
+ public static GUIContent enableLocalSpaceTriplanarMapping = new GUIContent("Local Space", "If True Triplanar Mapping is Calculated in Local Space");
+ public static GUIContent triplanarMappingBlendSharpness = new GUIContent("Blend Sharpness", "The Power of the Blend with the Normal");
+ public static GUIContent directionalLight = new GUIContent("Directional Light", "Affected by One Unity Directional Light");
+ public static GUIContent specularHighlights = new GUIContent("Specular Highlights", "Calculate Specular Highlights");
+ public static GUIContent sphericalHarmonics = new GUIContent("Spherical Harmonics", "Read From Spherical Harmonics Data for Ambient Light");
+ public static GUIContent reflections = new GUIContent("Reflections", "Calculate Glossy Reflections");
+ public static GUIContent refraction = new GUIContent("Refraction", "Calculate Refraction");
+ public static GUIContent refractiveIndex = new GUIContent("Refractive Index", "Ratio of Indices of Refraction at the Surface Interface");
+ public static GUIContent rimLight = new GUIContent("Rim Light", "Enable Rim (Fresnel) Lighting");
+ public static GUIContent rimColor = new GUIContent("Color", "Rim Highlight Color");
+ public static GUIContent rimPower = new GUIContent("Power", "Rim Highlight Saturation");
+ public static GUIContent vertexColors = new GUIContent("Vertex Colors", "Enable Vertex Color Tinting");
+ public static GUIContent vertexExtrusion = new GUIContent("Vertex Extrusion", "Enable Vertex Extrusion Along the Vertex Normal");
+ public static GUIContent vertexExtrusionValue = new GUIContent("Extrusion Value", "How Far to Extrude the Vertex Along the Vertex Normal");
+ public static GUIContent vertexExtrusionSmoothNormals = new GUIContent("Use Smooth Normals", "Should Vertex Extrusion use the Smooth Normals in UV3, or Default Normals");
+ public static GUIContent blendedClippingWidth = new GUIContent("Blended Clipping Width", "The Width of the Clipping Primitive Clip Fade Region on Non-Cutout Materials");
+ public static GUIContent clippingBorder = new GUIContent("Clipping Border", "Enable a Border Along the Clipping Primitive's Edge");
+ public static GUIContent clippingBorderWidth = new GUIContent("Width", "Width of the Clipping Border");
+ public static GUIContent clippingBorderColor = new GUIContent("Color", "Interpolated Color of the Clipping Border");
+ public static GUIContent nearPlaneFade = new GUIContent("Near Fade", "Objects Disappear (Turn to Black/Transparent) as the Camera (or Hover/Proximity Light) Nears Them");
+ public static GUIContent nearLightFade = new GUIContent("Use Light", "A Hover or Proximity Light (Rather Than the Camera) Determines Near Fade Distance");
+ public static GUIContent fadeBeginDistance = new GUIContent("Fade Begin", "Distance From Camera (or Hover/Proximity Light) to Begin Fade In");
+ public static GUIContent fadeCompleteDistance = new GUIContent("Fade Complete", "Distance From Camera (or Hover/Proximity Light) When Fade is Fully In");
+ public static GUIContent fadeMinValue = new GUIContent("Fade Min Value", "Clamps the Fade Amount to a Minimum Value");
+ public static GUIContent hoverLight = new GUIContent("Hover Light", "Enable utilization of Hover Light(s)");
+ public static GUIContent enableHoverColorOverride = new GUIContent("Override Color", "Override Global Hover Light Color for this Material");
+ public static GUIContent hoverColorOverride = new GUIContent("Color", "Override Hover Light Color");
+ public static GUIContent proximityLight = new GUIContent("Proximity Light", "Enable utilization of Proximity Light(s)");
+ public static GUIContent enableProximityLightColorOverride = new GUIContent("Override Color", "Override Global Proximity Light Color for this Material");
+ public static GUIContent proximityLightCenterColorOverride = new GUIContent("Center Color", "The Override Color of the ProximityLight Gradient at the Center (RGB) and (A) is Gradient Extent");
+ public static GUIContent proximityLightMiddleColorOverride = new GUIContent("Middle Color", "The Override Color of the ProximityLight Gradient at the Middle (RGB) and (A) is Gradient Extent");
+ public static GUIContent proximityLightOuterColorOverride = new GUIContent("Outer Color", "The Override Color of the ProximityLight Gradient at the Outer Edge (RGB) and (A) is Gradient Extent");
+ public static GUIContent proximityLightSubtractive = new GUIContent("Subtractive", "Proximity Lights Remove Light from a Surface, Used to Mimic a Shadow");
+ public static GUIContent proximityLightTwoSided = new GUIContent("Two Sided", "Proximity Lights Apply to Both Sides of a Surface");
+ public static GUIContent fluentLightIntensity = new GUIContent("Light Intensity", "Intensity Scaler for All Hover and Proximity Lights");
+ public static GUIContent roundCorners = new GUIContent("Round Corners", "(Assumes UVs Specify Borders of Surface, Works Best on Unity Cube, Quad, and Plane)");
+ public static GUIContent roundCornerRadius = new GUIContent("Unit Radius", "Rounded Rectangle Corner Unit Sphere Radius");
+ public static GUIContent roundCornersRadius = new GUIContent("Corners Radius", "UpLeft-UpRight-BottomRight-BottomLeft");
+ public static GUIContent roundCornerMargin = new GUIContent("Margin %", "Distance From Geometry Edge");
+ public static GUIContent independentCorners = new GUIContent("Independent Corners", "Manage each corner separately");
+ public static GUIContent borderLight = new GUIContent("Border Light", "Enable Border Lighting (Assumes UVs Specify Borders of Surface, Works Best on Unity Cube, Quad, and Plane)");
+ public static GUIContent borderLightUsesHoverColor = new GUIContent("Use Hover Color", "Border Color Comes From Hover Light Color Override");
+ public static GUIContent borderLightReplacesAlbedo = new GUIContent("Replace Albedo", "Border Light Replaces Albedo (Replacement Rather Than Additive)");
+ public static GUIContent borderLightOpaque = new GUIContent("Opaque Borders", "Borders Override Alpha Value to Appear Opaque");
+ public static GUIContent borderWidth = new GUIContent("Width %", "Uniform Width Along Border as a % of the Smallest XYZ Dimension");
+ public static GUIContent borderMinValue = new GUIContent("Brightness", "Brightness Scaler");
+ public static GUIContent edgeSmoothingValue = new GUIContent("Edge Smoothing Value", "Smooths Edges When Round Corners and Transparency Is Enabled");
+ public static GUIContent borderLightOpaqueAlpha = new GUIContent("Alpha", "Alpha value of \"opaque\" borders.");
+ public static GUIContent innerGlow = new GUIContent("Inner Glow", "Enable Inner Glow (Assumes UVs Specify Borders of Surface, Works Best on Unity Cube, Quad, and Plane)");
+ public static GUIContent innerGlowColor = new GUIContent("Color", "Inner Glow Color (RGB) and Intensity (A)");
+ public static GUIContent innerGlowPower = new GUIContent("Power", "Power Exponent to Control Glow");
+ public static GUIContent iridescence = new GUIContent("Iridescence", "Simulated Iridescence via Albedo Changes with the Angle of Observation)");
+ public static GUIContent iridescentSpectrumMap = new GUIContent("Spectrum Map", "Spectrum of Colors to Apply (Usually a Texture with ROYGBIV from Left to Right)");
+ public static GUIContent iridescenceIntensity = new GUIContent("Intensity", "Intensity of Iridescence");
+ public static GUIContent iridescenceThreshold = new GUIContent("Threshold", "Threshold Window to Sample From the Spectrum Map");
+ public static GUIContent iridescenceAngle = new GUIContent("Angle", "Surface Angle");
+ public static GUIContent environmentColoring = new GUIContent("Environment Coloring", "Change Color Based on View");
+ public static GUIContent environmentColorThreshold = new GUIContent("Threshold", "Threshold When Environment Coloring Should Appear Based on Surface Normal");
+ public static GUIContent environmentColorIntensity = new GUIContent("Intensity", "Intensity (or Brightness) of the Environment Coloring");
+ public static GUIContent environmentColorX = new GUIContent("X-Axis Color", "Color Along the World Space X-Axis");
+ public static GUIContent environmentColorY = new GUIContent("Y-Axis Color", "Color Along the World Space Y-Axis");
+ public static GUIContent environmentColorZ = new GUIContent("Z-Axis Color", "Color Along the World Space Z-Axis");
+ public static GUIContent stencil = new GUIContent("Enable Stencil Testing", "Enabled Stencil Testing Operations");
+ public static GUIContent stencilReference = new GUIContent("Stencil Reference", "Value to Compared Against (if Comparison is Anything but Always) and/or the Value to be Written to the Buffer (if Either Pass, Fail or ZFail is Set to Replace)");
+ public static GUIContent stencilComparison = new GUIContent("Stencil Comparison", "Function to Compare the Reference Value to");
+ public static GUIContent stencilOperation = new GUIContent("Stencil Operation", "What to do When the Stencil Test Passes");
+ public static GUIContent ignoreZScale = new GUIContent("Ignore Z Scale", "For Features That Use Object Scale (Round Corners, Border Light, etc.), Ignore the Z Scale of the Object");
+ }
+
+ protected MaterialProperty albedoMap;
+ protected MaterialProperty albedoColor;
+ protected MaterialProperty albedoAlphaMode;
+ protected MaterialProperty albedoAssignedAtRuntime;
+ protected MaterialProperty alphaCutoff;
+ protected MaterialProperty enableChannelMap;
+ protected MaterialProperty channelMap;
+ protected MaterialProperty enableNormalMap;
+ protected MaterialProperty normalMap;
+ protected MaterialProperty normalMapScale;
+ protected MaterialProperty enableEmission;
+ protected MaterialProperty emissiveColor;
+ protected MaterialProperty enableTriplanarMapping;
+ protected MaterialProperty enableSSAA;
+ protected MaterialProperty mipmapBias;
+ protected MaterialProperty enableLocalSpaceTriplanarMapping;
+ protected MaterialProperty triplanarMappingBlendSharpness;
+ protected MaterialProperty metallic;
+ protected MaterialProperty smoothness;
+ protected MaterialProperty directionalLight;
+ protected MaterialProperty specularHighlights;
+ protected MaterialProperty sphericalHarmonics;
+ protected MaterialProperty reflections;
+ protected MaterialProperty refraction;
+ protected MaterialProperty refractiveIndex;
+ protected MaterialProperty rimLight;
+ protected MaterialProperty rimColor;
+ protected MaterialProperty rimPower;
+ protected MaterialProperty vertexColors;
+ protected MaterialProperty vertexExtrusion;
+ protected MaterialProperty vertexExtrusionValue;
+ protected MaterialProperty vertexExtrusionSmoothNormals;
+ protected MaterialProperty blendedClippingWidth;
+ protected MaterialProperty clippingBorder;
+ protected MaterialProperty clippingBorderWidth;
+ protected MaterialProperty clippingBorderColor;
+ protected MaterialProperty nearPlaneFade;
+ protected MaterialProperty nearLightFade;
+ protected MaterialProperty fadeBeginDistance;
+ protected MaterialProperty fadeCompleteDistance;
+ protected MaterialProperty fadeMinValue;
+ protected MaterialProperty hoverLight;
+ protected MaterialProperty enableHoverColorOverride;
+ protected MaterialProperty hoverColorOverride;
+ protected MaterialProperty proximityLight;
+ protected MaterialProperty enableProximityLightColorOverride;
+ protected MaterialProperty proximityLightCenterColorOverride;
+ protected MaterialProperty proximityLightMiddleColorOverride;
+ protected MaterialProperty proximityLightOuterColorOverride;
+ protected MaterialProperty proximityLightSubtractive;
+ protected MaterialProperty proximityLightTwoSided;
+ protected MaterialProperty fluentLightIntensity;
+ protected MaterialProperty roundCorners;
+ protected MaterialProperty roundCornerRadius;
+ protected MaterialProperty roundCornerMargin;
+ protected MaterialProperty independentCorners;
+ protected MaterialProperty roundCornersRadius;
+ protected MaterialProperty borderLight;
+ protected MaterialProperty borderLightUsesHoverColor;
+ protected MaterialProperty borderLightReplacesAlbedo;
+ protected MaterialProperty borderLightOpaque;
+ protected MaterialProperty borderWidth;
+ protected MaterialProperty borderMinValue;
+ protected MaterialProperty edgeSmoothingValue;
+ protected MaterialProperty borderLightOpaqueAlpha;
+ protected MaterialProperty innerGlow;
+ protected MaterialProperty innerGlowColor;
+ protected MaterialProperty innerGlowPower;
+ protected MaterialProperty iridescence;
+ protected MaterialProperty iridescentSpectrumMap;
+ protected MaterialProperty iridescenceIntensity;
+ protected MaterialProperty iridescenceThreshold;
+ protected MaterialProperty iridescenceAngle;
+ protected MaterialProperty environmentColoring;
+ protected MaterialProperty environmentColorThreshold;
+ protected MaterialProperty environmentColorIntensity;
+ protected MaterialProperty environmentColorX;
+ protected MaterialProperty environmentColorY;
+ protected MaterialProperty environmentColorZ;
+ protected MaterialProperty stencil;
+ protected MaterialProperty stencilReference;
+ protected MaterialProperty stencilComparison;
+ protected MaterialProperty stencilOperation;
+ protected MaterialProperty ignoreZScale;
+
+ protected override void FindProperties(MaterialProperty[] props)
+ {
+ base.FindProperties(props);
+
+ albedoMap = FindProperty("_MainTex", props);
+ albedoColor = FindProperty("_Color", props);
+ albedoAlphaMode = FindProperty("_AlbedoAlphaMode", props);
+ albedoAssignedAtRuntime = FindProperty("_AlbedoAssignedAtRuntime", props);
+ alphaCutoff = FindProperty("_Cutoff", props);
+ metallic = FindProperty("_Metallic", props);
+ smoothness = FindProperty("_Smoothness", props);
+ enableChannelMap = FindProperty("_EnableChannelMap", props);
+ channelMap = FindProperty("_ChannelMap", props);
+ enableNormalMap = FindProperty("_EnableNormalMap", props);
+ normalMap = FindProperty("_NormalMap", props);
+ normalMapScale = FindProperty("_NormalMapScale", props);
+ enableEmission = FindProperty("_EnableEmission", props);
+ emissiveColor = FindProperty("_EmissiveColor", props);
+ enableTriplanarMapping = FindProperty("_EnableTriplanarMapping", props);
+ enableSSAA = FindProperty("_EnableSSAA", props);
+ mipmapBias = FindProperty("_MipmapBias", props);
+ enableLocalSpaceTriplanarMapping = FindProperty("_EnableLocalSpaceTriplanarMapping", props);
+ triplanarMappingBlendSharpness = FindProperty("_TriplanarMappingBlendSharpness", props);
+ directionalLight = FindProperty("_DirectionalLight", props);
+ specularHighlights = FindProperty("_SpecularHighlights", props);
+ sphericalHarmonics = FindProperty("_SphericalHarmonics", props);
+ reflections = FindProperty("_Reflections", props);
+ refraction = FindProperty("_Refraction", props);
+ refractiveIndex = FindProperty("_RefractiveIndex", props);
+ rimLight = FindProperty("_RimLight", props);
+ rimColor = FindProperty("_RimColor", props);
+ rimPower = FindProperty("_RimPower", props);
+ vertexColors = FindProperty("_VertexColors", props);
+ vertexExtrusion = FindProperty("_VertexExtrusion", props);
+ vertexExtrusionValue = FindProperty("_VertexExtrusionValue", props);
+ vertexExtrusionSmoothNormals = FindProperty("_VertexExtrusionSmoothNormals", props);
+ blendedClippingWidth = FindProperty("_BlendedClippingWidth", props);
+ clippingBorder = FindProperty("_ClippingBorder", props);
+ clippingBorderWidth = FindProperty("_ClippingBorderWidth", props);
+ clippingBorderColor = FindProperty("_ClippingBorderColor", props);
+ nearPlaneFade = FindProperty("_NearPlaneFade", props);
+ nearLightFade = FindProperty("_NearLightFade", props);
+ fadeBeginDistance = FindProperty("_FadeBeginDistance", props);
+ fadeCompleteDistance = FindProperty("_FadeCompleteDistance", props);
+ fadeMinValue = FindProperty("_FadeMinValue", props);
+ hoverLight = FindProperty("_HoverLight", props);
+ enableHoverColorOverride = FindProperty("_EnableHoverColorOverride", props);
+ hoverColorOverride = FindProperty("_HoverColorOverride", props);
+ proximityLight = FindProperty("_ProximityLight", props);
+ enableProximityLightColorOverride = FindProperty("_EnableProximityLightColorOverride", props);
+ proximityLightCenterColorOverride = FindProperty("_ProximityLightCenterColorOverride", props);
+ proximityLightMiddleColorOverride = FindProperty("_ProximityLightMiddleColorOverride", props);
+ proximityLightOuterColorOverride = FindProperty("_ProximityLightOuterColorOverride", props);
+ proximityLightSubtractive = FindProperty("_ProximityLightSubtractive", props);
+ proximityLightTwoSided = FindProperty("_ProximityLightTwoSided", props);
+ fluentLightIntensity = FindProperty("_FluentLightIntensity", props);
+ roundCorners = FindProperty("_RoundCorners", props);
+ roundCornerRadius = FindProperty("_RoundCornerRadius", props);
+ roundCornersRadius = FindProperty("_RoundCornersRadius", props);
+ roundCornerMargin = FindProperty("_RoundCornerMargin", props);
+ independentCorners = FindProperty("_IndependentCorners", props);
+ borderLight = FindProperty("_BorderLight", props);
+ borderLightUsesHoverColor = FindProperty("_BorderLightUsesHoverColor", props);
+ borderLightReplacesAlbedo = FindProperty("_BorderLightReplacesAlbedo", props);
+ borderLightOpaque = FindProperty("_BorderLightOpaque", props);
+ borderWidth = FindProperty("_BorderWidth", props);
+ borderMinValue = FindProperty("_BorderMinValue", props);
+ edgeSmoothingValue = FindProperty("_EdgeSmoothingValue", props);
+ borderLightOpaqueAlpha = FindProperty("_BorderLightOpaqueAlpha", props);
+ innerGlow = FindProperty("_InnerGlow", props);
+ innerGlowColor = FindProperty("_InnerGlowColor", props);
+ innerGlowPower = FindProperty("_InnerGlowPower", props);
+ iridescence = FindProperty("_Iridescence", props);
+ iridescentSpectrumMap = FindProperty("_IridescentSpectrumMap", props);
+ iridescenceIntensity = FindProperty("_IridescenceIntensity", props);
+ iridescenceThreshold = FindProperty("_IridescenceThreshold", props);
+ iridescenceAngle = FindProperty("_IridescenceAngle", props);
+ environmentColoring = FindProperty("_EnvironmentColoring", props);
+ environmentColorThreshold = FindProperty("_EnvironmentColorThreshold", props);
+ environmentColorIntensity = FindProperty("_EnvironmentColorIntensity", props);
+ environmentColorX = FindProperty("_EnvironmentColorX", props);
+ environmentColorY = FindProperty("_EnvironmentColorY", props);
+ environmentColorZ = FindProperty("_EnvironmentColorZ", props);
+ stencil = FindProperty("_Stencil", props);
+ stencilReference = FindProperty("_StencilReference", props);
+ stencilComparison = FindProperty(Styles.stencilComparisonName, props);
+ stencilOperation = FindProperty(Styles.stencilOperationName, props);
+ ignoreZScale = FindProperty("_IgnoreZScale", props);
+ }
+
+ public override void OnGUI(MaterialEditor materialEditor, MaterialProperty[] props)
+ {
+ Material material = (Material)materialEditor.target;
+
+ base.OnGUI(materialEditor, props);
+
+ MainMapOptions(materialEditor, material);
+ RenderingOptions(materialEditor, material);
+ FluentOptions(materialEditor, material);
+ AdvancedOptions(materialEditor, material);
+ }
+
+ public override void AssignNewShaderToMaterial(Material material, Shader oldShader, Shader newShader)
+ {
+ // Cache old shader properties with potentially different names than the new shader.
+ float? smoothness = GetFloatProperty(material, "_Glossiness");
+ float? diffuse = GetFloatProperty(material, "_UseDiffuse");
+ float? specularHighlights = GetFloatProperty(material, "_SpecularHighlights");
+ float? normalMap = null;
+ Texture normalMapTexture = material.GetTexture("_BumpMap");
+ float? normalMapScale = GetFloatProperty(material, "_BumpScale");
+ float? emission = null;
+ Color? emissionColor = GetColorProperty(material, "_EmissionColor");
+ float? reflections = null;
+ float? rimLighting = null;
+ Vector4? textureScaleOffset = null;
+ float? cullMode = GetFloatProperty(material, "_Cull");
+
+ if (oldShader)
+ {
+ if (oldShader.name.Contains("Standard"))
+ {
+ normalMap = material.IsKeywordEnabled("_NORMALMAP") ? 1.0f : 0.0f;
+ emission = material.IsKeywordEnabled("_EMISSION") ? 1.0f : 0.0f;
+ reflections = GetFloatProperty(material, "_GlossyReflections");
+ }
+ else if (oldShader.name.Contains("Fast Configurable"))
+ {
+ normalMap = material.IsKeywordEnabled("_USEBUMPMAP_ON") ? 1.0f : 0.0f;
+ emission = GetFloatProperty(material, "_UseEmissionColor");
+ reflections = GetFloatProperty(material, "_UseReflections");
+ rimLighting = GetFloatProperty(material, "_UseRimLighting");
+ textureScaleOffset = GetVectorProperty(material, "_TextureScaleOffset");
+ }
+ }
+
+ base.AssignNewShaderToMaterial(material, oldShader, newShader);
+
+ // Apply old shader properties to the new shader.
+ SetShaderFeatureActive(material, null, "_Smoothness", smoothness);
+ SetShaderFeatureActive(material, "_DIRECTIONAL_LIGHT", "_DirectionalLight", diffuse);
+ SetShaderFeatureActive(material, "_SPECULAR_HIGHLIGHTS", "_SpecularHighlights", specularHighlights);
+ SetShaderFeatureActive(material, "_NORMAL_MAP", "_EnableNormalMap", normalMap);
+
+ if (normalMapTexture)
+ {
+ material.SetTexture("_NormalMap", normalMapTexture);
+ }
+
+ SetShaderFeatureActive(material, null, "_NormalMapScale", normalMapScale);
+ SetShaderFeatureActive(material, "_EMISSION", "_EnableEmission", emission);
+ SetColorProperty(material, "_EmissiveColor", emissionColor);
+ SetShaderFeatureActive(material, "_REFLECTIONS", "_Reflections", reflections);
+ SetShaderFeatureActive(material, "_RIM_LIGHT", "_RimLight", rimLighting);
+ SetVectorProperty(material, "_MainTex_ST", textureScaleOffset);
+ SetShaderFeatureActive(material, null, "_CullMode", cullMode);
+
+ // Setup the rendering mode based on the old shader.
+ if (oldShader == null || !oldShader.name.Contains(LegacyShadersPath))
+ {
+ SetupMaterialWithRenderingMode(material, (RenderingMode)material.GetFloat(BaseStyles.renderingModeName), CustomRenderingMode.Opaque, -1);
+ }
+ else
+ {
+ RenderingMode mode = RenderingMode.Opaque;
+
+ if (oldShader.name.Contains(TransparentCutoutShadersPath))
+ {
+ mode = RenderingMode.Cutout;
+ }
+ else if (oldShader.name.Contains(TransparentShadersPath))
+ {
+ mode = RenderingMode.Fade;
+ }
+
+ material.SetFloat(BaseStyles.renderingModeName, (float)mode);
+
+ MaterialChanged(material);
+ }
+ }
+
+ protected override void MaterialChanged(Material material)
+ {
+ SetupMaterialWithAlbedo(material, albedoMap, albedoAlphaMode, albedoAssignedAtRuntime);
+
+ base.MaterialChanged(material);
+ }
+
+ protected void MainMapOptions(MaterialEditor materialEditor, Material material)
+ {
+ GUILayout.Label(Styles.primaryMapsTitle, EditorStyles.boldLabel);
+
+ materialEditor.TexturePropertySingleLine(Styles.albedo, albedoMap, albedoColor);
+
+ if (albedoMap.textureValue == null)
+ {
+ materialEditor.ShaderProperty(albedoAssignedAtRuntime, Styles.albedoAssignedAtRuntime, 2);
+ }
+
+ materialEditor.ShaderProperty(enableChannelMap, Styles.enableChannelMap);
+
+ if (PropertyEnabled(enableChannelMap))
+ {
+ EditorGUI.indentLevel += 2;
+ materialEditor.TexturePropertySingleLine(Styles.channelMap, channelMap);
+ GUILayout.Box("Metallic (Red), Occlusion (Green), Emission (Blue), Smoothness (Alpha)", EditorStyles.helpBox, Array.Empty());
+ EditorGUI.indentLevel -= 2;
+ }
+
+ if (!PropertyEnabled(enableChannelMap))
+ {
+ EditorGUI.indentLevel += 2;
+
+ materialEditor.ShaderProperty(albedoAlphaMode, albedoAlphaMode.displayName);
+
+ if ((RenderingMode)renderingMode.floatValue == RenderingMode.Cutout ||
+ (RenderingMode)renderingMode.floatValue == RenderingMode.Custom)
+ {
+ materialEditor.ShaderProperty(alphaCutoff, Styles.alphaCutoff.text);
+ }
+
+ if ((AlbedoAlphaMode)albedoAlphaMode.floatValue != AlbedoAlphaMode.Metallic)
+ {
+ materialEditor.ShaderProperty(metallic, Styles.metallic);
+ }
+
+ if ((AlbedoAlphaMode)albedoAlphaMode.floatValue != AlbedoAlphaMode.Smoothness)
+ {
+ materialEditor.ShaderProperty(smoothness, Styles.smoothness);
+ }
+
+ SetupMaterialWithAlbedo(material, albedoMap, albedoAlphaMode, albedoAssignedAtRuntime);
+
+ EditorGUI.indentLevel -= 2;
+ }
+
+ if (PropertyEnabled(directionalLight) ||
+ PropertyEnabled(reflections) ||
+ PropertyEnabled(rimLight) ||
+ PropertyEnabled(environmentColoring))
+ {
+ materialEditor.ShaderProperty(enableNormalMap, Styles.enableNormalMap);
+
+ if (PropertyEnabled(enableNormalMap))
+ {
+ EditorGUI.indentLevel += 2;
+ materialEditor.TexturePropertySingleLine(Styles.normalMap, normalMap, normalMap.textureValue != null ? normalMapScale : null);
+ EditorGUI.indentLevel -= 2;
+ }
+ }
+
+ materialEditor.ShaderProperty(enableEmission, Styles.enableEmission);
+
+ if (PropertyEnabled(enableEmission))
+ {
+ materialEditor.ShaderProperty(emissiveColor, Styles.emissiveColor, 2);
+ }
+
+ GUI.enabled = !PropertyEnabled(enableSSAA);
+ materialEditor.ShaderProperty(enableTriplanarMapping, Styles.enableTriplanarMapping);
+
+ if (PropertyEnabled(enableTriplanarMapping))
+ {
+ materialEditor.ShaderProperty(enableLocalSpaceTriplanarMapping, Styles.enableLocalSpaceTriplanarMapping, 2);
+ materialEditor.ShaderProperty(triplanarMappingBlendSharpness, Styles.triplanarMappingBlendSharpness, 2);
+ }
+ GUI.enabled = true;
+
+ GUI.enabled = !PropertyEnabled(enableTriplanarMapping);
+ // SSAA implementation based off this article: https://medium.com/@bgolus/sharper-mipmapping-using-shader-based-supersampling-ed7aadb47bec
+ materialEditor.ShaderProperty(enableSSAA, Styles.enableSSAA);
+
+ if (PropertyEnabled(enableSSAA))
+ {
+ materialEditor.ShaderProperty(mipmapBias, Styles.mipmapBias, 2);
+ }
+ GUI.enabled = true;
+
+ EditorGUILayout.Space();
+ materialEditor.TextureScaleOffsetProperty(albedoMap);
+ }
+
+ protected void RenderingOptions(MaterialEditor materialEditor, Material material)
+ {
+ EditorGUILayout.Space();
+ GUILayout.Label(Styles.renderingOptionsTitle, EditorStyles.boldLabel);
+
+ materialEditor.ShaderProperty(directionalLight, Styles.directionalLight);
+
+ if (PropertyEnabled(directionalLight))
+ {
+ materialEditor.ShaderProperty(specularHighlights, Styles.specularHighlights, 2);
+ }
+
+ materialEditor.ShaderProperty(sphericalHarmonics, Styles.sphericalHarmonics);
+
+ materialEditor.ShaderProperty(reflections, Styles.reflections);
+
+ if (PropertyEnabled(reflections))
+ {
+ materialEditor.ShaderProperty(refraction, Styles.refraction, 2);
+
+ if (PropertyEnabled(refraction))
+ {
+ materialEditor.ShaderProperty(refractiveIndex, Styles.refractiveIndex, 4);
+ }
+ }
+
+ materialEditor.ShaderProperty(rimLight, Styles.rimLight);
+
+ if (PropertyEnabled(rimLight))
+ {
+ materialEditor.ShaderProperty(rimColor, Styles.rimColor, 2);
+ materialEditor.ShaderProperty(rimPower, Styles.rimPower, 2);
+ }
+
+ materialEditor.ShaderProperty(vertexColors, Styles.vertexColors);
+
+ materialEditor.ShaderProperty(vertexExtrusion, Styles.vertexExtrusion);
+
+ if (PropertyEnabled(vertexExtrusion))
+ {
+ materialEditor.ShaderProperty(vertexExtrusionValue, Styles.vertexExtrusionValue, 2);
+ materialEditor.ShaderProperty(vertexExtrusionSmoothNormals, Styles.vertexExtrusionSmoothNormals, 2);
+ }
+
+ if ((RenderingMode)renderingMode.floatValue != RenderingMode.Opaque &&
+ (RenderingMode)renderingMode.floatValue != RenderingMode.Cutout)
+ {
+ materialEditor.ShaderProperty(blendedClippingWidth, Styles.blendedClippingWidth);
+ GUILayout.Box(string.Format(Styles.propertiesComponentHelp, nameof(ClippingPrimitive), "other clipping"), EditorStyles.helpBox, Array.Empty());
+ }
+
+ materialEditor.ShaderProperty(clippingBorder, Styles.clippingBorder);
+
+ if (PropertyEnabled(clippingBorder))
+ {
+ materialEditor.ShaderProperty(clippingBorderWidth, Styles.clippingBorderWidth, 2);
+ materialEditor.ShaderProperty(clippingBorderColor, Styles.clippingBorderColor, 2);
+ GUILayout.Box(string.Format(Styles.propertiesComponentHelp, nameof(ClippingPrimitive), "other clipping"), EditorStyles.helpBox, Array.Empty());
+ }
+
+ materialEditor.ShaderProperty(nearPlaneFade, Styles.nearPlaneFade);
+
+ if (PropertyEnabled(nearPlaneFade))
+ {
+ materialEditor.ShaderProperty(nearLightFade, Styles.nearLightFade, 2);
+ materialEditor.ShaderProperty(fadeBeginDistance, Styles.fadeBeginDistance, 2);
+ materialEditor.ShaderProperty(fadeCompleteDistance, Styles.fadeCompleteDistance, 2);
+ materialEditor.ShaderProperty(fadeMinValue, Styles.fadeMinValue, 2);
+ }
+ }
+
+ protected void FluentOptions(MaterialEditor materialEditor, Material material)
+ {
+ EditorGUILayout.Space();
+ GUILayout.Label(Styles.fluentOptionsTitle, EditorStyles.boldLabel);
+ RenderingMode mode = (RenderingMode)renderingMode.floatValue;
+ CustomRenderingMode customMode = (CustomRenderingMode)customRenderingMode.floatValue;
+
+ materialEditor.ShaderProperty(hoverLight, Styles.hoverLight);
+
+ if (PropertyEnabled(hoverLight))
+ {
+ GUILayout.Box(string.Format(Styles.propertiesComponentHelp, nameof(HoverLight), Styles.hoverLight.text), EditorStyles.helpBox, Array.Empty());
+
+ materialEditor.ShaderProperty(enableHoverColorOverride, Styles.enableHoverColorOverride, 2);
+
+ if (PropertyEnabled(enableHoverColorOverride))
+ {
+ materialEditor.ShaderProperty(hoverColorOverride, Styles.hoverColorOverride, 4);
+ }
+ }
+
+ materialEditor.ShaderProperty(proximityLight, Styles.proximityLight);
+
+ if (PropertyEnabled(proximityLight))
+ {
+ materialEditor.ShaderProperty(enableProximityLightColorOverride, Styles.enableProximityLightColorOverride, 2);
+
+ if (PropertyEnabled(enableProximityLightColorOverride))
+ {
+ materialEditor.ShaderProperty(proximityLightCenterColorOverride, Styles.proximityLightCenterColorOverride, 4);
+ materialEditor.ShaderProperty(proximityLightMiddleColorOverride, Styles.proximityLightMiddleColorOverride, 4);
+ materialEditor.ShaderProperty(proximityLightOuterColorOverride, Styles.proximityLightOuterColorOverride, 4);
+ }
+
+ materialEditor.ShaderProperty(proximityLightSubtractive, Styles.proximityLightSubtractive, 2);
+ materialEditor.ShaderProperty(proximityLightTwoSided, Styles.proximityLightTwoSided, 2);
+ GUILayout.Box(string.Format(Styles.propertiesComponentHelp, nameof(ProximityLight), Styles.proximityLight.text), EditorStyles.helpBox, Array.Empty());
+ }
+
+ materialEditor.ShaderProperty(borderLight, Styles.borderLight);
+
+ if (PropertyEnabled(borderLight))
+ {
+ materialEditor.ShaderProperty(borderWidth, Styles.borderWidth, 2);
+
+ materialEditor.ShaderProperty(borderMinValue, Styles.borderMinValue, 2);
+
+ materialEditor.ShaderProperty(borderLightReplacesAlbedo, Styles.borderLightReplacesAlbedo, 2);
+
+ if (PropertyEnabled(hoverLight) && PropertyEnabled(enableHoverColorOverride))
+ {
+ materialEditor.ShaderProperty(borderLightUsesHoverColor, Styles.borderLightUsesHoverColor, 2);
+ }
+
+ if (mode == RenderingMode.Cutout || mode == RenderingMode.Fade || mode == RenderingMode.Transparent ||
+ (mode == RenderingMode.Custom && customMode == CustomRenderingMode.Cutout) ||
+ (mode == RenderingMode.Custom && customMode == CustomRenderingMode.Fade))
+ {
+ materialEditor.ShaderProperty(borderLightOpaque, Styles.borderLightOpaque, 2);
+
+ if (PropertyEnabled(borderLightOpaque))
+ {
+ materialEditor.ShaderProperty(borderLightOpaqueAlpha, Styles.borderLightOpaqueAlpha, 4);
+ }
+ }
+ }
+
+ if (PropertyEnabled(hoverLight) || PropertyEnabled(proximityLight) || PropertyEnabled(borderLight))
+ {
+ materialEditor.ShaderProperty(fluentLightIntensity, Styles.fluentLightIntensity);
+ }
+
+ materialEditor.ShaderProperty(roundCorners, Styles.roundCorners);
+
+ if (PropertyEnabled(roundCorners))
+ {
+ materialEditor.ShaderProperty(independentCorners, Styles.independentCorners, 2);
+ if (PropertyEnabled(independentCorners))
+ {
+ materialEditor.ShaderProperty(roundCornersRadius, Styles.roundCornersRadius, 2);
+ }
+ else
+ {
+ materialEditor.ShaderProperty(roundCornerRadius, Styles.roundCornerRadius, 2);
+ }
+
+ materialEditor.ShaderProperty(roundCornerMargin, Styles.roundCornerMargin, 2);
+ }
+
+ if (PropertyEnabled(roundCorners) || PropertyEnabled(borderLight))
+ {
+ materialEditor.ShaderProperty(edgeSmoothingValue, Styles.edgeSmoothingValue);
+ }
+
+ materialEditor.ShaderProperty(innerGlow, Styles.innerGlow);
+
+ if (PropertyEnabled(innerGlow))
+ {
+ materialEditor.ShaderProperty(innerGlowColor, Styles.innerGlowColor, 2);
+ materialEditor.ShaderProperty(innerGlowPower, Styles.innerGlowPower, 2);
+ }
+
+ materialEditor.ShaderProperty(iridescence, Styles.iridescence);
+
+ if (PropertyEnabled(iridescence))
+ {
+ EditorGUI.indentLevel += 2;
+ materialEditor.TexturePropertySingleLine(Styles.iridescentSpectrumMap, iridescentSpectrumMap);
+ EditorGUI.indentLevel -= 2;
+ materialEditor.ShaderProperty(iridescenceIntensity, Styles.iridescenceIntensity, 2);
+ materialEditor.ShaderProperty(iridescenceThreshold, Styles.iridescenceThreshold, 2);
+ materialEditor.ShaderProperty(iridescenceAngle, Styles.iridescenceAngle, 2);
+ }
+
+ materialEditor.ShaderProperty(environmentColoring, Styles.environmentColoring);
+
+ if (PropertyEnabled(environmentColoring))
+ {
+ materialEditor.ShaderProperty(environmentColorThreshold, Styles.environmentColorThreshold, 2);
+ materialEditor.ShaderProperty(environmentColorIntensity, Styles.environmentColorIntensity, 2);
+ materialEditor.ShaderProperty(environmentColorX, Styles.environmentColorX, 2);
+ materialEditor.ShaderProperty(environmentColorY, Styles.environmentColorY, 2);
+ materialEditor.ShaderProperty(environmentColorZ, Styles.environmentColorZ, 2);
+ }
+ }
+
+ protected void AdvancedOptions(MaterialEditor materialEditor, Material material)
+ {
+ EditorGUILayout.Space();
+ GUILayout.Label(Styles.advancedOptionsTitle, EditorStyles.boldLabel);
+
+ EditorGUI.BeginChangeCheck();
+
+ materialEditor.ShaderProperty(renderQueueOverride, BaseStyles.renderQueueOverride);
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ MaterialChanged(material);
+ }
+
+ // Show the RenderQueueField but do not allow users to directly manipulate it. That is done via the renderQueueOverride.
+ GUI.enabled = false;
+ materialEditor.RenderQueueField();
+
+ // Enable instancing to disable batching. Static and dynamic batching will normalize the object scale, which breaks
+ // features which utilize object scale.
+ GUI.enabled = !ScaleRequired();
+
+ if (!GUI.enabled && !material.enableInstancing)
+ {
+ material.enableInstancing = true;
+ }
+
+ materialEditor.EnableInstancingField();
+
+ GUI.enabled = true;
+
+ materialEditor.ShaderProperty(stencil, Styles.stencil);
+
+ if (PropertyEnabled(stencil))
+ {
+ materialEditor.ShaderProperty(stencilReference, Styles.stencilReference, 2);
+ materialEditor.ShaderProperty(stencilComparison, Styles.stencilComparison, 2);
+ materialEditor.ShaderProperty(stencilOperation, Styles.stencilOperation, 2);
+ }
+ else
+ {
+ // When stencil is disable, revert to the default stencil operations. Note, when tested on D3D11 hardware the stencil state
+ // is still set even when the CompareFunction.Disabled is selected, but this does not seem to affect performance.
+ material.SetInt(Styles.stencilComparisonName, (int)CompareFunction.Disabled);
+ material.SetInt(Styles.stencilOperationName, (int)StencilOp.Keep);
+ }
+
+ if (ScaleRequired())
+ {
+ materialEditor.ShaderProperty(ignoreZScale, Styles.ignoreZScale);
+ }
+ }
+
+ protected bool ScaleRequired()
+ {
+ return PropertyEnabled(vertexExtrusion) ||
+ PropertyEnabled(roundCorners) ||
+ PropertyEnabled(borderLight) ||
+ (PropertyEnabled(enableTriplanarMapping) && PropertyEnabled(enableLocalSpaceTriplanarMapping));
+ }
+
+ protected static void SetupMaterialWithAlbedo(Material material, MaterialProperty albedoMap, MaterialProperty albedoAlphaMode, MaterialProperty albedoAssignedAtRuntime)
+ {
+ if (albedoMap.textureValue || PropertyEnabled(albedoAssignedAtRuntime))
+ {
+ material.DisableKeyword(Styles.disableAlbedoMapName);
+ }
+ else
+ {
+ material.EnableKeyword(Styles.disableAlbedoMapName);
+ }
+
+ switch ((AlbedoAlphaMode)albedoAlphaMode.floatValue)
+ {
+ case AlbedoAlphaMode.Transparency:
+ {
+ material.DisableKeyword(Styles.albedoMapAlphaMetallicName);
+ material.DisableKeyword(Styles.albedoMapAlphaSmoothnessName);
+ break;
+ }
+
+ case AlbedoAlphaMode.Metallic:
+ {
+ material.EnableKeyword(Styles.albedoMapAlphaMetallicName);
+ material.DisableKeyword(Styles.albedoMapAlphaSmoothnessName);
+ break;
+ }
+
+ case AlbedoAlphaMode.Smoothness:
+ {
+ material.DisableKeyword(Styles.albedoMapAlphaMetallicName);
+ material.EnableKeyword(Styles.albedoMapAlphaSmoothnessName);
+ break;
+ }
+ }
+ }
+
+#if UNITY_2019_1_OR_NEWER
+ [MenuItem("Mixed Reality/Toolkit/Utilities/Upgrade MRTK Standard Shader for Universal Render Pipeline")]
+#else
+ [MenuItem("Mixed Reality/Toolkit/Utilities/Upgrade MRTK Standard Shader for Lightweight Render Pipeline")]
+#endif
+ protected static void UpgradeShaderForUniversalRenderPipeline()
+ {
+ string confirmationMessage;
+#if UNITY_2019_1_OR_NEWER
+ confirmationMessage = "This will alter the MRTK Standard Shader for use with Unity's Universal Render Pipeline. You cannot undo this action.";
+#else
+ confirmationMessage = "This will alter the MRTK Standard Shader for use with Unity's Lightweight Render Pipeline. You cannot undo this action.";
+#endif
+
+ if (EditorUtility.DisplayDialog("Upgrade MRTK Standard Shader?",
+ confirmationMessage,
+ "Ok",
+ "Cancel"))
+ {
+ string path = AssetDatabase.GetAssetPath(StandardShaderUtility.MrtkStandardShader);
+
+ if (!string.IsNullOrEmpty(path))
+ {
+ try
+ {
+ string upgradedShader = File.ReadAllText(path);
+
+#if UNITY_2019_1_OR_NEWER
+ upgradedShader = upgradedShader.Replace("Tags{ \"RenderType\" = \"Opaque\" \"LightMode\" = \"ForwardBase\" }",
+ "Tags{ \"RenderType\" = \"Opaque\" \"LightMode\" = \"UniversalForward\" }");
+#else
+ upgradedShader = upgradedShader.Replace("Tags{ \"RenderType\" = \"Opaque\" \"LightMode\" = \"ForwardBase\" }",
+ "Tags{ \"RenderType\" = \"Opaque\" \"LightMode\" = \"LightweightForward\" }");
+#endif
+
+ upgradedShader = upgradedShader.Replace("//#define _RENDER_PIPELINE",
+ "#define _RENDER_PIPELINE");
+
+ File.WriteAllText(path, upgradedShader);
+ AssetDatabase.Refresh();
+
+#if UNITY_2019_1_OR_NEWER
+ Debug.LogFormat("Upgraded {0} for use with the Universal Render Pipeline.", path);
+#else
+ Debug.LogFormat("Upgraded {0} for use with the Lightweight Render Pipeline.", path);
+#endif
+ }
+ catch (Exception e)
+ {
+ Debug.LogException(e);
+ }
+ }
+ else
+ {
+ Debug.LogErrorFormat("Failed to get asset path to: {0}", StandardShaderUtility.MrtkStandardShaderName);
+ }
+ }
+ }
+
+#if UNITY_2019_1_OR_NEWER
+ [MenuItem("Mixed Reality/Toolkit/Utilities/Upgrade MRTK Standard Shader for Universal Render Pipeline", true)]
+#else
+ [MenuItem("Mixed Reality/Toolkit/Utilities/Upgrade MRTK Standard Shader for Lightweight Render Pipeline", true)]
+#endif
+ protected static bool UpgradeShaderForUniversalRenderPipelineValidate()
+ {
+ // If a scriptable render pipeline is not present, no need to upgrade the shader.
+ return GraphicsSettings.renderPipelineAsset != null;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityStandardShaderGUI.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityStandardShaderGUI.cs.meta
new file mode 100644
index 0000000..47e5657
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityStandardShaderGUI.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 987d0f891c554d668f53217451de169b
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityTextMeshProShaderGUI.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityTextMeshProShaderGUI.cs
new file mode 100644
index 0000000..e627aad
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityTextMeshProShaderGUI.cs
@@ -0,0 +1,48 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using TMPro.EditorUtilities;
+using UnityEditor;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom TMP_SDFShaderGUI inspector for the "Mixed Reality Toolkit/TextMeshPro" shader.
+ /// Adds the ability to change the depth write mode, and a warning about depth write
+ /// when depth buffer sharing is enabled.
+ ///
+ public class MixedRealityTextMeshProShaderGUI : TMP_SDFShaderGUI
+ {
+ protected override void DoGUI()
+ {
+ BeginPanel("Mode", true);
+ DoModePanel();
+ EndPanel();
+
+ base.DoGUI();
+ }
+
+ protected void DoModePanel()
+ {
+ EditorGUI.indentLevel += 1;
+
+ var depthWrite = FindProperty("_ZWrite", m_Properties, false);
+
+ if (depthWrite != null)
+ {
+ m_Editor.ShaderProperty(depthWrite, depthWrite.displayName);
+
+ if (depthWrite.floatValue.Equals(0.0f))
+ {
+ if (MixedRealityToolkitShaderGUIUtilities.DisplayDepthWriteWarning(m_Editor))
+ {
+ depthWrite.floatValue = 1.0f;
+ }
+ }
+ }
+
+ EditorGUI.indentLevel -= 1;
+ EditorGUILayout.Space();
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityTextMeshProShaderGUI.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityTextMeshProShaderGUI.cs.meta
new file mode 100644
index 0000000..6f7617a
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityTextMeshProShaderGUI.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b322d4ab96d0da446be114e721d59c1c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitFacadeHandler.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitFacadeHandler.cs
new file mode 100644
index 0000000..b0f2b77
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitFacadeHandler.cs
@@ -0,0 +1,131 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using System.Collections.Generic;
+using UnityEditor;
+using UnityEditor.SceneManagement;
+using UnityEngine;
+using UnityEngine.SceneManagement;
+
+namespace Microsoft.MixedReality.Toolkit.Utilities.Facades
+{
+ ///
+ /// Links service facade objects to active services.
+ ///
+ ///
+ /// This feature is being deprecated in 2.5 and will be removed in a future release.
+ /// The code that remains will actively seek to remove existing facades in scenes to ensure that
+ /// developers that update to 2.5 will get their scenes cleaned up appropriately.
+ ///
+ [InitializeOnLoad]
+ public static class MixedRealityToolkitFacadeHandler
+ {
+ private static readonly List childrenToDelete = new List();
+
+ // While a scene save is occurring, facade creation is disabled
+ // and currently present facades get deleted.
+ private static bool sceneSaving = false;
+
+ static MixedRealityToolkitFacadeHandler()
+ {
+#if UNITY_2019_1_OR_NEWER
+ SceneView.duringSceneGui += OnSceneGUI;
+#else
+ SceneView.onSceneGUIDelegate += OnSceneGUI;
+#endif
+ EditorSceneManager.sceneSaving += OnSceneSaving;
+ EditorSceneManager.sceneSaved += OnSceneSaved;
+ }
+
+ #region callbacks
+
+ private static void OnSceneGUI(SceneView sceneView)
+ {
+ UpdateServiceFacades();
+ }
+
+ private static void OnSceneSaving(Scene scene, string path)
+ {
+ sceneSaving = true;
+ CleanupCurrentFacades();
+ }
+
+ private static void OnSceneSaved(Scene scene)
+ {
+ sceneSaving = false;
+ }
+
+ #endregion
+
+ private static void CleanupCurrentFacades()
+ {
+ foreach (MixedRealityToolkit toolkitInstance in GameObject.FindObjectsOfType())
+ {
+ DestroyAllChildren(toolkitInstance);
+ }
+ }
+
+ private static void UpdateServiceFacades()
+ {
+ // If compiling or saving, don't modify service facades
+ if (sceneSaving || EditorApplication.isCompiling)
+ {
+ return;
+ }
+
+ // If MRTK has no active instance
+ // or there is no active profile for the active instance
+ // or we are instructed to not use service inspectors
+ // Return early and clean up any facade instances
+ if (!MixedRealityToolkit.IsInitialized ||
+ !MixedRealityToolkit.Instance.HasActiveProfile ||
+#pragma warning disable 0618
+ !MixedRealityToolkit.Instance.ActiveProfile.UseServiceInspectors)
+#pragma warning restore 0618
+ {
+ DestroyFacades();
+ return;
+ }
+ }
+
+ private static void DestroyFacades()
+ {
+ for (int i = ServiceFacade.ActiveFacadeObjects.Count - 1; i >= 0; i--)
+ {
+ var facade = ServiceFacade.ActiveFacadeObjects[i];
+ if (facade != null)
+ {
+ GameObjectExtensions.DestroyGameObject(facade.gameObject);
+ }
+ }
+
+ ServiceFacade.ActiveFacadeObjects.Clear();
+ }
+
+ private static void DestroyAllChildren(MixedRealityToolkit instance)
+ {
+ Transform instanceTransform = instance.transform;
+
+ childrenToDelete.Clear();
+ foreach (Transform child in instanceTransform.transform)
+ {
+ childrenToDelete.Add(child);
+ }
+
+ foreach (ServiceFacade facade in ServiceFacade.ActiveFacadeObjects)
+ {
+ if (!childrenToDelete.Contains(facade.transform))
+ {
+ childrenToDelete.Add(facade.transform);
+ }
+ }
+
+ foreach (Transform child in childrenToDelete)
+ {
+ GameObjectExtensions.DestroyGameObject(child.gameObject);
+ }
+
+ childrenToDelete.Clear();
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitFacadeHandler.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitFacadeHandler.cs.meta
new file mode 100644
index 0000000..7b9e2f7
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitFacadeHandler.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 3e5187c03457caa448ff3226ff038bb8
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitHelpLinks.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitHelpLinks.cs
new file mode 100644
index 0000000..72389ac
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitHelpLinks.cs
@@ -0,0 +1,34 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// Creates menu items to show users how to get help
+ ///
+ public class MixedRealityToolkitHelpLinks : MonoBehaviour
+ {
+ internal const string MRTKIssuePageUrl = "https://github.com/microsoft/MixedRealityToolkit-Unity/issues";
+ internal const string MRTKDocsUrl = "https://aka.ms/mrtk2docs";
+ internal const string MRTKAPIRefUrl = "https://aka.ms/mrtk2api";
+
+ [MenuItem("Mixed Reality/Toolkit/Help/Show Documentation", false)]
+ private static void ShowDocumentation()
+ {
+ Application.OpenURL(MRTKDocsUrl);
+ }
+ [MenuItem("Mixed Reality/Toolkit/Help/Show API Reference", false)]
+ private static void ShowAPIReference()
+ {
+ Application.OpenURL(MRTKAPIRefUrl);
+ }
+ [MenuItem("Mixed Reality/Toolkit/Help/File a bug report", false)]
+ private static void FileBugReport()
+ {
+ Application.OpenURL(MRTKIssuePageUrl);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitHelpLinks.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitHelpLinks.cs.meta
new file mode 100644
index 0000000..83ab5bb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitHelpLinks.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b8ba0ad648d35da43bd56771a74f6cd4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitInspector.cs
new file mode 100644
index 0000000..7491646
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitInspector.cs
@@ -0,0 +1,111 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ [CustomEditor(typeof(MixedRealityToolkit))]
+ public class MixedRealityToolkitInspector : UnityEditor.Editor
+ {
+ private SerializedProperty activeProfile;
+ private UnityEditor.Editor activeProfileEditor;
+ private Object cachedProfile;
+
+ private void OnEnable()
+ {
+ activeProfile = serializedObject.FindProperty("activeProfile");
+ cachedProfile = activeProfile.objectReferenceValue;
+ }
+
+ public override void OnInspectorGUI()
+ {
+ MixedRealityToolkit instance = (MixedRealityToolkit)target;
+
+ if (MixedRealityToolkit.Instance == null && instance.isActiveAndEnabled)
+ { // See if an active instance exists at all. If it doesn't register this instance preemptively.
+ MixedRealityToolkit.SetActiveInstance(instance);
+ }
+
+ if (!instance.IsActiveInstance)
+ {
+ EditorGUILayout.HelpBox("This instance of the toolkit is inactive. There can only be one active instance loaded at any time.", MessageType.Warning);
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ if (GUILayout.Button("Select Active Instance"))
+ {
+ Selection.activeGameObject = MixedRealityToolkit.Instance.gameObject;
+ }
+
+ if (GUILayout.Button("Make this the Active Instance"))
+ {
+ MixedRealityToolkit.SetActiveInstance(instance);
+ }
+ }
+ return;
+ }
+
+ serializedObject.Update();
+
+ // If no profile is assigned, then warn user
+ if (activeProfile.objectReferenceValue == null)
+ {
+ EditorGUILayout.HelpBox("MixedRealityToolkit cannot initialize unless an Active Profile is assigned!", MessageType.Error);
+ }
+
+ bool changed = MixedRealityInspectorUtility.DrawProfileDropDownList(activeProfile, null, activeProfile.objectReferenceValue, typeof(MixedRealityToolkitConfigurationProfile), false, false) ||
+ cachedProfile != activeProfile.objectReferenceValue;
+
+ serializedObject.ApplyModifiedProperties();
+
+ if (changed)
+ {
+ TryResetConfiguration();
+ }
+
+ if (activeProfile.objectReferenceValue != null && activeProfileEditor == null)
+ {
+ // For the configuration profile, show the default inspector GUI
+ activeProfileEditor = CreateEditor(activeProfile.objectReferenceValue);
+ }
+
+ if (activeProfileEditor != null)
+ {
+ activeProfileEditor.OnInspectorGUI();
+ }
+ }
+
+ private void TryResetConfiguration()
+ {
+ var newProfile = (MixedRealityToolkitConfigurationProfile)activeProfile.objectReferenceValue;
+ try
+ {
+ if (!Application.isPlaying)
+ {
+ MixedRealityToolkit.Instance.ResetConfiguration(newProfile);
+ }
+ else
+ {
+ MixedRealityToolkit.Instance.ActiveProfile = newProfile;
+ }
+
+ activeProfileEditor = null;
+ cachedProfile = activeProfile.objectReferenceValue;
+ }
+ catch (System.Exception e)
+ {
+ Debug.LogError($"Failed to switch MRTK profile to {newProfile?.name}:\n{e}");
+ }
+ }
+
+ [MenuItem("Mixed Reality/Toolkit/Add to Scene and Configure...")]
+ public static void CreateMixedRealityToolkitGameObject()
+ {
+ MixedRealityInspectorUtility.AddMixedRealityToolkitToScene();
+ Selection.activeObject = MixedRealityToolkit.Instance;
+ EditorGUIUtility.PingObject(MixedRealityToolkit.Instance);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitInspector.cs.meta
new file mode 100644
index 0000000..dccc0c8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: ba2e1a978ccd4cba83b89f5ff458e977
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitShaderGUIUtilities.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitShaderGUIUtilities.cs
new file mode 100644
index 0000000..9f4b74c
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitShaderGUIUtilities.cs
@@ -0,0 +1,54 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A collection of shared functionality for MRTK shader GUIs.
+ ///
+ public static class MixedRealityToolkitShaderGUIUtilities
+ {
+ ///
+ /// GUI content styles which are common among shader GUIs.
+ ///
+ public static class Styles
+ {
+ public static readonly GUIContent DepthWriteWarning = new GUIContent("Warning: Depth buffer sharing is enabled for this project, but this material does not write depth. Enabling depth will improve reprojection, but may cause rendering artifacts in translucent materials.");
+ public static readonly GUIContent DepthWriteFixNowButton = new GUIContent("Fix Now", "Enables Depth Write For This Material");
+ }
+
+ ///
+ /// Displays a depth write warning and fix button if depth buffer sharing is enabled.
+ ///
+ /// The material editor to display the warning in.
+ /// The title of the dialog window to display when the user selects the fix button.
+ /// The message in the dialog window when the user selects the fix button.
+ /// True if the user opted to fix the warning, false otherwise.
+ public static bool DisplayDepthWriteWarning(MaterialEditor materialEditor, string dialogTitle = "Depth Write", string dialogMessage = "Change this material to write to the depth buffer?")
+ {
+ bool dialogConfirmed = false;
+
+ if (MixedRealityOptimizeUtils.IsDepthBufferSharingEnabled())
+ {
+ var defaultValue = EditorStyles.helpBox.richText;
+ EditorStyles.helpBox.richText = true;
+
+ if (materialEditor.HelpBoxWithButton(Styles.DepthWriteWarning, Styles.DepthWriteFixNowButton))
+ {
+ if (EditorUtility.DisplayDialog(dialogTitle, dialogMessage, "Yes", "No"))
+ {
+ dialogConfirmed = true;
+ }
+ }
+
+ EditorStyles.helpBox.richText = defaultValue;
+ }
+
+ return dialogConfirmed;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitShaderGUIUtilities.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitShaderGUIUtilities.cs.meta
new file mode 100644
index 0000000..3acf040
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityToolkitShaderGUIUtilities.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 97a49d6c607c32b449bb332cd755610d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityWireframeShaderGUI.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityWireframeShaderGUI.cs
new file mode 100644
index 0000000..caface0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityWireframeShaderGUI.cs
@@ -0,0 +1,104 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// A custom shader inspector for the "Mixed Reality Toolkit/Wireframe" shader.
+ ///
+ public class MixedRealityWireframeShaderGUI : MixedRealityShaderGUI
+ {
+ protected static class Styles
+ {
+ public static string mainPropertiesTitle = "Main Properties";
+ public static string advancedOptionsTitle = "Advanced Options";
+
+ public static GUIContent baseColor = new GUIContent("Base Color", "Color of faces");
+ public static GUIContent wireColor = new GUIContent("Wire Color", "Color of wires");
+ public static GUIContent wireThickness = new GUIContent("Wire Thickness", "Thickness of wires");
+ }
+
+ protected MaterialProperty baseColor;
+ protected MaterialProperty wireColor;
+ protected MaterialProperty wireThickness;
+
+ public override void OnGUI(MaterialEditor materialEditor, MaterialProperty[] props)
+ {
+ Material material = (Material)materialEditor.target;
+
+ base.OnGUI(materialEditor, props);
+
+ GUILayout.Label(Styles.mainPropertiesTitle, EditorStyles.boldLabel);
+ materialEditor.ShaderProperty(baseColor, Styles.baseColor);
+ materialEditor.ShaderProperty(wireColor, Styles.wireColor);
+ materialEditor.ShaderProperty(wireThickness, Styles.wireThickness);
+
+ AdvancedOptions(materialEditor, material);
+ }
+
+ protected override void FindProperties(MaterialProperty[] props)
+ {
+ base.FindProperties(props);
+
+ baseColor = FindProperty("_BaseColor", props);
+ wireColor = FindProperty("_WireColor", props);
+ wireThickness = FindProperty("_WireThickness", props);
+ }
+
+ public override void AssignNewShaderToMaterial(Material material, Shader oldShader, Shader newShader)
+ {
+ float? cullMode = GetFloatProperty(material, "_Cull");
+
+ base.AssignNewShaderToMaterial(material, oldShader, newShader);
+
+ SetShaderFeatureActive(material, null, BaseStyles.cullModeName, cullMode);
+
+ // Setup the rendering mode based on the old shader.
+ if (oldShader == null || !oldShader.name.Contains(LegacyShadersPath))
+ {
+ SetupMaterialWithRenderingMode(material, (RenderingMode)material.GetFloat(BaseStyles.renderingModeName), CustomRenderingMode.Opaque, -1);
+ }
+ else
+ {
+ RenderingMode mode = RenderingMode.Opaque;
+
+ if (oldShader.name.Contains(TransparentCutoutShadersPath))
+ {
+ mode = RenderingMode.Cutout;
+ }
+ else if (oldShader.name.Contains(TransparentShadersPath))
+ {
+ mode = RenderingMode.Fade;
+ }
+
+ material.SetFloat(BaseStyles.renderingModeName, (float)mode);
+
+ MaterialChanged(material);
+ }
+ }
+
+ protected void AdvancedOptions(MaterialEditor materialEditor, Material material)
+ {
+ GUILayout.Label(Styles.advancedOptionsTitle, EditorStyles.boldLabel);
+
+ EditorGUI.BeginChangeCheck();
+
+ materialEditor.ShaderProperty(renderQueueOverride, BaseStyles.renderQueueOverride);
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ MaterialChanged(material);
+ }
+
+ // Show the RenderQueueField but do not allow users to directly manipulate it. That is done via the renderQueueOverride.
+ GUI.enabled = false;
+ materialEditor.RenderQueueField();
+ GUI.enabled = true;
+
+ materialEditor.EnableInstancingField();
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityWireframeShaderGUI.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityWireframeShaderGUI.cs.meta
new file mode 100644
index 0000000..deccdda
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/MixedRealityWireframeShaderGUI.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: beab471bae7ba484d8c3a51dc9c3cbf4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles.meta
new file mode 100644
index 0000000..d466f8d
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles.meta
@@ -0,0 +1,8 @@
+fileFormatVersion: 2
+guid: 17f654b833624b21bdb7784f7d9a4ae8
+folderAsset: yes
+DefaultImporter:
+ externalObjects: {}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityProfileInspector.cs
new file mode 100644
index 0000000..b6f3358
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityProfileInspector.cs
@@ -0,0 +1,239 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System;
+using System.Linq;
+using System.Text;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// Base class for all Inspectors to inherit from.
+ ///
+ public abstract class BaseMixedRealityProfileInspector : UnityEditor.Editor
+ {
+ private static readonly StringBuilder dropdownKeyBuilder = new StringBuilder();
+
+ protected virtual void OnEnable()
+ {
+ if (target == null)
+ {
+ // Either when we are recompiling, or the inspector window is hidden behind another one, the target can get destroyed (null) and thereby will raise an ArgumentException when accessing serializedObject. For now, just return.
+ return;
+ }
+ }
+
+ ///
+ /// Renders a non-editable object field and an editable dropdown of a profile.
+ ///
+ public static void RenderReadOnlyProfile(SerializedProperty property)
+ {
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ EditorGUI.BeginDisabledGroup(true);
+ EditorGUILayout.ObjectField(property.objectReferenceValue != null ? "" : property.displayName, property.objectReferenceValue, typeof(BaseMixedRealityProfile), false, GUILayout.ExpandWidth(true));
+ EditorGUI.EndDisabledGroup();
+ }
+
+ if (property.objectReferenceValue != null)
+ {
+ bool showReadOnlyProfile = SessionState.GetBool(property.name + ".ReadOnlyProfile", false);
+
+ using (new EditorGUI.IndentLevelScope())
+ {
+ RenderFoldout(ref showReadOnlyProfile, property.displayName, () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ UnityEditor.Editor subProfileEditor = CreateEditor(property.objectReferenceValue);
+ // If this is a default MRTK configuration profile, ask it to render as a sub-profile
+ if (typeof(BaseMixedRealityToolkitConfigurationProfileInspector).IsAssignableFrom(subProfileEditor.GetType()))
+ {
+ BaseMixedRealityToolkitConfigurationProfileInspector configProfile = (BaseMixedRealityToolkitConfigurationProfileInspector)subProfileEditor;
+ configProfile.RenderAsSubProfile = true;
+ }
+ subProfileEditor.OnInspectorGUI();
+ }
+ });
+ }
+
+ SessionState.SetBool(property.name + ".ReadOnlyProfile", showReadOnlyProfile);
+ }
+ }
+
+ ///
+ /// Renders a .
+ ///
+ /// the property.
+ /// Profile type to filter available values to set on the provided property. If null, defaults to type
+ /// If true, draw the clone button, if false, don't
+ /// if true, render box around profile content, if false, don't
+ /// Optional service type to limit available profile types.
+ /// Optional parameter to used to specify that a profile must be selected
+ /// True, if the profile changed.
+ protected static bool RenderProfile(SerializedProperty property, Type profileType, bool showCloneButton = true, bool renderProfileInBox = false, Type serviceType = null, bool profileRequiredOverride = false)
+ {
+ return RenderProfileInternal(property, profileType, showCloneButton, renderProfileInBox, serviceType, profileRequiredOverride);
+ }
+
+ ///
+ /// Renders a .
+ ///
+ /// the property.
+ /// If true, draw the clone button, if false, don't
+ /// if true, render box around profile content, if false, don't
+ /// Optional service type to limit available profile types.
+ /// Optional parameter to used to specify that a profile must be selected
+ /// True, if the profile changed.
+ private static bool RenderProfileInternal(SerializedProperty property, Type profileType,
+ bool showCloneButton, bool renderProfileInBox, Type serviceType = null, bool profileRequiredOverride = false)
+ {
+ var profile = property.serializedObject.targetObject as BaseMixedRealityProfile;
+ bool changed = false;
+ var oldObject = property.objectReferenceValue;
+
+ if (profileType != null && !profileType.IsSubclassOf(typeof(BaseMixedRealityProfile)) && profileType != typeof(BaseMixedRealityProfile))
+ {
+ // If they've drag-and-dropped a non-profile scriptable object, set it to null.
+ profileType = null;
+ }
+
+ // If we're constraining this to a service type, check whether the profile is valid
+ // If it isn't, issue a warning.
+ if (serviceType != null && oldObject != null)
+ {
+ if (!MixedRealityProfileUtility.IsProfileForService(oldObject.GetType(), serviceType))
+ {
+ EditorGUILayout.HelpBox("This profile is not supported for " + serviceType.Name + ". Using an unsupported service may result in unexpected behavior.", MessageType.Warning);
+ }
+ }
+
+ Type[] profileTypes = new Type[] { };
+
+ bool requiresProfile = IsProfileRequired(serviceType) || profileRequiredOverride;
+ if (profileType == null)
+ {
+ // Find the profile type so we can limit the available object field options
+ if (serviceType != null)
+ {
+ // If GetProfileTypesForService has a count greater than one, then it won't be possible to use
+ // EditorGUILayout.ObjectField to restrict the set of profiles to a single type - in this
+ // case all profiles of BaseMixedRealityProfile will be visible in the picker.
+ //
+ // However in the case where there is just a single profile type for the service, we can improve
+ // upon the user experience by limiting the set of things that show in the picker by restricting
+ // the set of profiles listed to only that type.
+ profileTypes = MixedRealityProfileUtility.GetProfileTypesForService(serviceType).ToArray();
+ }
+ }
+ else
+ {
+ profileTypes = new Type[] { profileType };
+ }
+
+ // Draw the profile dropdown if a valid profileType exists
+ if (profileTypes.Length != 0)
+ {
+ changed |= MixedRealityInspectorUtility.DrawProfileDropDownList(property, profile, oldObject, profileTypes, requiresProfile, showCloneButton);
+ }
+ else if (requiresProfile)
+ {
+ EditorGUILayout.HelpBox("No ProfileType exists which is suitable for " + serviceType.Name + ". This service requires a profile to function properly!", MessageType.Error);
+ }
+
+ Debug.Assert(profile != null, "No profile was set in OnEnable. Did you forget to call base.OnEnable in a derived profile class?");
+
+ // Draw the sub-profile editor
+ MixedRealityInspectorUtility.DrawSubProfileEditor(property.objectReferenceValue, renderProfileInBox);
+
+ return changed;
+ }
+
+ ///
+ /// Render Bold/HelpBox style Foldout
+ ///
+ /// reference bool for current visibility state of foldout
+ /// Title in foldout
+ /// code to execute to render inside of foldout
+ /// optional argument, current show/hide state will be tracked associated with provided preference key
+ protected static void RenderFoldout(ref bool currentState, string title, Action renderContent, string preferenceKey = null)
+ {
+ EditorGUILayout.BeginVertical(EditorStyles.helpBox);
+
+ bool isValidPreferenceKey = !string.IsNullOrEmpty(preferenceKey);
+ bool state = currentState;
+ if (isValidPreferenceKey)
+ {
+ state = SessionState.GetBool(preferenceKey, currentState);
+ }
+
+ currentState = EditorGUILayout.Foldout(state, title, true, MixedRealityStylesUtility.BoldFoldoutStyle);
+
+ if (isValidPreferenceKey && currentState != state)
+ {
+ SessionState.SetBool(preferenceKey, currentState);
+ }
+
+ if (currentState)
+ {
+ renderContent();
+ }
+
+ EditorGUILayout.EndVertical();
+ }
+
+ private static string GetSubProfileDropdownKey(SerializedProperty property)
+ {
+ if (property.objectReferenceValue == null)
+ {
+ throw new Exception("Can't get sub profile dropdown key for a property that is null.");
+ }
+
+ dropdownKeyBuilder.Clear();
+ dropdownKeyBuilder.Append("MRTK_SubProfile_ShowDropdown_");
+ dropdownKeyBuilder.Append(property.name);
+ dropdownKeyBuilder.Append("_");
+ dropdownKeyBuilder.Append(property.objectReferenceValue.GetType().Name);
+ return dropdownKeyBuilder.ToString();
+ }
+
+ ///
+ /// Checks if the profile is locked
+ ///
+ protected static bool IsProfileLock(BaseMixedRealityProfile profile)
+ {
+ return MixedRealityProjectPreferences.LockProfiles && !profile.IsCustomProfile;
+ }
+
+ ///
+ /// Inspect the attributes of the provided system type to determine if a configuration profile is required.
+ ///
+ /// The system type representing the service.
+ ///
+ /// True if the service is decorated with an attribute indicating a profile is required, false otherwise.
+ ///
+ protected static bool IsProfileRequired(SystemType serviceType)
+ {
+ return IsProfileRequired(serviceType?.Type);
+ }
+
+ ///
+ /// Inspect the attributes of the provided type to determine if a configuration profile is required.
+ ///
+ /// The type representing the service.
+ ///
+ /// True if the type is decorated with an attribute indicating a profile is required, false otherwise.
+ ///
+ protected static bool IsProfileRequired(Type type)
+ {
+ // Services marked with the MixedRealityExtensionServiceAttribute (or a derivative)
+ // support specifying whether or not a profile is required.
+ MixedRealityExtensionServiceAttribute attribute = (type != null) ? MixedRealityExtensionServiceAttribute.Find(type) : null;
+ return attribute != null && attribute.RequiresProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityProfileInspector.cs.meta
new file mode 100644
index 0000000..0b6f46b
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 695870a3b55a4b5ca6885049aea1921f
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityToolkitConfigurationProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityToolkitConfigurationProfileInspector.cs
new file mode 100644
index 0000000..59a2f93
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityToolkitConfigurationProfileInspector.cs
@@ -0,0 +1,263 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor.Search;
+using System.Reflection;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// Base class for all Mixed Reality Toolkit specific inspectors to inherit from.
+ ///
+ public abstract class BaseMixedRealityToolkitConfigurationProfileInspector : BaseMixedRealityProfileInspector
+ {
+ public bool RenderAsSubProfile { get; set; }
+
+ private static GUIContent WarningIconContent = null;
+
+ ///
+ /// Helper function to determine if the current profile is assigned to the active instance of MRTK.
+ /// In some cases profile data refers to other profile data in the MRTK config profile.
+ /// In these cases, we don't want to render when the active instance isn't using this profile,
+ /// because it may produce an inaccurate combination of settings.
+ ///
+ protected abstract bool IsProfileInActiveInstance();
+
+ ///
+ /// Internal enum used for back navigation along profile hierarchy.
+ /// Indicates what type of parent profile the current profile will return to for going back
+ ///
+ protected enum BackProfileType
+ {
+ Configuration,
+ Input,
+ SpatialAwareness,
+ RegisteredServices
+ };
+
+ // NOTE: Must match number of elements in BackProfileType
+ protected readonly string[] BackProfileDescriptions = {
+ "Back to Configuration Profile",
+ "Back to Input Profile",
+ "Back to Spatial Awareness Profile",
+ "Back to Registered Service Providers Profile"
+ };
+
+ protected virtual void Awake()
+ {
+ if (WarningIconContent == null)
+ {
+ WarningIconContent = new GUIContent(EditorGUIUtility.IconContent("console.warnicon").image,
+ "This profile is part of the default set from the Mixed Reality Toolkit SDK. You can make a copy of this profile, and customize it if needed.");
+ }
+ }
+
+ ///
+ /// Render the Mixed Reality Toolkit Logo and search field.
+ ///
+ /// True if the rest of the inspector should be drawn.
+ protected bool RenderMRTKLogoAndSearch()
+ {
+ // If we're being rendered as a sub profile, don't show the logo
+ if (RenderAsSubProfile)
+ {
+ return true;
+ }
+
+ if (MixedRealitySearchInspectorUtility.DrawSearchInterface(target))
+ {
+ return false;
+ }
+
+ MixedRealityInspectorUtility.RenderMixedRealityToolkitLogo();
+ return true;
+ }
+
+ ///
+ /// Draws a documentation link for the service.
+ ///
+ protected void RenderDocumentation(Object profileObject)
+ {
+ if (profileObject == null)
+ { // Can't proceed if profile is null.
+ return;
+ }
+
+ HelpURLAttribute helpURL = profileObject.GetType().GetCustomAttribute();
+ if (helpURL != null)
+ {
+ InspectorUIUtility.RenderDocumentationButton(helpURL.URL);
+ }
+ }
+
+ protected bool DrawBacktrackProfileButton(BackProfileType returnProfileTarget = BackProfileType.Configuration)
+ {
+ // We cannot select the correct profile if there is no instance
+ if (!MixedRealityToolkit.IsInitialized)
+ {
+ return false;
+ }
+
+ string backText = BackProfileDescriptions[(int)returnProfileTarget];
+ BaseMixedRealityProfile backProfile = null;
+ switch (returnProfileTarget)
+ {
+ case BackProfileType.Configuration:
+ backProfile = MixedRealityToolkit.Instance.ActiveProfile;
+ break;
+ case BackProfileType.Input:
+ backProfile = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile;
+ break;
+ case BackProfileType.SpatialAwareness:
+ backProfile = MixedRealityToolkit.Instance.ActiveProfile.SpatialAwarenessSystemProfile;
+ break;
+ case BackProfileType.RegisteredServices:
+ backProfile = MixedRealityToolkit.Instance.ActiveProfile.RegisteredServiceProvidersProfile;
+ break;
+ }
+
+ return DrawBacktrackProfileButton(backText, backProfile);
+ }
+
+ ///
+ /// Renders a button that will take user back to a specified profile object
+ ///
+ /// True if button was clicked
+ protected bool DrawBacktrackProfileButton(string message, UnityEngine.Object activeObject)
+ {
+ // If we're being rendered as a sub profile, don't show the button
+ if (RenderAsSubProfile)
+ {
+ return false;
+ }
+
+ if (GUILayout.Button(message))
+ {
+ Selection.activeObject = activeObject;
+ return true;
+ }
+
+ return false;
+ }
+
+ ///
+ /// Helper function to render header correctly for all profiles
+ ///
+ /// Title of profile
+ /// profile tooltip describing purpose
+ /// The profile object. Used to re-select the object after MRTK instance is created.
+ /// profile properties are full initialized for rendering
+ /// Text for back button if not rendering as sub-profile
+ /// Target profile to return to if not rendering as sub-profile
+ /// True if the rest of the profile should be rendered.
+ protected bool RenderProfileHeader(string title, string description, Object selectionObject, bool isProfileInitialized = true, BackProfileType returnProfileTarget = BackProfileType.Configuration)
+ {
+ if (!RenderMRTKLogoAndSearch())
+ {
+ CheckEditorPlayMode();
+ return false;
+ }
+
+ var profile = target as BaseMixedRealityProfile;
+ if (!RenderAsSubProfile)
+ {
+ CheckEditorPlayMode();
+
+ if (!profile.IsCustomProfile)
+ {
+ EditorGUILayout.HelpBox("Default MRTK profiles cannot be edited. Create a clone of this profile to modify settings.", MessageType.Warning);
+ if (GUILayout.Button(new GUIContent("Clone")))
+ {
+ MixedRealityProfileCloneWindow.OpenWindow(null, (BaseMixedRealityProfile)target, null);
+ }
+ }
+
+ if (IsProfileInActiveInstance())
+ {
+ DrawBacktrackProfileButton(returnProfileTarget);
+ }
+
+ if (!isProfileInitialized)
+ {
+ if (!MixedRealityToolkit.IsInitialized)
+ {
+ EditorGUILayout.HelpBox("There is not a MRTK instance in your scene. Some properties may not be editable", MessageType.Error);
+ if (InspectorUIUtility.RenderIndentedButton(new GUIContent("Add Mixed Reality Toolkit instance to scene"), EditorStyles.miniButton))
+ {
+ MixedRealityInspectorUtility.AddMixedRealityToolkitToScene(MixedRealityInspectorUtility.GetDefaultConfigProfile());
+ // After the toolkit has been created, set the selection back to this item so the user doesn't get lost
+ Selection.activeObject = selectionObject;
+ }
+ }
+ else if (!MixedRealityToolkit.Instance.HasActiveProfile)
+ {
+ EditorGUILayout.HelpBox("There is no active profile assigned in the current MRTK instance. Some properties may not be editable.", MessageType.Error);
+ }
+ }
+ }
+ else
+ {
+ if (!isProfileInitialized && profile.IsCustomProfile)
+ {
+ EditorGUILayout.HelpBox("Some properties may not be editable in this profile. Please refer to the error messages below to resolve editing.", MessageType.Warning);
+ }
+ }
+
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ EditorGUILayout.LabelField(new GUIContent(title, description), EditorStyles.boldLabel, GUILayout.ExpandWidth(true));
+ RenderDocumentation(selectionObject);
+ }
+
+ EditorGUILayout.LabelField(string.Empty, GUI.skin.horizontalSlider);
+
+ return true;
+ }
+
+ ///
+ /// If application is playing, then show warning to the user and disable inspector GUI
+ ///
+ /// true if application is playing, false otherwise
+ protected bool CheckEditorPlayMode()
+ {
+ if (Application.isPlaying)
+ {
+ EditorGUILayout.HelpBox("Mixed Reality Toolkit settings cannot be edited while in play mode.", MessageType.Warning);
+ GUI.enabled = false;
+ return true;
+ }
+
+ return false;
+ }
+
+ ///
+ /// Check if various input settings are set correctly to read the input actions for the active MRTK instance. If any failures, show appropriate error message
+ ///
+ protected void CheckMixedRealityInputActions()
+ {
+ if (MixedRealityToolkit.IsInitialized && MixedRealityToolkit.Instance.HasActiveProfile)
+ {
+ if (!MixedRealityToolkit.Instance.ActiveProfile.IsInputSystemEnabled)
+ {
+ EditorGUILayout.HelpBox("No input system is enabled, or you need to specify the type in the main configuration profile.", MessageType.Warning);
+ }
+
+ if (MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile == null)
+ {
+ EditorGUILayout.HelpBox("No input system profile found, please specify an input system profile in the main configuration.", MessageType.Error);
+ }
+ else if (MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile == null)
+ {
+ EditorGUILayout.HelpBox("No input actions profile found, please specify an input action profile in the main configuration.", MessageType.Error);
+ }
+ else if (!IsProfileInActiveInstance())
+ {
+ EditorGUILayout.HelpBox("This profile is not assigned to the active MRTK instance in your scene. Some properties may not be editable", MessageType.Error);
+ }
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityToolkitConfigurationProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityToolkitConfigurationProfileInspector.cs.meta
new file mode 100644
index 0000000..5101567
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/BaseMixedRealityToolkitConfigurationProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 19c8b3dc140744b1bb892aaf02032169
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/DataProviderAccessServiceInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/DataProviderAccessServiceInspector.cs
new file mode 100644
index 0000000..b88b0cc
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/DataProviderAccessServiceInspector.cs
@@ -0,0 +1,273 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System;
+using System.Collections.Generic;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// Abstract class providing base functionality for data provider management in inspector. Useful for core systems that follow data provider access model.
+ /// Designed to target ScriptableObject profile classes that configure services who support data providers.
+ /// These profile ScriptableObject classes should contain an array of IMixedRealityServiceConfigurations that configure a list of data providers for this service configuration
+ ///
+ public abstract class BaseDataProviderServiceInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ ///
+ /// Container class used to store references to serialized properties on a target
+ ///
+ protected class ServiceConfigurationProperties
+ {
+ internal SerializedProperty componentName;
+ internal SerializedProperty componentType;
+ internal SerializedProperty providerProfile;
+ internal SerializedProperty runtimePlatform;
+ }
+
+ ///
+ /// Allows implementations of a IMixedRealityDataProviderAccess system's inspector to provide custom data provider representations for data providers.
+ ///
+ /// SerializedProperty object that wraps references to array of stored on the inspected target object.
+ protected abstract SerializedProperty GetDataProviderConfigurationList();
+
+ ///
+ /// Builds container object with SerializedProperty references to associated properties on the supplied reference
+ ///
+ /// SerializedProperty reference pointing to instance
+ protected abstract ServiceConfigurationProperties GetDataProviderConfigurationProperties(SerializedProperty providerEntry);
+
+ ///
+ /// Returns direct instance at provided index in target object's array of configurations
+ ///
+ protected abstract IMixedRealityServiceConfiguration GetDataProviderConfiguration(int index);
+
+ private SerializedProperty providerConfigurations;
+ private List providerFoldouts = new List();
+
+#if UNITY_2019
+ private static readonly GUIContent GeneralProvidersLabel = new GUIContent("General Providers");
+#endif // UNITY_2019
+
+ private readonly XRPipelineUtility xrPipelineUtility = new XRPipelineUtility();
+ private readonly List delayedDisplayProviders = new List();
+
+ private static readonly GUIContent ComponentTypeLabel = new GUIContent("Type");
+ private static readonly GUIContent SupportedPlatformsLabel = new GUIContent("Supported Platform(s)");
+
+ private const string NewDataProvider = "New data provider";
+
+ ///
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+#if UNITY_2019
+ xrPipelineUtility.Enable();
+#endif // UNITY_2019
+
+ providerConfigurations = GetDataProviderConfigurationList();
+
+ if (providerFoldouts == null || providerFoldouts.Count != providerConfigurations.arraySize)
+ {
+ providerFoldouts = new List(new bool[providerConfigurations.arraySize]);
+ }
+ }
+
+ ///
+ /// Adds a new data provider profile entry (i.e ) to array list of target object
+ /// Utilizes GetDataProviderConfigurationList() to get SerializedProperty object that represents array to insert against
+ ///
+ protected virtual void AddDataProvider()
+ {
+ providerConfigurations.InsertArrayElementAtIndex(providerConfigurations.arraySize);
+ SerializedProperty provider = providerConfigurations.GetArrayElementAtIndex(providerConfigurations.arraySize - 1);
+
+ ServiceConfigurationProperties providerProperties = GetDataProviderConfigurationProperties(provider);
+ providerProperties.componentName.stringValue = $"{NewDataProvider} {providerConfigurations.arraySize - 1}";
+ providerProperties.runtimePlatform.intValue = -1;
+ providerProperties.providerProfile.objectReferenceValue = null;
+
+ serializedObject.ApplyModifiedProperties();
+
+ SystemType providerType = GetDataProviderConfiguration(providerConfigurations.arraySize - 1).ComponentType;
+ providerType.Type = null;
+
+ providerFoldouts.Add(false);
+ }
+
+ ///
+ /// Removed given index item from array list.
+ /// Utilizes GetDataProviderConfigurationList() to get SerializedProperty object that represents array to delete against.
+ ///
+ protected virtual void RemoveDataProvider(int index)
+ {
+ providerConfigurations.DeleteArrayElementAtIndex(index);
+ serializedObject.ApplyModifiedProperties();
+
+ providerFoldouts.RemoveAt(index);
+ }
+
+ ///
+ /// Applies the given concrete data provider type properties to the provided instance (as represented by ).
+ /// Requires on concrete type class to pull initial values
+ /// that will be applied to the container SerializedProperties
+ ///
+ protected virtual void ApplyProviderConfiguration(Type dataProviderType, ServiceConfigurationProperties providerProperties)
+ {
+ if (dataProviderType != null)
+ {
+ if (MixedRealityExtensionServiceAttribute.Find(dataProviderType) is MixedRealityDataProviderAttribute providerAttribute)
+ {
+ providerProperties.componentName.stringValue = !string.IsNullOrWhiteSpace(providerAttribute.Name) ? providerAttribute.Name : dataProviderType.Name;
+ providerProperties.providerProfile.objectReferenceValue = providerAttribute.DefaultProfile;
+ providerProperties.runtimePlatform.intValue = (int)providerAttribute.RuntimePlatforms;
+ }
+ else
+ {
+ providerProperties.componentName.stringValue = dataProviderType.Name;
+ }
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ ///
+ /// Render list of data provider configuration profiles in inspector. Use provided add and remove content labels for the insert/remove buttons
+ /// Returns true if any property has changed in this render pass, false otherwise
+ ///
+ protected bool RenderDataProviderList(GUIContent addContentLabel, GUIContent removeContentLabel, string errorMsg, Type dataProviderProfileType = null)
+ {
+ bool changed = false;
+
+ using (new EditorGUILayout.VerticalScope())
+ {
+ if (providerConfigurations == null || providerConfigurations.arraySize == 0)
+ {
+ EditorGUILayout.HelpBox(errorMsg, MessageType.Info);
+ }
+
+ if (InspectorUIUtility.RenderIndentedButton(addContentLabel, EditorStyles.miniButton))
+ {
+ AddDataProvider();
+ return true;
+ }
+
+#if UNITY_2019
+ xrPipelineUtility.RenderXRPipelineTabs();
+#endif // UNITY_2019
+
+ delayedDisplayProviders.Clear();
+
+ for (int i = 0; i < providerConfigurations.arraySize; i++)
+ {
+ SystemType serviceType = GetDataProviderConfiguration(i).ComponentType;
+
+ if (serviceType.Type != null && MixedRealityExtensionServiceAttribute.Find(serviceType.Type) is MixedRealityDataProviderAttribute providerAttribute)
+ {
+ // Using == here to compare flags because we want to know if this is the only supported pipeline
+ // Providers that support multiple pipelines are rendered below the tabbed section
+ if (providerAttribute.SupportedUnityXRPipelines == xrPipelineUtility.SelectedPipeline)
+ {
+ changed |= RenderDataProviderEntry(i, removeContentLabel, serviceType, dataProviderProfileType);
+ delayedDisplayProviders.Add(null);
+ }
+ else if (providerAttribute.SupportedUnityXRPipelines == (SupportedUnityXRPipelines)(-1))
+ {
+ delayedDisplayProviders.Add(serviceType);
+ }
+ else
+ {
+ // Add null to ensure the delayedDisplayProviders list has an identical size to providerConfigurations.arraySize
+ // This is so we can iterate through without keeping track of i separately
+ delayedDisplayProviders.Add(null);
+ }
+ }
+ else
+ {
+ delayedDisplayProviders.Add(serviceType);
+ }
+ }
+
+#if UNITY_2019
+ EditorGUILayout.LabelField(GeneralProvidersLabel, EditorStyles.boldLabel);
+#endif // UNITY_2019
+
+ for (int i = 0; i < delayedDisplayProviders.Count; i++)
+ {
+ SystemType service = delayedDisplayProviders[i];
+ if (service != null)
+ {
+ changed |= RenderDataProviderEntry(i, removeContentLabel, service, dataProviderProfileType);
+ }
+ }
+
+ return changed;
+ }
+ }
+
+ ///
+ /// Renders properties of instance at provided index in inspector.
+ /// Also renders inspector view of data provider's profile object and its contents if applicable and foldout is expanded.
+ ///
+ private bool RenderDataProviderEntry(int index, GUIContent removeContent, SystemType serviceType, Type dataProviderProfileType = null)
+ {
+ bool changed = false;
+ SerializedProperty provider = providerConfigurations.GetArrayElementAtIndex(index);
+ ServiceConfigurationProperties providerProperties = GetDataProviderConfigurationProperties(provider);
+
+ // Don't hide new data providers added via the UI, otherwise there's no easy way to change their type
+ if (serviceType?.Type == null && !MixedRealityProjectPreferences.ShowNullDataProviders && !providerProperties.componentName.stringValue.StartsWith(NewDataProvider))
+ {
+ return false;
+ }
+
+ using (new EditorGUILayout.VerticalScope(EditorStyles.helpBox))
+ {
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ if (index < 0 || index >= providerFoldouts.Count) index = 0;
+ providerFoldouts[index] = EditorGUILayout.Foldout(providerFoldouts[index], providerProperties.componentName.stringValue, true);
+
+ if (GUILayout.Button(removeContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ RemoveDataProvider(index);
+ return true;
+ }
+ }
+
+ if (providerFoldouts[index])
+ {
+ using (var c = new EditorGUI.ChangeCheckScope())
+ {
+ EditorGUILayout.PropertyField(providerProperties.componentType, ComponentTypeLabel);
+ if (c.changed)
+ {
+ serializedObject.ApplyModifiedProperties();
+ ApplyProviderConfiguration(serviceType.Type, providerProperties);
+ return true;
+ }
+
+ EditorGUILayout.PropertyField(providerProperties.runtimePlatform, SupportedPlatformsLabel);
+ changed = c.changed;
+ }
+
+ changed |= RenderProfile(providerProperties.providerProfile, dataProviderProfileType, true, false, serviceType);
+
+ serializedObject.ApplyModifiedProperties();
+ }
+
+ if (IsProfileRequired(serviceType) &&
+ (providerProperties.providerProfile.objectReferenceValue == null))
+ {
+ EditorGUILayout.HelpBox($"{providerProperties.componentName.stringValue} requires a profile.", MessageType.Warning);
+ }
+ }
+
+ return changed;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/DataProviderAccessServiceInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/DataProviderAccessServiceInspector.cs.meta
new file mode 100644
index 0000000..96d87ac
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/DataProviderAccessServiceInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 9e4e9344957d87e4faa7aadf356d70f3
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityBoundaryVisualizationProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityBoundaryVisualizationProfileInspector.cs
new file mode 100644
index 0000000..ce465cb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityBoundaryVisualizationProfileInspector.cs
@@ -0,0 +1,143 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Boundary.Editor
+{
+ [CustomEditor(typeof(MixedRealityBoundaryVisualizationProfile))]
+ public class MixedRealityBoundaryVisualizationProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private SerializedProperty boundaryHeight;
+
+ private SerializedProperty showFloor;
+ private SerializedProperty floorMaterial;
+ private SerializedProperty floorScale;
+ private SerializedProperty floorPhysicsLayer;
+
+ private SerializedProperty showPlayArea;
+ private SerializedProperty playAreaMaterial;
+ private SerializedProperty playAreaPhysicsLayer;
+
+ private SerializedProperty showTrackedArea;
+ private SerializedProperty trackedAreaMaterial;
+ private SerializedProperty trackedAreaPhysicsLayer;
+
+ private SerializedProperty showBoundaryWalls;
+ private SerializedProperty boundaryWallMaterial;
+ private SerializedProperty boundaryWallsPhysicsLayer;
+
+ private SerializedProperty showBoundaryCeiling;
+ private SerializedProperty boundaryCeilingMaterial;
+ private SerializedProperty ceilingPhysicsLayer;
+
+ private const string ProfileTitle = "Boundary Visualization Settings";
+ private const string ProfileDescription = "Boundary visualizations can help users stay oriented and comfortable in the experience.";
+
+ private readonly GUIContent showContent = new GUIContent("Show");
+ private readonly GUIContent scaleContent = new GUIContent("Scale");
+ private readonly GUIContent materialContent = new GUIContent("Material");
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ boundaryHeight = serializedObject.FindProperty("boundaryHeight");
+
+ showFloor = serializedObject.FindProperty("showFloor");
+ floorMaterial = serializedObject.FindProperty("floorMaterial");
+ floorScale = serializedObject.FindProperty("floorScale");
+ floorPhysicsLayer = serializedObject.FindProperty("floorPhysicsLayer");
+
+ showPlayArea = serializedObject.FindProperty("showPlayArea");
+ playAreaMaterial = serializedObject.FindProperty("playAreaMaterial");
+ playAreaPhysicsLayer = serializedObject.FindProperty("playAreaPhysicsLayer");
+
+ showTrackedArea = serializedObject.FindProperty("showTrackedArea");
+ trackedAreaMaterial = serializedObject.FindProperty("trackedAreaMaterial");
+ trackedAreaPhysicsLayer = serializedObject.FindProperty("trackedAreaPhysicsLayer");
+
+ showBoundaryWalls = serializedObject.FindProperty("showBoundaryWalls");
+ boundaryWallMaterial = serializedObject.FindProperty("boundaryWallMaterial");
+ boundaryWallsPhysicsLayer = serializedObject.FindProperty("boundaryWallsPhysicsLayer");
+
+ showBoundaryCeiling = serializedObject.FindProperty("showBoundaryCeiling");
+ boundaryCeilingMaterial = serializedObject.FindProperty("boundaryCeilingMaterial");
+ ceilingPhysicsLayer = serializedObject.FindProperty("ceilingPhysicsLayer");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("General Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(boundaryHeight);
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Floor Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showFloor, showContent);
+ EditorGUILayout.PropertyField(floorMaterial, materialContent);
+ var prevWideMode = EditorGUIUtility.wideMode;
+ EditorGUIUtility.wideMode = true;
+ EditorGUILayout.PropertyField(floorScale, scaleContent, GUILayout.ExpandWidth(true));
+ EditorGUIUtility.wideMode = prevWideMode;
+ EditorGUILayout.PropertyField(floorPhysicsLayer);
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Play Area Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showPlayArea, showContent);
+ EditorGUILayout.PropertyField(playAreaMaterial, materialContent);
+ EditorGUILayout.PropertyField(playAreaPhysicsLayer);
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Tracked Area Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showTrackedArea, showContent);
+ EditorGUILayout.PropertyField(trackedAreaMaterial, materialContent);
+ EditorGUILayout.PropertyField(trackedAreaPhysicsLayer);
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Boundary Wall Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showBoundaryWalls, showContent);
+ EditorGUILayout.PropertyField(boundaryWallMaterial, materialContent);
+ EditorGUILayout.PropertyField(boundaryWallsPhysicsLayer);
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Boundary Ceiling Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showBoundaryCeiling, showContent);
+ EditorGUILayout.PropertyField(boundaryCeilingMaterial, materialContent);
+ EditorGUILayout.PropertyField(ceilingPhysicsLayer);
+ }
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.BoundaryVisualizationProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityBoundaryVisualizationProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityBoundaryVisualizationProfileInspector.cs.meta
new file mode 100644
index 0000000..1d9400e
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityBoundaryVisualizationProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: d44f14e15b3447f4846c13ca5c82129c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityCameraProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityCameraProfileInspector.cs
new file mode 100644
index 0000000..bfef929
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityCameraProfileInspector.cs
@@ -0,0 +1,169 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.CameraSystem;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ ///
+ /// Class handles rendering inspector view of MixedRealityCameraProfile object
+ ///
+ [CustomEditor(typeof(MixedRealityCameraProfile))]
+ public class MixedRealityCameraProfileInspector : BaseDataProviderServiceInspector
+ {
+ private bool showProviders = false;
+ private const string showProvidersPreferenceKey = "ShowCameraSystem_DataProviders_PreferenceKey";
+
+ private bool showDisplaySettings = false;
+ private const string showDisplaySettingsPreferenceKey = "ShowCameraSystem_DisplaySettings_PreferenceKey";
+
+ private SerializedProperty opaqueNearClip;
+ private SerializedProperty opaqueFarClip;
+ private SerializedProperty opaqueClearFlags;
+ private SerializedProperty opaqueBackgroundColor;
+ private SerializedProperty opaqueQualityLevel;
+
+ private SerializedProperty transparentNearClip;
+ private SerializedProperty transparentFarClip;
+ private SerializedProperty transparentClearFlags;
+ private SerializedProperty transparentBackgroundColor;
+ private SerializedProperty transparentQualityLevel;
+
+ private const string DataProviderErrorMsg = "The Mixed Reality Camera System will use default settings.\nAdd a settings provider to customize the camera.";
+ private static readonly GUIContent AddProviderTitle = new GUIContent("+ Add Camera Settings Provider", "Add Camera Settings Provider");
+ private static readonly GUIContent RemoveProviderTitle = new GUIContent("-", "Remove Camera Settings Provider");
+
+ private readonly GUIContent nearClipTitle = new GUIContent("Near Clip");
+ private readonly GUIContent farClipTitle = new GUIContent("Far Clip");
+ private readonly GUIContent clearFlagsTitle = new GUIContent("Clear Flags");
+ private readonly GUIContent backgroundColorTitle = new GUIContent("Background Color");
+
+ private const string profileTitle = "Camera Settings";
+ private const string profileDescription = "The Camera Profile helps configure cross platform camera settings.";
+
+ ///
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ opaqueNearClip = serializedObject.FindProperty("nearClipPlaneOpaqueDisplay");
+ opaqueFarClip = serializedObject.FindProperty("farClipPlaneOpaqueDisplay");
+ opaqueClearFlags = serializedObject.FindProperty("cameraClearFlagsOpaqueDisplay");
+ opaqueBackgroundColor = serializedObject.FindProperty("backgroundColorOpaqueDisplay");
+ opaqueQualityLevel = serializedObject.FindProperty("opaqueQualityLevel");
+
+ transparentNearClip = serializedObject.FindProperty("nearClipPlaneTransparentDisplay");
+ transparentFarClip = serializedObject.FindProperty("farClipPlaneTransparentDisplay");
+ transparentClearFlags = serializedObject.FindProperty("cameraClearFlagsTransparentDisplay");
+ transparentBackgroundColor = serializedObject.FindProperty("backgroundColorTransparentDisplay");
+ transparentQualityLevel = serializedObject.FindProperty("transparentQualityLevel");
+ }
+
+ ///
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(profileTitle, profileDescription, target))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ RenderFoldout(ref showProviders, "Camera Settings Providers", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ bool changed = RenderDataProviderList(AddProviderTitle, RemoveProviderTitle, DataProviderErrorMsg, typeof(BaseCameraSettingsProfile));
+
+ if (changed && MixedRealityToolkit.IsInitialized)
+ {
+ EditorApplication.delayCall += () => MixedRealityToolkit.Instance.ResetConfiguration(MixedRealityToolkit.Instance.ActiveProfile);
+ }
+ }
+ }, showProvidersPreferenceKey);
+
+ RenderFoldout(ref showDisplaySettings, "Display Settings", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ EditorGUILayout.LabelField("Opaque", EditorStyles.boldLabel);
+ EditorGUILayout.PropertyField(opaqueNearClip, nearClipTitle);
+ EditorGUILayout.PropertyField(opaqueFarClip, farClipTitle);
+ EditorGUILayout.PropertyField(opaqueClearFlags, clearFlagsTitle);
+
+ if ((CameraClearFlags)opaqueClearFlags.intValue == CameraClearFlags.Color)
+ {
+ EditorGUILayout.PropertyField(opaqueBackgroundColor, backgroundColorTitle);
+ }
+
+ opaqueQualityLevel.intValue = EditorGUILayout.Popup("Quality Setting", opaqueQualityLevel.intValue, QualitySettings.names);
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Transparent", EditorStyles.boldLabel);
+
+ EditorGUILayout.PropertyField(transparentNearClip, nearClipTitle);
+ EditorGUILayout.PropertyField(transparentFarClip, farClipTitle);
+ EditorGUILayout.PropertyField(transparentClearFlags, clearFlagsTitle);
+
+ if ((CameraClearFlags)transparentClearFlags.intValue == CameraClearFlags.Color)
+ {
+ EditorGUILayout.PropertyField(transparentBackgroundColor, backgroundColorTitle);
+ }
+
+ transparentQualityLevel.intValue = EditorGUILayout.Popup("Quality Setting", transparentQualityLevel.intValue, QualitySettings.names);
+ }
+ }, showDisplaySettingsPreferenceKey);
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ ///
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.CameraProfile;
+ }
+
+ #region DataProvider Inspector Utilities
+
+ ///
+ protected override SerializedProperty GetDataProviderConfigurationList()
+ {
+ return serializedObject.FindProperty("settingsConfigurations");
+ }
+
+ ///
+ protected override ServiceConfigurationProperties GetDataProviderConfigurationProperties(SerializedProperty providerEntry)
+ {
+ return new ServiceConfigurationProperties()
+ {
+ componentName = providerEntry.FindPropertyRelative("componentName"),
+ componentType = providerEntry.FindPropertyRelative("componentType"),
+ providerProfile = providerEntry.FindPropertyRelative("settingsProfile"),
+ runtimePlatform = providerEntry.FindPropertyRelative("runtimePlatform"),
+ };
+ }
+
+ ///
+ protected override IMixedRealityServiceConfiguration GetDataProviderConfiguration(int index)
+ {
+ var profile = target as MixedRealityCameraProfile;
+ var configurations = (profile != null) ? profile.SettingsConfigurations : null;
+ if (configurations != null && index >= 0 && index < configurations.Length)
+ {
+ return configurations[index];
+ }
+
+ return null;
+ }
+
+ #endregion
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityCameraProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityCameraProfileInspector.cs.meta
new file mode 100644
index 0000000..71ef569
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityCameraProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 2c3efa79d92745718c99c2cbb0bffb78
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerMappingProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerMappingProfileInspector.cs
new file mode 100644
index 0000000..af12113
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerMappingProfileInspector.cs
@@ -0,0 +1,377 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Input.UnityInput;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System;
+using System.Collections.Generic;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ [CustomEditor(typeof(MixedRealityControllerMappingProfile))]
+ public class MixedRealityControllerMappingProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private readonly struct ControllerMappingSignature
+ {
+ public SupportedControllerType SupportedControllerType { get; }
+ public Handedness Handedness { get; }
+
+ public ControllerMappingSignature(SupportedControllerType supportedControllerType, Handedness handedness)
+ {
+ SupportedControllerType = supportedControllerType;
+ Handedness = handedness;
+ }
+ }
+
+ private struct ControllerRenderProfile
+ {
+ public SupportedControllerType SupportedControllerType;
+ public Handedness Handedness;
+ public MixedRealityInteractionMapping[] Interactions;
+
+ public ControllerRenderProfile(SupportedControllerType supportedControllerType, Handedness handedness, MixedRealityInteractionMapping[] interactions)
+ {
+ SupportedControllerType = supportedControllerType;
+ Handedness = handedness;
+ Interactions = interactions;
+ }
+ }
+
+ private static readonly GUIContent ControllerAddButtonContent = new GUIContent("+ Add a New Controller Definition");
+ private static readonly GUIContent ControllerMinusButtonContent = new GUIContent("-", "Remove Controller Definition");
+ private static readonly GUIContent GenericTypeContent = new GUIContent("Generic Type");
+ private static readonly GUIContent HandednessTypeContent = new GUIContent("Handedness");
+
+ private static MixedRealityControllerMappingProfile thisProfile;
+
+ private SerializedProperty mixedRealityControllerMappings;
+
+ private static bool showControllerDefinitions = false;
+
+ private const string ProfileTitle = "Controller Input Mapping Settings";
+ private const string ProfileDescription = "Use this profile to define all the controllers and their inputs your users will be able to use in your application.\n\n" +
+ "You'll want to define all your Input Actions first. They can then be wired up to hardware sensors, controllers, gestures, and other input devices.";
+
+ private readonly List controllerRenderList = new List();
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ mixedRealityControllerMappings = serializedObject.FindProperty("mixedRealityControllerMappings");
+ thisProfile = target as MixedRealityControllerMappingProfile;
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ RenderControllerList(mixedRealityControllerMappings);
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.ControllerMappingProfile;
+ }
+
+ private void RenderControllerList(SerializedProperty controllerList)
+ {
+ if (thisProfile.MixedRealityControllerMappings.Length != controllerList.arraySize) { return; }
+
+ if (InspectorUIUtility.RenderIndentedButton(ControllerAddButtonContent, EditorStyles.miniButton))
+ {
+ AddController(controllerList, typeof(GenericJoystickController));
+ return;
+ }
+
+ controllerRenderList.Clear();
+
+ // Generating the set of controllers that belong to each Controller Mapping Signature
+ Dictionary> controllersAffectedByMappingSignatures = new Dictionary>();
+ for (int i = 0; i < thisProfile.MixedRealityControllerMappings.Length; i++)
+ {
+ MixedRealityControllerMapping controllerMapping = thisProfile.MixedRealityControllerMappings[i];
+ Type controllerType = controllerMapping.ControllerType;
+ if (controllerType == null) { continue; }
+
+ Handedness handedness = controllerMapping.Handedness;
+ SupportedControllerType supportedControllerType = controllerMapping.SupportedControllerType;
+
+ ControllerMappingSignature currentSignature = new ControllerMappingSignature(supportedControllerType, handedness);
+ if (!controllersAffectedByMappingSignatures.ContainsKey(currentSignature))
+ {
+ controllersAffectedByMappingSignatures.Add(currentSignature, new List());
+ }
+ controllersAffectedByMappingSignatures[currentSignature].Add(controllerType.ToString());
+ }
+
+ showControllerDefinitions = EditorGUILayout.Foldout(showControllerDefinitions, "Controller Definitions", true);
+ if (showControllerDefinitions)
+ {
+ using (var outerVerticalScope = new GUILayout.VerticalScope())
+ {
+ GUILayout.HorizontalScope horizontalScope = null;
+
+ for (int i = 0; i < thisProfile.MixedRealityControllerMappings.Length; i++)
+ {
+ MixedRealityControllerMapping controllerMapping = thisProfile.MixedRealityControllerMappings[i];
+ Type controllerType = controllerMapping.ControllerType;
+ if (controllerType == null) { continue; }
+
+ Handedness handedness = controllerMapping.Handedness;
+ bool useCustomInteractionMappings = controllerMapping.HasCustomInteractionMappings;
+ SupportedControllerType supportedControllerType = controllerMapping.SupportedControllerType;
+
+ var controllerMappingProperty = controllerList.GetArrayElementAtIndex(i);
+ var handednessProperty = controllerMappingProperty.FindPropertyRelative("handedness");
+
+ #region Profile Migration
+
+ // Between MRTK v2 RC2 and GA, the HoloLens clicker and HoloLens voice select input were migrated from
+ // SupportedControllerType.WindowsMixedReality && Handedness.None to SupportedControllerType.GGVHand && Handedness.None
+ if (supportedControllerType == SupportedControllerType.WindowsMixedReality && handedness == Handedness.None)
+ {
+ for (int j = 0; j < thisProfile.MixedRealityControllerMappings.Length; j++)
+ {
+ if (thisProfile.MixedRealityControllerMappings[j].SupportedControllerType == SupportedControllerType.GGVHand &&
+ thisProfile.MixedRealityControllerMappings[j].Handedness == Handedness.None)
+ {
+ if (horizontalScope != null) { horizontalScope.Dispose(); horizontalScope = null; }
+
+ serializedObject.ApplyModifiedProperties();
+
+ for (int k = 0; k < controllerMapping.Interactions.Length; k++)
+ {
+ MixedRealityInteractionMapping currentMapping = controllerMapping.Interactions[k];
+
+ if (currentMapping.InputType == DeviceInputType.Select)
+ {
+ thisProfile.MixedRealityControllerMappings[j].Interactions[0].MixedRealityInputAction = currentMapping.MixedRealityInputAction;
+ }
+ else if (currentMapping.InputType == DeviceInputType.SpatialGrip)
+ {
+ thisProfile.MixedRealityControllerMappings[j].Interactions[1].MixedRealityInputAction = currentMapping.MixedRealityInputAction;
+ }
+ }
+
+ serializedObject.Update();
+ controllerList.DeleteArrayElementAtIndex(i);
+ EditorUtility.DisplayDialog("Mappings updated", "The \"HoloLens Voice and Clicker\" mappings have been migrated to a new serialization. Please save this asset.", "Okay, thanks!");
+ return;
+ }
+ }
+ }
+
+ #endregion Profile Migration
+
+ if (!useCustomInteractionMappings)
+ {
+ bool skip = false;
+
+ // Merge controllers with the same supported controller type.
+ for (int j = 0; j < controllerRenderList.Count; j++)
+ {
+ if (controllerRenderList[j].SupportedControllerType == supportedControllerType &&
+ controllerRenderList[j].Handedness == handedness)
+ {
+ try
+ {
+ thisProfile.MixedRealityControllerMappings[i].SynchronizeInputActions(controllerRenderList[j].Interactions);
+ }
+ catch (ArgumentException e)
+ {
+ Debug.LogError($"Controller mappings between {thisProfile.MixedRealityControllerMappings[i].Description} and {controllerMapping.Description} do not match. Error message: {e.Message}");
+ }
+ serializedObject.ApplyModifiedProperties();
+ skip = true;
+ }
+ }
+
+ if (skip) { continue; }
+ }
+
+ controllerRenderList.Add(new ControllerRenderProfile(supportedControllerType, handedness, thisProfile.MixedRealityControllerMappings[i].Interactions));
+
+ string controllerTitle = thisProfile.MixedRealityControllerMappings[i].Description;
+ var interactionsProperty = controllerMappingProperty.FindPropertyRelative("interactions");
+
+ if (useCustomInteractionMappings)
+ {
+ if (horizontalScope != null) { horizontalScope.Dispose(); horizontalScope = null; }
+
+ GUILayout.Space(24f);
+
+ using (var verticalScope = new GUILayout.VerticalScope())
+ {
+ using (horizontalScope = new GUILayout.HorizontalScope())
+ {
+ EditorGUILayout.LabelField(controllerTitle, EditorStyles.boldLabel);
+
+ if (GUILayout.Button(ControllerMinusButtonContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ controllerList.DeleteArrayElementAtIndex(i);
+ return;
+ }
+ }
+
+ EditorGUI.BeginChangeCheck();
+
+ // Generic Type dropdown
+ Type[] genericTypes = MixedRealityControllerMappingProfile.CustomControllerMappingTypes;
+ var genericTypeListContent = new GUIContent[genericTypes.Length];
+ var genericTypeListIds = new int[genericTypes.Length];
+ int currentGenericType = -1;
+ for (int genericTypeIdx = 0; genericTypeIdx < genericTypes.Length; genericTypeIdx++)
+ {
+ var attribute = MixedRealityControllerAttribute.Find(genericTypes[genericTypeIdx]);
+ if (attribute != null)
+ {
+ genericTypeListContent[genericTypeIdx] = new GUIContent(attribute.SupportedControllerType.ToString().Replace("Generic", "").ToProperCase() + " Controller");
+ }
+ else
+ {
+ genericTypeListContent[genericTypeIdx] = new GUIContent("Unknown Controller");
+ }
+
+ genericTypeListIds[genericTypeIdx] = genericTypeIdx;
+
+ if (controllerType == genericTypes[genericTypeIdx])
+ {
+ currentGenericType = genericTypeIdx;
+ }
+ }
+ Debug.Assert(currentGenericType != -1);
+
+ currentGenericType = EditorGUILayout.IntPopup(GenericTypeContent, currentGenericType, genericTypeListContent, genericTypeListIds);
+ controllerType = genericTypes[currentGenericType];
+
+ {
+ // Handedness dropdown
+ var attribute = MixedRealityControllerAttribute.Find(controllerType);
+ if (attribute != null && attribute.SupportedHandedness.Length >= 1)
+ {
+ // Make sure handedness is valid for the selected controller type.
+ if (Array.IndexOf(attribute.SupportedHandedness, (Handedness)handednessProperty.intValue) < 0)
+ {
+ handednessProperty.intValue = (int)attribute.SupportedHandedness[0];
+ }
+
+ if (attribute.SupportedHandedness.Length >= 2)
+ {
+ var handednessListContent = new GUIContent[attribute.SupportedHandedness.Length];
+ var handednessListIds = new int[attribute.SupportedHandedness.Length];
+ for (int handednessIdx = 0; handednessIdx < attribute.SupportedHandedness.Length; handednessIdx++)
+ {
+ handednessListContent[handednessIdx] = new GUIContent(attribute.SupportedHandedness[handednessIdx].ToString());
+ handednessListIds[handednessIdx] = (int)attribute.SupportedHandedness[handednessIdx];
+ }
+
+ handednessProperty.intValue = EditorGUILayout.IntPopup(HandednessTypeContent, handednessProperty.intValue, handednessListContent, handednessListIds);
+ }
+ }
+ else
+ {
+ handednessProperty.intValue = (int)Handedness.None;
+ }
+ }
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ interactionsProperty.ClearArray();
+ serializedObject.ApplyModifiedProperties();
+ thisProfile.MixedRealityControllerMappings[i].ControllerType.Type = genericTypes[currentGenericType];
+ thisProfile.MixedRealityControllerMappings[i].SetDefaultInteractionMapping(true);
+ serializedObject.ApplyModifiedProperties();
+ return;
+ }
+
+ if (InspectorUIUtility.RenderIndentedButton("Edit Input Action Map"))
+ {
+ ControllerPopupWindow.Show(controllerMapping, interactionsProperty, handedness);
+ }
+
+ if (InspectorUIUtility.RenderIndentedButton("Reset Input Actions"))
+ {
+ ResetInputActions(ref thisProfile.MixedRealityControllerMappings[i]);
+ }
+ }
+ }
+ else
+ {
+ if (handedness != Handedness.Right)
+ {
+ if (horizontalScope != null) { horizontalScope.Dispose(); horizontalScope = null; }
+ horizontalScope = new GUILayout.HorizontalScope();
+ }
+
+
+ EditorGUILayout.BeginHorizontal(EditorStyles.helpBox);
+ var buttonContent = new GUIContent(controllerTitle, ControllerMappingLibrary.GetControllerTextureScaled(controllerType, handedness));
+ if (GUILayout.Button(buttonContent, MixedRealityStylesUtility.ControllerButtonStyle, GUILayout.Height(128f), GUILayout.MinWidth(32), GUILayout.ExpandWidth(true)))
+ {
+ ControllerMappingSignature buttonSignature = new ControllerMappingSignature(supportedControllerType, handedness);
+ ControllerPopupWindow.Show(controllerMapping, interactionsProperty, handedness, controllersAffectedByMappingSignatures[buttonSignature]);
+ }
+ if (GUILayout.Button(EditorGUIUtility.IconContent("_Menu"), new GUIStyle("iconButton")))
+ {
+ // create the menu and add items to it
+ GenericMenu menu = new GenericMenu();
+
+ // Caching the index of this controller mapping for the anonymous function
+ int index = i;
+ menu.AddItem(new GUIContent("Reset to default input actions"), false, () => ResetInputActions(ref thisProfile.MixedRealityControllerMappings[index]));
+ menu.ShowAsContext();
+ }
+ EditorGUILayout.EndHorizontal();
+ }
+ }
+
+ if (horizontalScope != null) { horizontalScope.Dispose(); horizontalScope = null; }
+ }
+ }
+ }
+
+ ///
+ /// Resets the input actions of the controller mapping according to the mapping's GetDefaultInteractionMappings() function
+ ///
+ /// A reference to the controller mapping struct getting reset
+ private void ResetInputActions(ref MixedRealityControllerMapping controllerMapping)
+ {
+ controllerMapping.SetDefaultInteractionMapping(true);
+ serializedObject.ApplyModifiedProperties();
+ ControllerPopupWindow.RepaintWindow();
+ }
+
+ private void AddController(SerializedProperty controllerList, Type controllerType)
+ {
+ controllerList.InsertArrayElementAtIndex(controllerList.arraySize);
+ var index = controllerList.arraySize - 1;
+ var mixedRealityControllerMapping = controllerList.GetArrayElementAtIndex(index);
+ var handednessProperty = mixedRealityControllerMapping.FindPropertyRelative("handedness");
+ handednessProperty.intValue = (int)Handedness.None;
+ var interactionsProperty = mixedRealityControllerMapping.FindPropertyRelative("interactions");
+ interactionsProperty.ClearArray();
+ serializedObject.ApplyModifiedProperties();
+ thisProfile.MixedRealityControllerMappings[index].ControllerType.Type = controllerType;
+ thisProfile.MixedRealityControllerMappings[index].SetDefaultInteractionMapping(true);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerMappingProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerMappingProfileInspector.cs.meta
new file mode 100644
index 0000000..9bf38be
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerMappingProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: f3febf47bc78475ea1ed698283efd91a
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerVisualizationProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerVisualizationProfileInspector.cs
new file mode 100644
index 0000000..dedf443
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerVisualizationProfileInspector.cs
@@ -0,0 +1,311 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Input.UnityInput;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ [CustomEditor(typeof(MixedRealityControllerVisualizationProfile))]
+ public class MixedRealityControllerVisualizationProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private static readonly GUIContent ControllerAddButtonContent = new GUIContent("+ Add a New Controller Definition");
+ private static readonly GUIContent ControllerMinusButtonContent = new GUIContent("-", "Remove Controller Definition");
+
+ private static readonly GUIContent[] HandednessSelections =
+ {
+ new GUIContent("Left Hand"),
+ new GUIContent("Right Hand"),
+ new GUIContent("Both"),
+ };
+
+ private SerializedProperty renderMotionControllers;
+ private SerializedProperty defaultControllerVisualizationType;
+
+ private SerializedProperty usePlatformControllerModels;
+ private SerializedProperty platformControllerModelMaterial;
+ private SerializedProperty globalLeftHandedControllerModel;
+ private SerializedProperty globalRightHandedControllerModel;
+ private SerializedProperty globalLeftHandModel;
+ private SerializedProperty globalRightHandModel;
+
+ private static bool showControllerDefinitions = true;
+ private SerializedProperty controllerVisualizationSettings;
+
+ private readonly XRPipelineUtility xrPipelineUtility = new XRPipelineUtility();
+
+ private MixedRealityControllerVisualizationProfile thisProfile;
+
+ private float defaultLabelWidth;
+
+ private const string ProfileTitle = "Controller Visualization Settings";
+ private const string ProfileDescription = "Define all the custom controller visualizations you'd like to use for each controller type when they're rendered in the scene.\n\n" +
+ "Global settings are the default fallback, and any specific controller definitions take precedence.";
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ defaultLabelWidth = EditorGUIUtility.labelWidth;
+
+#if UNITY_2019
+ xrPipelineUtility.Enable();
+#endif // UNITY_2019
+
+ thisProfile = target as MixedRealityControllerVisualizationProfile;
+
+ renderMotionControllers = serializedObject.FindProperty("renderMotionControllers");
+ defaultControllerVisualizationType = serializedObject.FindProperty("defaultControllerVisualizationType");
+ usePlatformControllerModels = serializedObject.FindProperty("usePlatformModels");
+ platformControllerModelMaterial = serializedObject.FindProperty("platformModelMaterial");
+ globalLeftHandedControllerModel = serializedObject.FindProperty("globalLeftControllerModel");
+ globalRightHandedControllerModel = serializedObject.FindProperty("globalRightControllerModel");
+ globalLeftHandModel = serializedObject.FindProperty("globalLeftHandVisualizer");
+ globalRightHandModel = serializedObject.FindProperty("globalRightHandVisualizer");
+ controllerVisualizationSettings = serializedObject.FindProperty("controllerVisualizationSettings");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUILayout.LabelField("Visualization Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(renderMotionControllers);
+
+ EditorGUILayout.PropertyField(defaultControllerVisualizationType);
+
+ if (thisProfile.DefaultControllerVisualizationType == null ||
+ thisProfile.DefaultControllerVisualizationType.Type == null)
+ {
+ EditorGUILayout.HelpBox("A default controller visualization type must be defined!", MessageType.Error);
+ }
+ }
+
+ var leftHandModelPrefab = globalLeftHandedControllerModel.objectReferenceValue as GameObject;
+ var rightHandModelPrefab = globalRightHandedControllerModel.objectReferenceValue as GameObject;
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Global Controller Model Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(usePlatformControllerModels);
+ if (usePlatformControllerModels.boolValue)
+ {
+ EditorGUILayout.PropertyField(platformControllerModelMaterial);
+ }
+
+ if (usePlatformControllerModels.boolValue && (leftHandModelPrefab != null || rightHandModelPrefab != null))
+ {
+ EditorGUILayout.HelpBox("When platform models are used, an attempt is made to obtain controller models from the platform SDK. The global left and right models are only shown if no model can be obtained.", MessageType.Warning);
+ }
+
+ EditorGUI.BeginChangeCheck();
+ leftHandModelPrefab = EditorGUILayout.ObjectField(new GUIContent(globalLeftHandedControllerModel.displayName, "Note: If the default model is not found, the fallback is the global left hand model."), leftHandModelPrefab, typeof(GameObject), false) as GameObject;
+
+ if (EditorGUI.EndChangeCheck() && CheckVisualizer(leftHandModelPrefab))
+ {
+ globalLeftHandedControllerModel.objectReferenceValue = leftHandModelPrefab;
+ }
+
+ EditorGUI.BeginChangeCheck();
+ rightHandModelPrefab = EditorGUILayout.ObjectField(new GUIContent(globalRightHandedControllerModel.displayName, "Note: If the default model is not found, the fallback is the global right hand model."), rightHandModelPrefab, typeof(GameObject), false) as GameObject;
+
+ if (EditorGUI.EndChangeCheck() && CheckVisualizer(rightHandModelPrefab))
+ {
+ globalRightHandedControllerModel.objectReferenceValue = rightHandModelPrefab;
+ }
+
+ EditorGUILayout.PropertyField(globalLeftHandModel);
+ EditorGUILayout.PropertyField(globalRightHandModel);
+ }
+
+ EditorGUIUtility.labelWidth = defaultLabelWidth;
+
+ EditorGUILayout.Space();
+ showControllerDefinitions = EditorGUILayout.Foldout(showControllerDefinitions, "Controller Definitions", true);
+ if (showControllerDefinitions)
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ RenderControllerList(controllerVisualizationSettings);
+ }
+ }
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.ControllerVisualizationProfile;
+ }
+
+ private void RenderControllerList(SerializedProperty controllerList)
+ {
+ if (thisProfile.ControllerVisualizationSettings.Length != controllerList.arraySize) { return; }
+
+ EditorGUILayout.Space();
+
+ if (InspectorUIUtility.RenderIndentedButton(ControllerAddButtonContent, EditorStyles.miniButton))
+ {
+ controllerList.InsertArrayElementAtIndex(controllerList.arraySize);
+ var index = controllerList.arraySize - 1;
+ var controllerSetting = controllerList.GetArrayElementAtIndex(index);
+
+ var mixedRealityControllerMappingDescription = controllerSetting.FindPropertyRelative("description");
+ mixedRealityControllerMappingDescription.stringValue = typeof(GenericJoystickController).Name;
+
+ var mixedRealityControllerHandedness = controllerSetting.FindPropertyRelative("handedness");
+ mixedRealityControllerHandedness.intValue = 1;
+
+ serializedObject.ApplyModifiedProperties();
+
+ thisProfile.ControllerVisualizationSettings[index].ControllerType.Type = typeof(GenericJoystickController);
+ return;
+ }
+
+#if UNITY_2019
+ xrPipelineUtility.RenderXRPipelineTabs();
+#endif // UNITY_2019
+
+ for (int i = 0; i < controllerList.arraySize; i++)
+ {
+ var controllerSetting = controllerList.GetArrayElementAtIndex(i);
+ var mixedRealityControllerMappingDescription = controllerSetting.FindPropertyRelative("description");
+ SystemType controllerType = thisProfile.ControllerVisualizationSettings[i].ControllerType;
+ bool hasValidType = controllerType != null &&
+ controllerType.Type != null;
+
+ if (hasValidType)
+ {
+ MixedRealityControllerAttribute controllerAttribute = MixedRealityControllerAttribute.Find(controllerType.Type);
+ if (controllerAttribute != null && !controllerAttribute.SupportedUnityXRPipelines.IsMaskSet(xrPipelineUtility.SelectedPipeline))
+ {
+ continue;
+ }
+ }
+ else if (!MixedRealityProjectPreferences.ShowNullDataProviders)
+ {
+ continue;
+ }
+
+ EditorGUILayout.Space();
+
+ mixedRealityControllerMappingDescription.stringValue = hasValidType
+ ? controllerType.Type.Name.ToProperCase()
+ : "Undefined Controller";
+
+ serializedObject.ApplyModifiedProperties();
+ SerializedProperty mixedRealityControllerHandedness = controllerSetting.FindPropertyRelative("handedness");
+
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ EditorGUILayout.LabelField($"{mixedRealityControllerMappingDescription.stringValue} {((Handedness)mixedRealityControllerHandedness.intValue).ToString().ToProperCase()} Hand{(mixedRealityControllerHandedness.intValue == (int)(Handedness.Both) ? "s" : "")}", EditorStyles.boldLabel);
+
+ if (GUILayout.Button(ControllerMinusButtonContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ controllerList.DeleteArrayElementAtIndex(i);
+ return;
+ }
+ }
+
+ EditorGUILayout.PropertyField(controllerSetting.FindPropertyRelative("controllerType"));
+ EditorGUILayout.PropertyField(controllerSetting.FindPropertyRelative("controllerVisualizationType"));
+
+ if (!hasValidType)
+ {
+ EditorGUILayout.HelpBox("A controller type must be defined!", MessageType.Error);
+ }
+
+ var handednessValue = mixedRealityControllerHandedness.intValue - 1;
+
+ // Reset in case it was set to something other than left, right, or both.
+ if (handednessValue < 0 || handednessValue > 2) { handednessValue = 0; }
+
+ EditorGUI.BeginChangeCheck();
+ handednessValue = EditorGUILayout.IntPopup(new GUIContent(mixedRealityControllerHandedness.displayName, mixedRealityControllerHandedness.tooltip), handednessValue, HandednessSelections, null);
+ if (EditorGUI.EndChangeCheck())
+ {
+ mixedRealityControllerHandedness.intValue = handednessValue + 1;
+ }
+
+ var overrideModel = controllerSetting.FindPropertyRelative("overrideModel");
+ var overrideModelPrefab = overrideModel.objectReferenceValue as GameObject;
+
+ var controllerUsePlatformModelOverride = controllerSetting.FindPropertyRelative("usePlatformModels");
+ EditorGUILayout.PropertyField(controllerUsePlatformModelOverride);
+ if (controllerUsePlatformModelOverride.boolValue)
+ {
+ var platformModelMaterial = controllerSetting.FindPropertyRelative("platformModelMaterial");
+ EditorGUILayout.PropertyField(platformModelMaterial);
+ }
+
+ if (controllerUsePlatformModelOverride.boolValue && overrideModelPrefab != null)
+ {
+ EditorGUILayout.HelpBox("When platform model is used, the override model will only be used if the default model cannot be loaded from the driver.", MessageType.Warning);
+ }
+
+ EditorGUI.BeginChangeCheck();
+ overrideModelPrefab = EditorGUILayout.ObjectField(new GUIContent(overrideModel.displayName, "If no override model is set, the global model is used."), overrideModelPrefab, typeof(GameObject), false) as GameObject;
+ if (overrideModelPrefab == null && !controllerUsePlatformModelOverride.boolValue)
+ {
+ EditorGUILayout.HelpBox("No override model was assigned and this controller will not attempt to use the platform's model, the global model will be used instead", MessageType.Warning);
+ }
+
+ if (EditorGUI.EndChangeCheck() && CheckVisualizer(overrideModelPrefab))
+ {
+ overrideModel.objectReferenceValue = overrideModelPrefab;
+ }
+ }
+ }
+
+ private bool CheckVisualizer(GameObject modelPrefab)
+ {
+ if (modelPrefab == null) { return true; }
+
+ if (PrefabUtility.GetPrefabAssetType(modelPrefab) == PrefabAssetType.NotAPrefab)
+ {
+ Debug.LogWarning("Assigned GameObject must be a prefab.");
+ return false;
+ }
+
+ var componentList = modelPrefab.GetComponentsInChildren();
+
+ if (componentList == null || componentList.Length == 0)
+ {
+ if (thisProfile.DefaultControllerVisualizationType != null &&
+ thisProfile.DefaultControllerVisualizationType.Type != null)
+ {
+ modelPrefab.AddComponent(thisProfile.DefaultControllerVisualizationType.Type);
+ return true;
+ }
+
+ Debug.LogError("No controller visualization type specified!");
+ }
+ else if (componentList.Length == 1)
+ {
+ return true;
+ }
+ else if (componentList.Length > 1)
+ {
+ Debug.LogWarning("Found too many IMixedRealityControllerVisualizer components on your prefab. There can only be one.");
+ }
+
+ return false;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerVisualizationProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerVisualizationProfileInspector.cs.meta
new file mode 100644
index 0000000..df94e28
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityControllerVisualizationProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 752bbdac6dcc46cb9881fa17c072ad02
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityDiagnosticsSystemProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityDiagnosticsSystemProfileInspector.cs
new file mode 100644
index 0000000..b27ca74
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityDiagnosticsSystemProfileInspector.cs
@@ -0,0 +1,94 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using UnityEditor;
+
+namespace Microsoft.MixedReality.Toolkit.Diagnostics.Editor
+{
+ [CustomEditor(typeof(MixedRealityDiagnosticsProfile))]
+ public class MixedRealityDiagnosticsSystemProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private SerializedProperty showDiagnostics;
+ private SerializedProperty showProfiler;
+ private SerializedProperty frameSampleRate;
+ private SerializedProperty windowAnchor;
+ private SerializedProperty windowOffset;
+ private SerializedProperty windowScale;
+ private SerializedProperty windowFollowSpeed;
+ private SerializedProperty showProfilerDuringMRC;
+
+ private const string ProfileTitle = "Diagnostic Settings";
+ private const string ProfileDescription = "Diagnostic visualizations can help monitor system resources and performance inside an application.";
+
+ // todo: coming soon
+ // private static bool showDebugPanelSettings = true;
+ // private SerializedProperty isDebugPanelVisible;
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ showDiagnostics = serializedObject.FindProperty("showDiagnostics");
+ showProfiler = serializedObject.FindProperty("showProfiler");
+ frameSampleRate = serializedObject.FindProperty("frameSampleRate");
+ windowAnchor = serializedObject.FindProperty("windowAnchor");
+ windowOffset = serializedObject.FindProperty("windowOffset");
+ windowScale = serializedObject.FindProperty("windowScale");
+ windowFollowSpeed = serializedObject.FindProperty("windowFollowSpeed");
+ showProfilerDuringMRC = serializedObject.FindProperty("showProfilerDuringMRC");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("General Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showDiagnostics);
+ if (!showDiagnostics.boolValue)
+ {
+ EditorGUILayout.Space();
+ EditorGUILayout.HelpBox("Diagnostic visualizations have been globally disabled.", MessageType.Info);
+ EditorGUILayout.Space();
+ }
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Profiler Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showProfiler);
+ EditorGUILayout.PropertyField(frameSampleRate);
+ EditorGUILayout.PropertyField(windowAnchor);
+ EditorGUILayout.PropertyField(windowOffset);
+ EditorGUILayout.PropertyField(windowScale);
+ EditorGUILayout.PropertyField(windowFollowSpeed);
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("HoloLens Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(showProfilerDuringMRC);
+ }
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.DiagnosticsSystemProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityDiagnosticsSystemProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityDiagnosticsSystemProfileInspector.cs.meta
new file mode 100644
index 0000000..7de7fcb
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityDiagnosticsSystemProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: b45f0bc9f7b177946a19ca6ae894edb4
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityEyeTrackingProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityEyeTrackingProfileInspector.cs
new file mode 100644
index 0000000..48b0362
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityEyeTrackingProfileInspector.cs
@@ -0,0 +1,49 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Input;
+using System.Linq;
+using UnityEditor;
+
+namespace Microsoft.MixedReality.Toolkit.Inspectors
+{
+ [CustomEditor(typeof(MixedRealityEyeTrackingProfile))]
+ public class MixedRealityEyeTrackingProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private SerializedProperty smoothEyeTracking;
+
+ private const string ProfileTitle = "Eye tracking Settings";
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+ smoothEyeTracking = serializedObject.FindProperty("smoothEyeTracking");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, string.Empty, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+ EditorGUILayout.PropertyField(smoothEyeTracking);
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.DataProviderConfigurations != null &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.DataProviderConfigurations.Any(s => profile == s.Profile);
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityEyeTrackingProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityEyeTrackingProfileInspector.cs.meta
new file mode 100644
index 0000000..b3c8c16
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityEyeTrackingProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: e1dc2bf9fb63a40428f55c55c86e326c
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityGesturesProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityGesturesProfileInspector.cs
new file mode 100644
index 0000000..57d1062
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityGesturesProfileInspector.cs
@@ -0,0 +1,256 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ [CustomEditor(typeof(MixedRealityGesturesProfile))]
+ public class MixedRealityGesturesProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private static readonly GUIContent MinusButtonContent = new GUIContent("-", "Remove defined Gesture");
+ private static readonly GUIContent AddButtonContent = new GUIContent("+ Add a New defined Gesture");
+ private static readonly GUIContent DescriptionContent = new GUIContent("Description", "The human readable description of the Gesture.");
+ private static readonly GUIContent GestureTypeContent = new GUIContent("Gesture Type", "The type of Gesture that will trigger the action.");
+ private static readonly GUIContent ActionContent = new GUIContent("Action", "The action to trigger when a Gesture is recognized.");
+
+ private const string ProfileTitle = "Gesture Settings";
+ private const string ProfileDescription = "This gesture map is any and all movements of part the user's body, especially a hand or the head, that raise actions through the input system.\n\n" +
+ "Note: Defined controllers can look up the list of gestures and raise the events based on specific criteria.";
+
+ private SerializedProperty gestures;
+ private SerializedProperty windowsManipulationGestureSettings;
+ private SerializedProperty useRailsNavigation;
+ private SerializedProperty windowsNavigationGestureSettings;
+ private SerializedProperty windowsRailsNavigationGestures;
+ private SerializedProperty windowsGestureAutoStart;
+
+ private MixedRealityGesturesProfile thisProfile;
+ private static GUIContent[] allGestureLabels;
+ private static int[] allGestureIds;
+ private static GUIContent[] actionLabels = Array.Empty();
+ private static int[] actionIds = Array.Empty();
+ private bool isInitialized = false;
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+ isInitialized = false;
+
+ gestures = serializedObject.FindProperty("gestures");
+ windowsManipulationGestureSettings = serializedObject.FindProperty("manipulationGestures");
+ useRailsNavigation = serializedObject.FindProperty("useRailsNavigation");
+ windowsNavigationGestureSettings = serializedObject.FindProperty("navigationGestures");
+ windowsRailsNavigationGestures = serializedObject.FindProperty("railsNavigationGestures");
+ windowsGestureAutoStart = serializedObject.FindProperty("windowsGestureAutoStart");
+
+ thisProfile = target as MixedRealityGesturesProfile;
+ Debug.Assert(thisProfile != null);
+
+ UpdateGestureLabels();
+
+ if (!IsProfileInActiveInstance()
+ || MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile == null)
+ {
+ return;
+ }
+
+ var inputActions = MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions;
+ actionLabels = inputActions.Select(action => new GUIContent(action.Description)).Prepend(new GUIContent("None")).ToArray();
+ actionIds = inputActions.Select(action => (int)action.Id).Prepend(0).ToArray();
+
+ isInitialized = true;
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.GesturesProfile;
+ }
+
+ private void UpdateGestureLabels()
+ {
+ var allGestureTypeNames = Enum.GetNames(typeof(GestureInputType));
+
+ var tempIds = new List();
+ var tempContent = new List();
+
+ for (int i = 0; i < allGestureTypeNames.Length; i++)
+ {
+ if (allGestureTypeNames[i].Equals("None") ||
+ thisProfile.Gestures.All(mapping => !allGestureTypeNames[i].Equals(mapping.GestureType.ToString())))
+ {
+ tempContent.Add(new GUIContent(allGestureTypeNames[i]));
+ tempIds.Add(i);
+ }
+ }
+
+ allGestureIds = tempIds.ToArray();
+ allGestureLabels = tempContent.ToArray();
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, isInitialized, BackProfileType.Input))
+ {
+ return;
+ }
+
+ CheckMixedRealityInputActions();
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Windows Gesture Settings", EditorStyles.boldLabel);
+ EditorGUILayout.PropertyField(windowsManipulationGestureSettings);
+ EditorGUILayout.PropertyField(windowsNavigationGestureSettings);
+ EditorGUILayout.PropertyField(useRailsNavigation);
+ EditorGUILayout.PropertyField(windowsRailsNavigationGestures);
+ EditorGUILayout.PropertyField(windowsGestureAutoStart);
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Defined Recognizable Gestures", EditorStyles.boldLabel);
+
+ RenderList(gestures);
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ private void RenderList(SerializedProperty list)
+ {
+ // Disable gestures list if we could not initialize successfully
+ using (new EditorGUI.DisabledGroupScope(!isInitialized))
+ {
+ EditorGUILayout.Space();
+ using (new EditorGUILayout.VerticalScope())
+ {
+ if (InspectorUIUtility.RenderIndentedButton(AddButtonContent, EditorStyles.miniButton))
+ {
+ list.arraySize += 1;
+ var speechCommand = list.GetArrayElementAtIndex(list.arraySize - 1);
+ var keyword = speechCommand.FindPropertyRelative("description");
+ keyword.stringValue = string.Empty;
+ var gestureType = speechCommand.FindPropertyRelative("gestureType");
+ gestureType.intValue = (int)GestureInputType.None;
+ var action = speechCommand.FindPropertyRelative("action");
+ var actionId = action.FindPropertyRelative("id");
+ actionId.intValue = 0;
+ var actionDescription = action.FindPropertyRelative("description");
+ actionDescription.stringValue = string.Empty;
+ var actionConstraint = action.FindPropertyRelative("axisConstraint");
+ actionConstraint.intValue = 0;
+ }
+
+ if (list == null || list.arraySize == 0)
+ {
+ EditorGUILayout.HelpBox("Define a new Gesture.", MessageType.Warning);
+ UpdateGestureLabels();
+ return;
+ }
+
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ var labelWidth = EditorGUIUtility.labelWidth;
+ EditorGUIUtility.labelWidth = 24f;
+ EditorGUILayout.LabelField(DescriptionContent, GUILayout.ExpandWidth(true));
+ EditorGUILayout.LabelField(GestureTypeContent, GUILayout.Width(80f));
+ EditorGUILayout.LabelField(ActionContent, GUILayout.Width(64f));
+ EditorGUILayout.LabelField(string.Empty, GUILayout.Width(24f));
+ EditorGUIUtility.labelWidth = labelWidth;
+ }
+
+ var inputActions = GetInputActions();
+
+ for (int i = 0; i < list.arraySize; i++)
+ {
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ SerializedProperty gesture = list.GetArrayElementAtIndex(i);
+ var keyword = gesture.FindPropertyRelative("description");
+ var gestureType = gesture.FindPropertyRelative("gestureType");
+ var action = gesture.FindPropertyRelative("action");
+ var actionId = action.FindPropertyRelative("id");
+ var actionDescription = action.FindPropertyRelative("description");
+ var actionConstraint = action.FindPropertyRelative("axisConstraint");
+
+ EditorGUILayout.PropertyField(keyword, GUIContent.none, GUILayout.ExpandWidth(true));
+
+ Debug.Assert(allGestureLabels.Length == allGestureIds.Length);
+
+ var gestureLabels = new GUIContent[allGestureLabels.Length + 1];
+ var gestureIds = new int[allGestureIds.Length + 1];
+
+ gestureLabels[0] = new GUIContent(((GestureInputType)gestureType.intValue).ToString());
+ gestureIds[0] = gestureType.intValue;
+
+ for (int j = 0; j < allGestureLabels.Length; j++)
+ {
+ gestureLabels[j + 1] = allGestureLabels[j];
+ gestureIds[j + 1] = allGestureIds[j];
+ }
+
+ EditorGUI.BeginChangeCheck();
+ gestureType.intValue = EditorGUILayout.IntPopup(GUIContent.none, gestureType.intValue, gestureLabels, gestureIds, GUILayout.Width(80f));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ serializedObject.ApplyModifiedProperties();
+ UpdateGestureLabels();
+ }
+
+ EditorGUI.BeginChangeCheck();
+
+ actionId.intValue = EditorGUILayout.IntPopup(GUIContent.none, actionId.intValue, actionLabels, actionIds, GUILayout.Width(64f));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ MixedRealityInputAction inputAction = MixedRealityInputAction.None;
+ int idx = actionId.intValue - 1;
+ if (idx >= 0 && idx < inputActions.Length)
+ {
+ inputAction = inputActions[idx];
+ }
+
+ actionDescription.stringValue = inputAction.Description;
+ actionConstraint.intValue = (int)inputAction.AxisConstraint;
+ serializedObject.ApplyModifiedProperties();
+ }
+
+ if (GUILayout.Button(MinusButtonContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ list.DeleteArrayElementAtIndex(i);
+ serializedObject.ApplyModifiedProperties();
+ UpdateGestureLabels();
+ }
+ }
+ }
+ }
+ }
+ }
+
+ private static MixedRealityInputAction[] GetInputActions()
+ {
+ if (!MixedRealityToolkit.IsInitialized ||
+ !MixedRealityToolkit.Instance.HasActiveProfile ||
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile == null ||
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile == null)
+ {
+ return Array.Empty();
+ }
+
+ return MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityGesturesProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityGesturesProfileInspector.cs.meta
new file mode 100644
index 0000000..c005f49
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityGesturesProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 82cccce472e145f4bcdf7338afc66e46
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityHandTrackingProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityHandTrackingProfileInspector.cs
new file mode 100644
index 0000000..6e2ebf6
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityHandTrackingProfileInspector.cs
@@ -0,0 +1,72 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Input;
+using UnityEditor;
+
+namespace Microsoft.MixedReality.Toolkit.Inspectors
+{
+ [CustomEditor(typeof(MixedRealityHandTrackingProfile))]
+ public class MixedRealityHandTrackingProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private SerializedProperty jointPrefab;
+ private SerializedProperty palmPrefab;
+ private SerializedProperty fingertipPrefab;
+ private SerializedProperty systemHandMeshMaterial;
+ private SerializedProperty riggedHandMeshMaterial;
+ private SerializedProperty handMeshVisualizationModes;
+ private SerializedProperty handJointVisualizationModes;
+
+ private const string ProfileTitle = "Articulated Hand Tracking Settings";
+ private const string ProfileDescription = "Use this for hand tracking settings.";
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ jointPrefab = serializedObject.FindProperty("jointPrefab");
+ fingertipPrefab = serializedObject.FindProperty("fingertipPrefab");
+ palmPrefab = serializedObject.FindProperty("palmPrefab");
+ systemHandMeshMaterial = serializedObject.FindProperty("systemHandMeshMaterial");
+ riggedHandMeshMaterial = serializedObject.FindProperty("riggedHandMeshMaterial");
+ handMeshVisualizationModes = serializedObject.FindProperty("handMeshVisualizationModes");
+ handJointVisualizationModes = serializedObject.FindProperty("handJointVisualizationModes");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUILayout.LabelField("General settings", EditorStyles.boldLabel);
+ EditorGUILayout.PropertyField(jointPrefab);
+ EditorGUILayout.PropertyField(palmPrefab);
+ EditorGUILayout.PropertyField(fingertipPrefab);
+ EditorGUILayout.PropertyField(systemHandMeshMaterial);
+ EditorGUILayout.PropertyField(riggedHandMeshMaterial);
+
+ EditorGUILayout.LabelField("Visualization settings", EditorStyles.boldLabel);
+ EditorGUILayout.PropertyField(handMeshVisualizationModes);
+ EditorGUILayout.PropertyField(handJointVisualizationModes);
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.HandTrackingProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityHandTrackingProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityHandTrackingProfileInspector.cs.meta
new file mode 100644
index 0000000..239e646
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityHandTrackingProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 1f0f672d65a59dc40b12814dad7f3122
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionRulesInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionRulesInspector.cs
new file mode 100644
index 0000000..c543bdf
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionRulesInspector.cs
@@ -0,0 +1,566 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System.Linq;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ [CustomEditor(typeof(MixedRealityInputActionRulesProfile))]
+ public class MixedRealityInputActionRulesInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private static readonly GUIContent RuleAddButtonContent = new GUIContent("+ Add a New Rule Definition");
+ private static readonly GUIContent RuleMinusButtonContent = new GUIContent("-", "Remove Rule Definition");
+ private static readonly GUIContent BaseActionContent = new GUIContent("Base Input Action:", "The Action that will raise new actions based on the criteria met");
+ private static readonly GUIContent RuleActionContent = new GUIContent("Rule Input Action:", "The Action that will be raised when the criteria is met");
+ private static readonly GUIContent CriteriaContent = new GUIContent("Action Criteria:", "The Criteria that must be met in order to raise the new Action");
+
+ private const string ProfileTitle = "Input Action Rule Settings";
+ private const string ProfileDescription = "Input Action Rules help define alternative Actions that will be raised based on specific criteria.\n\n" +
+ "You can create new rules by assigning a base Input Action below, then assigning the criteria you'd like to meet. When the criteria is met, the Rule's Action will be raised with the criteria value.\n\n" +
+ "Note: Rules can only be created for the same axis constraints.";
+
+ private SerializedProperty inputActionRulesDigital;
+ private SerializedProperty inputActionRulesSingleAxis;
+ private SerializedProperty inputActionRulesDualAxis;
+ private SerializedProperty inputActionRulesVectorAxis;
+ private SerializedProperty inputActionRulesQuaternionAxis;
+ private SerializedProperty inputActionRulesPoseAxis;
+
+ private int[] baseActionIds = System.Array.Empty();
+ private string[] baseActionLabels = System.Array.Empty();
+
+ // These are marked as static because this inspector will reset itself every refresh
+ // because it can be rendered as a sub-profile and thus OnEnable() is called every time
+ private static int[] ruleActionIds = System.Array.Empty();
+ private static string[] ruleActionLabels = System.Array.Empty();
+
+ private static int selectedBaseActionId = 0;
+ private static int selectedRuleActionId = 0;
+
+ private static MixedRealityInputAction currentBaseAction = MixedRealityInputAction.None;
+ private static MixedRealityInputAction currentRuleAction = MixedRealityInputAction.None;
+
+ private static bool currentBoolCriteria;
+ private static float currentSingleAxisCriteria;
+ private static Vector2 currentDualAxisCriteria;
+ private static Vector3 currentVectorCriteria;
+ private static Quaternion currentQuaternionCriteria;
+ private static MixedRealityPose currentPoseCriteria;
+
+ private static bool[] digitalFoldouts;
+ private static bool[] singleAxisFoldouts;
+ private static bool[] dualAxisFoldouts;
+ private static bool[] vectorFoldouts;
+ private static bool[] quaternionFoldouts;
+ private static bool[] poseFoldouts;
+
+ private MixedRealityInputActionRulesProfile thisProfile;
+ private bool isInitialized = false;
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+ isInitialized = false;
+
+ inputActionRulesDigital = serializedObject.FindProperty("inputActionRulesDigital");
+ inputActionRulesSingleAxis = serializedObject.FindProperty("inputActionRulesSingleAxis");
+ inputActionRulesDualAxis = serializedObject.FindProperty("inputActionRulesDualAxis");
+ inputActionRulesVectorAxis = serializedObject.FindProperty("inputActionRulesVectorAxis");
+ inputActionRulesQuaternionAxis = serializedObject.FindProperty("inputActionRulesQuaternionAxis");
+ inputActionRulesPoseAxis = serializedObject.FindProperty("inputActionRulesPoseAxis");
+
+ thisProfile = target as MixedRealityInputActionRulesProfile;
+
+ // Only reset if we haven't get done so
+ if (digitalFoldouts == null)
+ {
+ ResetCriteria();
+ }
+
+ if (!IsProfileInActiveInstance()
+ || MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile == null)
+ {
+ return;
+ }
+
+ var inputActions = GetInputActions();
+ baseActionLabels = inputActions.Where(action => action.AxisConstraint != AxisType.None && action.AxisConstraint != AxisType.Raw)
+ .Select(action => action.Description).ToArray();
+
+ baseActionIds = inputActions.Where(action => action.AxisConstraint != AxisType.None && action.AxisConstraint != AxisType.Raw)
+ .Select(action => (int)action.Id).ToArray();
+
+ isInitialized = true;
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, isInitialized, BackProfileType.Input))
+ {
+ return;
+ }
+
+ CheckMixedRealityInputActions();
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ selectedBaseActionId = RenderBaseInputAction(selectedBaseActionId, out currentBaseAction);
+
+ using (new EditorGUI.DisabledGroupScope(currentBaseAction == MixedRealityInputAction.None))
+ {
+ RenderCriteriaField(currentBaseAction);
+
+ if (selectedBaseActionId == selectedRuleActionId)
+ {
+ selectedRuleActionId = 0;
+ }
+
+ selectedRuleActionId = RenderRuleInputAction(selectedRuleActionId, out currentRuleAction);
+
+ EditorGUILayout.Space();
+ }
+
+ bool addButtonEnable = !RuleExists() &&
+ currentBaseAction != MixedRealityInputAction.None &&
+ currentRuleAction != MixedRealityInputAction.None &&
+ currentBaseAction.AxisConstraint != AxisType.None &&
+ currentBaseAction.AxisConstraint != AxisType.Raw;
+
+ using (new EditorGUI.DisabledGroupScope(!addButtonEnable))
+ {
+ if (InspectorUIUtility.RenderIndentedButton(RuleAddButtonContent, EditorStyles.miniButton))
+ {
+ AddRule();
+ ResetCriteria();
+ }
+ }
+
+ EditorGUILayout.Space();
+
+ var isWideMode = EditorGUIUtility.wideMode;
+ EditorGUIUtility.wideMode = true;
+
+ RenderList(inputActionRulesDigital, digitalFoldouts);
+ RenderList(inputActionRulesSingleAxis, singleAxisFoldouts);
+ RenderList(inputActionRulesDualAxis, dualAxisFoldouts);
+ RenderList(inputActionRulesVectorAxis, vectorFoldouts);
+ RenderList(inputActionRulesQuaternionAxis, quaternionFoldouts);
+ RenderList(inputActionRulesPoseAxis, poseFoldouts);
+
+ EditorGUIUtility.wideMode = isWideMode;
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionRulesProfile;
+ }
+
+ private bool RuleExists()
+ {
+ switch (currentBaseAction.AxisConstraint)
+ {
+ default:
+ return false;
+ case AxisType.Digital:
+ return thisProfile.InputActionRulesDigital.Any(digitalRule => digitalRule.BaseAction == currentBaseAction && digitalRule.RuleAction == currentRuleAction && digitalRule.Criteria == currentBoolCriteria);
+ case AxisType.SingleAxis:
+ return thisProfile.InputActionRulesSingleAxis.Any(singleAxisRule => singleAxisRule.BaseAction == currentBaseAction && singleAxisRule.RuleAction == currentRuleAction && singleAxisRule.Criteria.Equals(currentSingleAxisCriteria));
+ case AxisType.DualAxis:
+ return thisProfile.InputActionRulesDualAxis.Any(dualAxisRule => dualAxisRule.BaseAction == currentBaseAction && dualAxisRule.RuleAction == currentRuleAction && dualAxisRule.Criteria == currentDualAxisCriteria);
+ case AxisType.ThreeDofPosition:
+ return thisProfile.InputActionRulesVectorAxis.Any(vectorAxisRule => vectorAxisRule.BaseAction == currentBaseAction && vectorAxisRule.RuleAction == currentRuleAction && vectorAxisRule.Criteria == currentVectorCriteria);
+ case AxisType.ThreeDofRotation:
+ return thisProfile.InputActionRulesQuaternionAxis.Any(quaternionRule => quaternionRule.BaseAction == currentBaseAction && quaternionRule.RuleAction == currentRuleAction && quaternionRule.Criteria == currentQuaternionCriteria);
+ case AxisType.SixDof:
+ return thisProfile.InputActionRulesPoseAxis.Any(poseRule => poseRule.BaseAction == currentBaseAction && poseRule.RuleAction == currentRuleAction && poseRule.Criteria == currentPoseCriteria);
+ }
+ }
+
+ private void ResetCriteria()
+ {
+ selectedBaseActionId = 0;
+ selectedRuleActionId = 0;
+ currentBaseAction = MixedRealityInputAction.None;
+ currentRuleAction = MixedRealityInputAction.None;
+ currentBoolCriteria = false;
+ currentSingleAxisCriteria = 0f;
+ currentDualAxisCriteria = Vector2.zero;
+ currentVectorCriteria = Vector3.zero;
+ currentQuaternionCriteria = Quaternion.identity;
+ currentPoseCriteria = MixedRealityPose.ZeroIdentity;
+
+ digitalFoldouts = new bool[inputActionRulesDigital.arraySize];
+ singleAxisFoldouts = new bool[inputActionRulesSingleAxis.arraySize];
+ dualAxisFoldouts = new bool[inputActionRulesDualAxis.arraySize];
+ vectorFoldouts = new bool[inputActionRulesVectorAxis.arraySize];
+ quaternionFoldouts = new bool[inputActionRulesQuaternionAxis.arraySize];
+ poseFoldouts = new bool[inputActionRulesPoseAxis.arraySize];
+ }
+
+ private static void GetCompatibleActions(MixedRealityInputAction baseAction)
+ {
+ var inputActions = GetInputActions();
+
+ ruleActionLabels = inputActions.Where(inputAction => inputAction.AxisConstraint == baseAction.AxisConstraint && inputAction.Id != baseAction.Id)
+ .Select(action => action.Description).ToArray();
+
+ ruleActionIds = inputActions.Where(inputAction => inputAction.AxisConstraint == baseAction.AxisConstraint && inputAction.Id != baseAction.Id)
+ .Select(action => (int)action.Id).ToArray();
+ }
+
+ private void RenderCriteriaField(MixedRealityInputAction action, SerializedProperty criteriaValue = null)
+ {
+ var isWideMode = EditorGUIUtility.wideMode;
+ EditorGUIUtility.wideMode = true;
+ if (action != MixedRealityInputAction.None)
+ {
+ switch (action.AxisConstraint)
+ {
+ default:
+ EditorGUILayout.HelpBox("Base rule must have a valid axis constraint.", MessageType.Warning);
+ break;
+ case AxisType.Digital:
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ EditorGUILayout.LabelField(CriteriaContent, GUILayout.Width(128));
+ EditorGUI.BeginChangeCheck();
+ var boolValue = EditorGUILayout.Toggle(GUIContent.none, criteriaValue?.boolValue ?? currentBoolCriteria, GUILayout.Width(64), GUILayout.ExpandWidth(true));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (criteriaValue != null)
+ {
+ criteriaValue.boolValue = boolValue;
+ }
+ else
+ {
+ currentBoolCriteria = boolValue;
+ }
+ }
+ }
+ break;
+ case AxisType.SingleAxis:
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ EditorGUILayout.LabelField(CriteriaContent, GUILayout.Width(128));
+ EditorGUI.BeginChangeCheck();
+ var floatValue = EditorGUILayout.FloatField(GUIContent.none, criteriaValue?.floatValue ?? currentSingleAxisCriteria, GUILayout.Width(64), GUILayout.ExpandWidth(true));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (criteriaValue != null)
+ {
+ criteriaValue.floatValue = floatValue;
+ }
+ else
+ {
+ currentSingleAxisCriteria = floatValue;
+ }
+ }
+ }
+ break;
+ case AxisType.DualAxis:
+ EditorGUILayout.LabelField(CriteriaContent, GUILayout.Width(128));
+ using (new EditorGUI.IndentLevelScope())
+ {
+ EditorGUI.BeginChangeCheck();
+ var dualAxisValue = EditorGUILayout.Vector2Field("Position", criteriaValue?.vector2Value ?? currentDualAxisCriteria, GUILayout.Width(64), GUILayout.ExpandWidth(true));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (criteriaValue != null)
+ {
+ criteriaValue.vector2Value = dualAxisValue;
+ }
+ else
+ {
+ currentDualAxisCriteria = dualAxisValue;
+ }
+ }
+ }
+ break;
+ case AxisType.ThreeDofPosition:
+ EditorGUILayout.LabelField(CriteriaContent, GUILayout.Width(128));
+ using (new EditorGUI.IndentLevelScope())
+ {
+ EditorGUI.BeginChangeCheck();
+ var positionValue = EditorGUILayout.Vector3Field("Position", criteriaValue?.vector3Value ?? currentVectorCriteria, GUILayout.ExpandWidth(true));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (criteriaValue != null)
+ {
+ criteriaValue.vector3Value = positionValue;
+ }
+ else
+ {
+ currentVectorCriteria = positionValue;
+ }
+ }
+ }
+ break;
+ case AxisType.ThreeDofRotation:
+ EditorGUILayout.LabelField(CriteriaContent, GUILayout.Width(128));
+ using (new EditorGUI.IndentLevelScope())
+ {
+ EditorGUI.BeginChangeCheck();
+ var rotationValue = EditorGUILayout.Vector3Field("Rotation", criteriaValue?.quaternionValue.eulerAngles ?? currentQuaternionCriteria.eulerAngles, GUILayout.ExpandWidth(true));
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (criteriaValue != null)
+ {
+ criteriaValue.quaternionValue = Quaternion.Euler(rotationValue);
+ }
+ else
+ {
+ currentQuaternionCriteria = Quaternion.Euler(rotationValue);
+ }
+ }
+ }
+ break;
+ case AxisType.SixDof:
+ EditorGUILayout.LabelField(CriteriaContent, GUILayout.Width(128));
+ using (new EditorGUI.IndentLevelScope())
+ {
+ var posePosition = currentPoseCriteria.Position;
+ var poseRotation = currentPoseCriteria.Rotation;
+
+ if (criteriaValue != null)
+ {
+ posePosition = criteriaValue.FindPropertyRelative("position").vector3Value;
+ poseRotation = criteriaValue.FindPropertyRelative("rotation").quaternionValue;
+ }
+
+ EditorGUI.BeginChangeCheck();
+ posePosition = EditorGUILayout.Vector3Field("Position", posePosition);
+
+ poseRotation.eulerAngles = EditorGUILayout.Vector3Field("Rotation", poseRotation.eulerAngles);
+
+ if (EditorGUI.EndChangeCheck())
+ {
+ if (criteriaValue != null)
+ {
+ criteriaValue.FindPropertyRelative("position").vector3Value = posePosition;
+ criteriaValue.FindPropertyRelative("rotation").quaternionValue = poseRotation;
+ }
+ else
+ {
+ currentPoseCriteria.Position = posePosition;
+ currentPoseCriteria.Rotation = poseRotation;
+ }
+ }
+ }
+ break;
+ }
+
+ EditorGUIUtility.wideMode = isWideMode;
+ }
+ }
+
+ private void AddRule()
+ {
+ SerializedProperty rule;
+ switch (currentBaseAction.AxisConstraint)
+ {
+ case AxisType.Digital:
+ inputActionRulesDigital.arraySize += 1;
+ rule = inputActionRulesDigital.GetArrayElementAtIndex(inputActionRulesDigital.arraySize - 1);
+ rule.FindPropertyRelative("criteria").boolValue = currentBoolCriteria;
+ break;
+ case AxisType.SingleAxis:
+ inputActionRulesSingleAxis.arraySize += 1;
+ rule = inputActionRulesSingleAxis.GetArrayElementAtIndex(inputActionRulesSingleAxis.arraySize - 1);
+ rule.FindPropertyRelative("criteria").floatValue = currentSingleAxisCriteria;
+ break;
+ case AxisType.DualAxis:
+ inputActionRulesDualAxis.arraySize += 1;
+ rule = inputActionRulesDualAxis.GetArrayElementAtIndex(inputActionRulesDualAxis.arraySize - 1);
+ rule.FindPropertyRelative("criteria").vector2Value = currentDualAxisCriteria;
+ break;
+ case AxisType.ThreeDofPosition:
+ inputActionRulesVectorAxis.arraySize += 1;
+ rule = inputActionRulesVectorAxis.GetArrayElementAtIndex(inputActionRulesVectorAxis.arraySize - 1);
+ rule.FindPropertyRelative("criteria").vector3Value = currentVectorCriteria;
+ break;
+ case AxisType.ThreeDofRotation:
+ inputActionRulesQuaternionAxis.arraySize += 1;
+ rule = inputActionRulesQuaternionAxis.GetArrayElementAtIndex(inputActionRulesQuaternionAxis.arraySize - 1);
+ rule.FindPropertyRelative("criteria").quaternionValue = currentQuaternionCriteria;
+ break;
+ case AxisType.SixDof:
+ inputActionRulesPoseAxis.arraySize += 1;
+ rule = inputActionRulesPoseAxis.GetArrayElementAtIndex(inputActionRulesPoseAxis.arraySize - 1);
+ var criteria = rule.FindPropertyRelative("criteria");
+ criteria.FindPropertyRelative("position").vector3Value = currentPoseCriteria.Position;
+ criteria.FindPropertyRelative("rotation").quaternionValue = currentPoseCriteria.Rotation;
+ break;
+ default:
+ Debug.LogError("Invalid Axis Constraint!");
+ return;
+ }
+
+ var baseAction = rule.FindPropertyRelative("baseAction");
+ var baseActionId = baseAction.FindPropertyRelative("id");
+ var baseActionDescription = baseAction.FindPropertyRelative("description");
+ var baseActionConstraint = baseAction.FindPropertyRelative("axisConstraint");
+
+ baseActionId.intValue = (int)currentBaseAction.Id;
+ baseActionDescription.stringValue = currentBaseAction.Description;
+ baseActionConstraint.intValue = (int)currentBaseAction.AxisConstraint;
+
+ var ruleAction = rule.FindPropertyRelative("ruleAction");
+ var ruleActionId = ruleAction.FindPropertyRelative("id");
+ var ruleActionDescription = ruleAction.FindPropertyRelative("description");
+ var ruleActionConstraint = ruleAction.FindPropertyRelative("axisConstraint");
+
+ ruleActionId.intValue = (int)currentRuleAction.Id;
+ ruleActionDescription.stringValue = currentRuleAction.Description;
+ ruleActionConstraint.intValue = (int)currentRuleAction.AxisConstraint;
+ }
+
+ private int RenderBaseInputAction(int baseActionId, out MixedRealityInputAction action, bool isLocked = false)
+ {
+ using (new EditorGUI.DisabledGroupScope(!isInitialized))
+ {
+ action = MixedRealityInputAction.None;
+ EditorGUILayout.BeginHorizontal();
+ EditorGUILayout.LabelField(BaseActionContent);
+ EditorGUI.BeginChangeCheck();
+
+ if (!isLocked)
+ {
+ baseActionId = EditorGUILayout.IntPopup(baseActionId, baseActionLabels, baseActionIds, GUILayout.ExpandWidth(true));
+ }
+
+ var inputActions = GetInputActions();
+ for (int i = 0; i < inputActions.Length; i++)
+ {
+ if (baseActionId == (int)inputActions[i].Id)
+ {
+ action = inputActions[i];
+ }
+ }
+
+ if (action != MixedRealityInputAction.None)
+ {
+ GetCompatibleActions(action);
+ }
+
+ if (isLocked)
+ {
+ EditorGUILayout.LabelField(action.Description, EditorStyles.boldLabel, GUILayout.ExpandWidth(true));
+ }
+
+ EditorGUILayout.EndHorizontal();
+ }
+
+ return baseActionId;
+ }
+
+ private int RenderRuleInputAction(int ruleActionId, out MixedRealityInputAction action)
+ {
+ action = MixedRealityInputAction.None;
+ EditorGUILayout.BeginHorizontal();
+ EditorGUILayout.LabelField(RuleActionContent, GUILayout.Width(128));
+ EditorGUI.BeginChangeCheck();
+ ruleActionId = EditorGUILayout.IntPopup(ruleActionId, ruleActionLabels, ruleActionIds, GUILayout.ExpandWidth(true));
+
+ var inputActions = GetInputActions();
+ for (int i = 0; i < inputActions.Length; i++)
+ {
+ if (ruleActionId == (int)inputActions[i].Id)
+ {
+ action = inputActions[i];
+ }
+ }
+
+ EditorGUILayout.EndHorizontal();
+ return ruleActionId;
+ }
+
+ private void RenderList(SerializedProperty list, bool[] foldouts)
+ {
+ for (int i = 0; i < list?.arraySize; i++)
+ {
+ var rule = list.GetArrayElementAtIndex(i);
+ var criteria = rule.FindPropertyRelative("criteria");
+
+ var baseAction = rule.FindPropertyRelative("baseAction");
+ var baseActionId = baseAction.FindPropertyRelative("id");
+ var baseActionDescription = baseAction.FindPropertyRelative("description");
+ var baseActionConstraint = baseAction.FindPropertyRelative("axisConstraint");
+
+ var ruleAction = rule.FindPropertyRelative("ruleAction");
+ var ruleActionId = ruleAction.FindPropertyRelative("id");
+ var ruleActionDescription = ruleAction.FindPropertyRelative("description");
+ var ruleActionConstraint = ruleAction.FindPropertyRelative("axisConstraint");
+
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ foldouts[i] = EditorGUILayout.Foldout(foldouts[i], new GUIContent($"{baseActionDescription.stringValue} -> {ruleActionDescription.stringValue}"), true);
+
+ if (GUILayout.Button(RuleMinusButtonContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ list.DeleteArrayElementAtIndex(i);
+ return;
+ }
+ }
+
+ if (foldouts[i])
+ {
+ EditorGUI.indentLevel++;
+
+ MixedRealityInputAction newBaseAction;
+ baseActionId.intValue = RenderBaseInputAction(baseActionId.intValue, out newBaseAction, true);
+ baseActionDescription.stringValue = newBaseAction.Description;
+ baseActionConstraint.intValue = (int)newBaseAction.AxisConstraint;
+
+ if (baseActionId.intValue == ruleActionId.intValue || newBaseAction == MixedRealityInputAction.None || baseActionConstraint.intValue != ruleActionConstraint.intValue)
+ {
+ criteria.Reset();
+ ruleActionId.intValue = (int)MixedRealityInputAction.None.Id;
+ ruleActionDescription.stringValue = MixedRealityInputAction.None.Description;
+ ruleActionConstraint.intValue = (int)MixedRealityInputAction.None.AxisConstraint;
+ }
+
+ RenderCriteriaField(newBaseAction, criteria);
+
+ MixedRealityInputAction newRuleAction;
+ ruleActionId.intValue = RenderRuleInputAction(ruleActionId.intValue, out newRuleAction);
+ ruleActionDescription.stringValue = newRuleAction.Description;
+ ruleActionConstraint.intValue = (int)newRuleAction.AxisConstraint;
+ EditorGUI.indentLevel--;
+ }
+
+ EditorGUILayout.Space();
+ }
+ }
+
+ private static MixedRealityInputAction[] GetInputActions()
+ {
+ if (!MixedRealityToolkit.IsInitialized ||
+ !MixedRealityToolkit.Instance.HasActiveProfile ||
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile == null ||
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile == null)
+ {
+ return System.Array.Empty();
+ }
+
+ return MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile.InputActions;
+ }
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionRulesInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionRulesInspector.cs.meta
new file mode 100644
index 0000000..2cdcb88
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionRulesInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: 088654758d8f4856ba3ca7c6b2515958
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionsProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionsProfileInspector.cs
new file mode 100644
index 0000000..1def38f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionsProfileInspector.cs
@@ -0,0 +1,111 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ [CustomEditor(typeof(MixedRealityInputActionsProfile))]
+ public class MixedRealityInputActionsProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private static readonly GUIContent MinusButtonContent = new GUIContent("-", "Remove Action");
+ private static readonly GUIContent AddButtonContent = new GUIContent("+ Add a New Action", "Add New Action");
+ private static readonly GUIContent ActionContent = new GUIContent("Action", "The Name of the Action.");
+ private static readonly GUIContent AxisConstraintContent = new GUIContent("Axis Constraint", "Optional Axis Constraint for this input source.");
+ private const string ProfileTitle = "Input Action Settings";
+ private const string ProfileDescription = "Input Actions are any/all actions your users will be able to make when interacting with your application.\n\n" +
+ "After defining all your actions, you can then wire up these actions to hardware sensors, controllers, and other input devices.";
+
+ private static Vector2 scrollPosition = Vector2.zero;
+
+ private SerializedProperty inputActionList;
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ inputActionList = serializedObject.FindProperty("inputActions");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ RenderList(inputActionList);
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.InputActionsProfile;
+ }
+
+ private void RenderList(SerializedProperty list)
+ {
+ using (new EditorGUILayout.VerticalScope())
+ {
+ if (InspectorUIUtility.RenderIndentedButton(AddButtonContent, EditorStyles.miniButton))
+ {
+ list.arraySize += 1;
+ var inputAction = list.GetArrayElementAtIndex(list.arraySize - 1);
+ var inputActionId = inputAction.FindPropertyRelative("id");
+ var inputActionDescription = inputAction.FindPropertyRelative("description");
+ var inputActionConstraint = inputAction.FindPropertyRelative("axisConstraint");
+ inputActionConstraint.intValue = 0;
+ inputActionDescription.stringValue = $"New Action {inputActionId.intValue = list.arraySize}";
+ }
+
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ var labelWidth = EditorGUIUtility.labelWidth;
+ EditorGUIUtility.labelWidth = 36f;
+ EditorGUILayout.LabelField(ActionContent, GUILayout.ExpandWidth(true));
+ EditorGUILayout.LabelField(AxisConstraintContent, GUILayout.Width(96f));
+ EditorGUILayout.LabelField(string.Empty, GUILayout.Width(24f));
+ EditorGUIUtility.labelWidth = labelWidth;
+ }
+
+ scrollPosition = EditorGUILayout.BeginScrollView(scrollPosition, RenderAsSubProfile ? GUILayout.Height(100f) : GUILayout.ExpandHeight(true));
+
+ for (int i = 0; i < list.arraySize; i++)
+ {
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ var previousLabelWidth = EditorGUIUtility.labelWidth;
+ EditorGUIUtility.labelWidth = 64f;
+ SerializedProperty inputAction = list.GetArrayElementAtIndex(i);
+ SerializedProperty inputActionDescription = inputAction.FindPropertyRelative("description");
+ var inputActionConstraint = inputAction.FindPropertyRelative("axisConstraint");
+ EditorGUILayout.PropertyField(inputActionDescription, GUIContent.none);
+ EditorGUILayout.PropertyField(inputActionConstraint, GUIContent.none, GUILayout.Width(96f));
+ EditorGUIUtility.labelWidth = previousLabelWidth;
+
+ if (GUILayout.Button(MinusButtonContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ list.DeleteArrayElementAtIndex(i);
+ }
+
+ }
+ }
+
+ EditorGUILayout.EndScrollView();
+ }
+ EditorGUILayout.Space();
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionsProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionsProfileInspector.cs.meta
new file mode 100644
index 0000000..5237281
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputActionsProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c68a38578e3a4ccea6f8aea6476c618d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputSystemProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputSystemProfileInspector.cs
new file mode 100644
index 0000000..2705325
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputSystemProfileInspector.cs
@@ -0,0 +1,229 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ ///
+ /// Class handles rendering inspector view of MixedRealityInputSystemProfile object
+ ///
+ [CustomEditor(typeof(MixedRealityInputSystemProfile))]
+ public class MixedRealityInputSystemProfileInspector : BaseDataProviderServiceInspector
+ {
+ private const string DataProviderErrorMsg = "The Mixed Reality Input System requires one or more data providers.";
+ private static readonly GUIContent AddProviderContent = new GUIContent("+ Add Data Provider", "Add Data Provider");
+ private static readonly GUIContent RemoveProviderContent = new GUIContent("-", "Remove Data Provider");
+
+ private static bool showDataProviders = false;
+ private const string ShowInputSystem_DataProviders_PreferenceKey = "ShowInputSystem_DataProviders_PreferenceKey";
+
+ private SerializedProperty focusProviderType;
+ private SerializedProperty focusQueryBufferSize;
+ private SerializedProperty raycastProviderType;
+ private SerializedProperty focusIndividualCompoundCollider;
+ private SerializedProperty shouldUseGraphicsRaycast;
+
+ private static bool showPointerProperties = false;
+ private const string ShowInputSystem_Pointers_PreferenceKey = "ShowInputSystem_Pointers_PreferenceKey";
+ private SerializedProperty pointerProfile;
+
+ private static bool showActionsProperties = false;
+ private const string ShowInputSystem_Actions_PreferenceKey = "ShowInputSystem_Actions_PreferenceKey";
+ private SerializedProperty inputActionsProfile;
+ private SerializedProperty inputActionRulesProfile;
+
+ private static bool showControllerProperties = false;
+ private const string ShowInputSystem_Controller_PreferenceKey = "ShowInputSystem_Controller_PreferenceKey";
+ private SerializedProperty enableControllerMapping;
+ private SerializedProperty controllerMappingProfile;
+ private SerializedProperty controllerVisualizationProfile;
+
+ private static bool showGestureProperties = false;
+ private const string ShowInputSystem_Gesture_PreferenceKey = "ShowInputSystem_Gesture_PreferenceKey";
+ private SerializedProperty gesturesProfile;
+
+ private static bool showSpeechCommandsProperties = false;
+ private const string ShowInputSystem_Speech_PreferenceKey = "ShowInputSystem_Speech_PreferenceKey";
+ private SerializedProperty speechCommandsProfile;
+
+ private static bool showHandTrackingProperties = false;
+ private const string ShowInputSystem_HandTracking_PreferenceKey = "ShowInputSystem_HandTracking_PreferenceKey";
+ private SerializedProperty handTrackingProfile;
+
+ private const string ProfileTitle = "Input System Settings";
+ private const string ProfileDescription = "The Input System Profile helps developers configure input for cross-platform applications.";
+
+ ///
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ focusProviderType = serializedObject.FindProperty("focusProviderType");
+ focusQueryBufferSize = serializedObject.FindProperty("focusQueryBufferSize");
+ raycastProviderType = serializedObject.FindProperty("raycastProviderType");
+ focusIndividualCompoundCollider = serializedObject.FindProperty("focusIndividualCompoundCollider");
+ shouldUseGraphicsRaycast = serializedObject.FindProperty("shouldUseGraphicsRaycast");
+ inputActionsProfile = serializedObject.FindProperty("inputActionsProfile");
+ inputActionRulesProfile = serializedObject.FindProperty("inputActionRulesProfile");
+ pointerProfile = serializedObject.FindProperty("pointerProfile");
+ gesturesProfile = serializedObject.FindProperty("gesturesProfile");
+ speechCommandsProfile = serializedObject.FindProperty("speechCommandsProfile");
+ controllerMappingProfile = serializedObject.FindProperty("controllerMappingProfile");
+ enableControllerMapping = serializedObject.FindProperty("enableControllerMapping");
+ controllerVisualizationProfile = serializedObject.FindProperty("controllerVisualizationProfile");
+ handTrackingProfile = serializedObject.FindProperty("handTrackingProfile");
+ }
+
+ ///
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target))
+ {
+ return;
+ }
+
+ bool changed = false;
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUI.BeginChangeCheck();
+ EditorGUILayout.PropertyField(focusProviderType);
+ EditorGUILayout.PropertyField(focusQueryBufferSize);
+ EditorGUILayout.PropertyField(raycastProviderType);
+ EditorGUILayout.PropertyField(focusIndividualCompoundCollider);
+ EditorGUILayout.PropertyField(shouldUseGraphicsRaycast);
+ changed |= EditorGUI.EndChangeCheck();
+
+ EditorGUILayout.Space();
+
+ bool isSubProfile = RenderAsSubProfile;
+ if (!isSubProfile)
+ {
+ EditorGUI.indentLevel++;
+ }
+
+ RenderFoldout(ref showDataProviders, "Input Data Providers", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ changed |= RenderDataProviderList(AddProviderContent, RemoveProviderContent, DataProviderErrorMsg);
+ }
+ }, ShowInputSystem_DataProviders_PreferenceKey);
+
+ RenderFoldout(ref showPointerProperties, "Pointers", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ changed |= RenderProfile(pointerProfile, typeof(MixedRealityPointerProfile), true, false);
+ }
+ }, ShowInputSystem_Pointers_PreferenceKey);
+
+ RenderFoldout(ref showActionsProperties, "Input Actions", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ changed |= RenderProfile(inputActionsProfile, typeof(MixedRealityInputActionsProfile), true, false);
+ EditorGUILayout.Space();
+ EditorGUILayout.Space();
+ changed |= RenderProfile(inputActionRulesProfile, typeof(MixedRealityInputActionRulesProfile), true, false);
+ }
+ }, ShowInputSystem_Actions_PreferenceKey);
+
+ RenderFoldout(ref showControllerProperties, "Controllers", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ EditorGUILayout.PropertyField(enableControllerMapping);
+ changed |= RenderProfile(controllerMappingProfile, typeof(MixedRealityControllerMappingProfile), true, false);
+ EditorGUILayout.Space();
+ changed |= RenderProfile(controllerVisualizationProfile, null, true, false, typeof(IMixedRealityControllerVisualizer));
+ }
+ }, ShowInputSystem_Controller_PreferenceKey);
+
+ RenderFoldout(ref showGestureProperties, "Gestures", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ changed |= RenderProfile(gesturesProfile, typeof(MixedRealityGesturesProfile), true, false);
+ }
+ }, ShowInputSystem_Gesture_PreferenceKey);
+
+ RenderFoldout(ref showSpeechCommandsProperties, "Speech", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ changed |= RenderProfile(speechCommandsProfile, typeof(MixedRealitySpeechCommandsProfile), true, false);
+ }
+ }, ShowInputSystem_Speech_PreferenceKey);
+
+ RenderFoldout(ref showHandTrackingProperties, "Articulated Hand Tracking", () =>
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ changed |= RenderProfile(handTrackingProfile, typeof(MixedRealityHandTrackingProfile), true, false);
+ }
+ }, ShowInputSystem_HandTracking_PreferenceKey);
+
+ if (!isSubProfile)
+ {
+ EditorGUI.indentLevel--;
+ }
+
+ serializedObject.ApplyModifiedProperties();
+ }
+
+ if (changed && MixedRealityToolkit.IsInitialized)
+ {
+ EditorApplication.delayCall += () => MixedRealityToolkit.Instance.ResetConfiguration(MixedRealityToolkit.Instance.ActiveProfile);
+ }
+ }
+
+ ///
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile;
+ }
+
+ #region DataProvider Inspector Utilities
+
+ ///
+ protected override SerializedProperty GetDataProviderConfigurationList()
+ {
+ return serializedObject.FindProperty("dataProviderConfigurations");
+ }
+
+ ///
+ protected override ServiceConfigurationProperties GetDataProviderConfigurationProperties(SerializedProperty providerEntry)
+ {
+ return new ServiceConfigurationProperties()
+ {
+ componentName = providerEntry.FindPropertyRelative("componentName"),
+ componentType = providerEntry.FindPropertyRelative("componentType"),
+ providerProfile = providerEntry.FindPropertyRelative("deviceManagerProfile"),
+ runtimePlatform = providerEntry.FindPropertyRelative("runtimePlatform"),
+ };
+ }
+
+ ///
+ protected override IMixedRealityServiceConfiguration GetDataProviderConfiguration(int index)
+ {
+ MixedRealityInputSystemProfile profile = target as MixedRealityInputSystemProfile;
+ var configurations = (profile != null) ? profile.DataProviderConfigurations : null;
+ if (configurations != null && index >= 0 && index < configurations.Length)
+ {
+ return configurations[index];
+ }
+
+ return null;
+ }
+
+ #endregion
+ }
+}
\ No newline at end of file
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputSystemProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputSystemProfileInspector.cs.meta
new file mode 100644
index 0000000..e7f8af8
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityInputSystemProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: c098f43cbf0ed9a43ad84b590fafa459
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityMouseInputProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityMouseInputProfileInspector.cs
new file mode 100644
index 0000000..27a36b0
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityMouseInputProfileInspector.cs
@@ -0,0 +1,53 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using UnityEditor;
+
+namespace Microsoft.MixedReality.Toolkit.Input
+{
+ [CustomEditor(typeof(MixedRealityMouseInputProfile))]
+ public class MixedRealityMouseInputProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private const string ProfileTitle = "Mouse Input Settings";
+ private const string ProfileDescription = "Settings used to configure the behavior of mouse controllers.";
+
+ private SerializedProperty cursorSpeed;
+ private SerializedProperty wheelSpeed;
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+ cursorSpeed = serializedObject.FindProperty("cursorSpeed");
+ wheelSpeed = serializedObject.FindProperty("wheelSpeed");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+ EditorGUILayout.PropertyField(cursorSpeed);
+ EditorGUILayout.PropertyField(wheelSpeed);
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ if (!MixedRealityToolkit.IsInitialized || profile == null)
+ {
+ return false;
+ }
+
+ var mouseManager = MixedRealityToolkit.Instance.GetService(null, false);
+ return mouseManager != null && profile == mouseManager.ConfigurationProfile;
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityMouseInputProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityMouseInputProfileInspector.cs.meta
new file mode 100644
index 0000000..709ee53
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityMouseInputProfileInspector.cs.meta
@@ -0,0 +1,11 @@
+fileFormatVersion: 2
+guid: a4054f9166cdf974eb265d3209eb985d
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences: []
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityPointerProfileInspector.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityPointerProfileInspector.cs
new file mode 100644
index 0000000..37d5408
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityPointerProfileInspector.cs
@@ -0,0 +1,260 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Editor;
+using Microsoft.MixedReality.Toolkit.Utilities;
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Input.Editor
+{
+ [CustomEditor(typeof(MixedRealityPointerProfile))]
+ public class MixedRealityPointerProfileInspector : BaseMixedRealityToolkitConfigurationProfileInspector
+ {
+ private static readonly GUIContent ControllerTypeContent = new GUIContent("Controller Type", "The type of Controller this pointer will attach itself to at runtime.");
+ private static readonly GUIContent MinusButtonContent = new GUIContent("-", "Remove Pointer Option");
+ private static readonly GUIContent AddButtonContent = new GUIContent("+ Add a New Pointer Option", "Add Pointer Option");
+ private static readonly GUIContent GazeCursorPrefabContent = new GUIContent("Gaze Cursor Prefab");
+ private static readonly GUIContent UseEyeTrackingDataContent = new GUIContent("Use Eye Tracking Data");
+ private static readonly GUIContent RaycastLayerMaskContent = new GUIContent("Default Raycast LayerMasks");
+ private static readonly GUIContent PointerRaycastLayerMaskContent = new GUIContent("Pointer Raycast LayerMasks");
+
+#if UNITY_2019_3_OR_NEWER
+ private const string EnableGazeCapabilityContent = "To use eye tracking with UWP, the GazeInput capability needs to be set in the manifest." +
+ "\nPlease click the button below to set it in the Unity UWP Player Settings and check the Visual Studio appxmanifest capabilities to ensure it's enabled.";
+#endif // UNITY_2019_3_OR_NEWER
+
+ private const string ProfileTitle = "Pointer Settings";
+ private const string ProfileDescription = "Pointers attach themselves onto controllers as they are initialized.";
+
+ private SerializedProperty pointingExtent;
+ private SerializedProperty defaultRaycastLayerMasks;
+ private static bool showPointerOptionProperties = true;
+ private SerializedProperty pointerOptions;
+
+ private SerializedProperty debugDrawPointingRays;
+ private SerializedProperty debugDrawPointingRayColors;
+ private SerializedProperty gazeCursorPrefab;
+ private SerializedProperty gazeProviderType;
+ private SerializedProperty useHeadGazeOverride;
+ private SerializedProperty useEyeTrackingDataWhenAvailable;
+
+ private static bool showGazeProviderProperties = true;
+ private UnityEditor.Editor gazeProviderEditor;
+
+ private SerializedProperty pointerMediator;
+ private SerializedProperty primaryPointerSelector;
+
+ protected override void OnEnable()
+ {
+ base.OnEnable();
+
+ pointingExtent = serializedObject.FindProperty("pointingExtent");
+ defaultRaycastLayerMasks = serializedObject.FindProperty("pointingRaycastLayerMasks");
+ pointerOptions = serializedObject.FindProperty("pointerOptions");
+ debugDrawPointingRays = serializedObject.FindProperty("debugDrawPointingRays");
+ debugDrawPointingRayColors = serializedObject.FindProperty("debugDrawPointingRayColors");
+ gazeCursorPrefab = serializedObject.FindProperty("gazeCursorPrefab");
+ gazeProviderType = serializedObject.FindProperty("gazeProviderType");
+ useHeadGazeOverride = serializedObject.FindProperty("useHeadGazeOverride");
+ useEyeTrackingDataWhenAvailable = serializedObject.FindProperty("isEyeTrackingEnabled");
+ pointerMediator = serializedObject.FindProperty("pointerMediator");
+ primaryPointerSelector = serializedObject.FindProperty("primaryPointerSelector");
+ }
+
+ public override void OnInspectorGUI()
+ {
+ if (!RenderProfileHeader(ProfileTitle, ProfileDescription, target, true, BackProfileType.Input))
+ {
+ return;
+ }
+
+ using (new EditorGUI.DisabledGroupScope(IsProfileLock((BaseMixedRealityProfile)target)))
+ {
+ serializedObject.Update();
+
+ EditorGUILayout.Space();
+ EditorGUILayout.PropertyField(pointingExtent);
+ EditorGUILayout.PropertyField(defaultRaycastLayerMasks, RaycastLayerMaskContent, true);
+ EditorGUILayout.PropertyField(pointerMediator);
+ EditorGUILayout.PropertyField(primaryPointerSelector);
+
+ GUIStyle boldFoldout = new GUIStyle(EditorStyles.foldout) { fontStyle = FontStyle.Bold };
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Gaze Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.Space();
+ EditorGUILayout.PropertyField(gazeCursorPrefab, GazeCursorPrefabContent);
+ EditorGUILayout.PropertyField(gazeProviderType);
+ EditorGUILayout.PropertyField(useHeadGazeOverride);
+
+ EditorGUILayout.BeginHorizontal();
+ EditorGUILayout.PropertyField(useEyeTrackingDataWhenAvailable, UseEyeTrackingDataContent);
+ // Render a help link for getting started with eyetracking documentation
+ string helpURL = "https://docs.microsoft.com/windows/mixed-reality/mrtk-unity/features/input/eye-tracking/eye-tracking-basic-setup";
+ InspectorUIUtility.RenderDocumentationButton(helpURL);
+ EditorGUILayout.EndHorizontal();
+
+#if UNITY_2019_3_OR_NEWER
+ if (useEyeTrackingDataWhenAvailable.boolValue && MixedRealityOptimizeUtils.IsBuildTargetUWP() && !PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.GazeInput))
+ {
+ EditorGUILayout.HelpBox(EnableGazeCapabilityContent, MessageType.Warning);
+ if (InspectorUIUtility.RenderIndentedButton("Set GazeInput capability"))
+ {
+ PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.GazeInput, true);
+ }
+ }
+#endif // UNITY_2019_3_OR_NEWER
+
+ EditorGUILayout.Space();
+
+ showGazeProviderProperties = EditorGUILayout.Foldout(showGazeProviderProperties, "Gaze Provider Settings", true, boldFoldout);
+ if (showGazeProviderProperties && CameraCache.Main != null)
+ {
+ var gazeProvider = CameraCache.Main.GetComponent();
+ CreateCachedEditor((Object)gazeProvider, null, ref gazeProviderEditor);
+
+ // Provide a convenient way to toggle the gaze provider as enabled/disabled via editor
+ gazeProvider.Enabled = EditorGUILayout.Toggle("Enable Gaze Provider", gazeProvider.Enabled);
+
+ if (gazeProviderEditor != null)
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ // Draw out the rest of the Gaze Provider's settings
+ gazeProviderEditor.OnInspectorGUI();
+ }
+ }
+ }
+ }
+
+ EditorGUILayout.Space();
+ showPointerOptionProperties = EditorGUILayout.Foldout(showPointerOptionProperties, "Pointer Options", true, boldFoldout);
+
+ if (showPointerOptionProperties)
+ {
+ using (new EditorGUI.IndentLevelScope())
+ {
+ RenderPointerList(pointerOptions);
+ }
+ }
+
+ EditorGUILayout.Space();
+ EditorGUILayout.LabelField("Debug Settings", EditorStyles.boldLabel);
+ {
+ EditorGUILayout.PropertyField(debugDrawPointingRays);
+ EditorGUILayout.PropertyField(debugDrawPointingRayColors, true);
+ }
+
+ serializedObject.ApplyModifiedProperties();
+ }
+ }
+
+ protected override bool IsProfileInActiveInstance()
+ {
+ var profile = target as BaseMixedRealityProfile;
+ return MixedRealityToolkit.IsInitialized && profile != null &&
+ MixedRealityToolkit.Instance.HasActiveProfile &&
+ MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile != null &&
+ profile == MixedRealityToolkit.Instance.ActiveProfile.InputSystemProfile.PointerProfile;
+ }
+
+ private void RenderPointerList(SerializedProperty list)
+ {
+ if (InspectorUIUtility.RenderIndentedButton(AddButtonContent, EditorStyles.miniButton))
+ {
+ pointerOptions.arraySize += 1;
+
+ var newPointerOption = list.GetArrayElementAtIndex(list.arraySize - 1);
+ var controllerType = newPointerOption.FindPropertyRelative("controllerType");
+ var handedness = newPointerOption.FindPropertyRelative("handedness");
+ var prefab = newPointerOption.FindPropertyRelative("pointerPrefab");
+ var raycastLayerMask = newPointerOption.FindPropertyRelative("prioritizedLayerMasks");
+
+ // Reset new entry
+ controllerType.intValue = 0;
+ handedness.intValue = 0;
+ prefab.objectReferenceValue = null;
+ raycastLayerMask.arraySize = 0;
+ }
+
+ if (list == null || list.arraySize == 0)
+ {
+ EditorGUILayout.HelpBox("Create a new Pointer Option entry.", MessageType.Warning);
+ return;
+ }
+
+ bool anyPrefabChanged = false;
+
+ for (int i = 0; i < list.arraySize; i++)
+ {
+ using (new EditorGUILayout.VerticalScope(EditorStyles.helpBox))
+ {
+ Color prevColor = GUI.color;
+
+ var pointerOption = list.GetArrayElementAtIndex(i);
+ var controllerType = pointerOption.FindPropertyRelative("controllerType");
+ var handedness = pointerOption.FindPropertyRelative("handedness");
+ var prefab = pointerOption.FindPropertyRelative("pointerPrefab");
+ var prioritizedLayerMasks = pointerOption.FindPropertyRelative("prioritizedLayerMasks");
+
+ GameObject pointerPrefab = prefab.objectReferenceValue as GameObject;
+ IMixedRealityPointer pointer = pointerPrefab != null ? pointerPrefab.GetComponent() : null;
+
+ // Display an error if the prefab doesn't have a IMixedRealityPointer Component
+ if (pointer.IsNull())
+ {
+ InspectorUIUtility.DrawError($"The prefab associated with this pointer option needs an {typeof(IMixedRealityPointer).Name} component");
+ GUI.color = MixedRealityInspectorUtility.ErrorColor;
+ }
+
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ EditorGUILayout.PropertyField(prefab);
+ if (GUILayout.Button(MinusButtonContent, EditorStyles.miniButtonRight, GUILayout.Width(24f)))
+ {
+ list.DeleteArrayElementAtIndex(i);
+ break;
+ }
+ }
+
+ EditorGUILayout.PropertyField(controllerType, ControllerTypeContent);
+ EditorGUILayout.PropertyField(handedness);
+
+ // Ultimately sync the pointer prefab's value with the pointer option's
+ EditorGUI.BeginChangeCheck();
+ EditorGUILayout.PropertyField(prioritizedLayerMasks, PointerRaycastLayerMaskContent, true);
+ if (EditorGUI.EndChangeCheck() && pointer.IsNotNull())
+ {
+ Undo.RecordObject(pointerPrefab, "Sync Pointer Prefab");
+
+ int prioritizedLayerMasksCount = prioritizedLayerMasks.arraySize;
+ if (pointer.PrioritizedLayerMasksOverride?.Length != prioritizedLayerMasksCount)
+ {
+ pointer.PrioritizedLayerMasksOverride = new LayerMask[prioritizedLayerMasksCount];
+ }
+
+ for (int j = 0; j < prioritizedLayerMasksCount; j++)
+ {
+ pointer.PrioritizedLayerMasksOverride[j] = prioritizedLayerMasks.GetArrayElementAtIndex(j).intValue;
+ }
+
+ PrefabUtility.RecordPrefabInstancePropertyModifications(pointerPrefab);
+ EditorUtility.SetDirty(pointerPrefab);
+ anyPrefabChanged = true;
+ }
+
+ GUI.color = prevColor;
+ }
+ EditorGUILayout.Space();
+ }
+
+ if (anyPrefabChanged)
+ {
+ AssetDatabase.SaveAssets();
+ }
+ }
+ }
+}
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityPointerProfileInspector.cs.meta b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityPointerProfileInspector.cs.meta
new file mode 100644
index 0000000..96b848f
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityPointerProfileInspector.cs.meta
@@ -0,0 +1,12 @@
+fileFormatVersion: 2
+guid: 137cc53a3db84eb8a0e568d57b36c958
+MonoImporter:
+ externalObjects: {}
+ serializedVersion: 2
+ defaultReferences:
+ - logo: {instanceID: 0}
+ executionOrder: 0
+ icon: {instanceID: 0}
+ userData:
+ assetBundleName:
+ assetBundleVariant:
diff --git a/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityProfileCloneWindow.cs b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityProfileCloneWindow.cs
new file mode 100644
index 0000000..ef96cb1
--- /dev/null
+++ b/com.microsoft.mixedreality.toolkit.foundation/Core/Inspectors/Profiles/MixedRealityProfileCloneWindow.cs
@@ -0,0 +1,432 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+
+using Microsoft.MixedReality.Toolkit.Utilities.Editor;
+using System.Collections.Generic;
+using System.IO;
+using System.Reflection;
+using UnityEditor;
+using UnityEngine;
+
+namespace Microsoft.MixedReality.Toolkit.Editor
+{
+ public class MixedRealityProfileCloneWindow : EditorWindow
+ {
+ public enum ProfileCloneBehavior
+ {
+ ///
+ /// Use the existing reference.
+ ///
+ UseExisting,
+
+ ///
+ /// Create a clone of the sub-profile.
+ ///
+ CloneExisting,
+
+ ///
+ /// Manually select a profile.
+ ///
+ UseSubstitution,
+
+ ///
+ /// Set the reference to null.
+ ///
+ LeaveEmpty,
+ }
+
+ private struct SubProfileAction
+ {
+ public SubProfileAction(
+ ProfileCloneBehavior behavior,
+ SerializedProperty property,
+ Object substitutionReference,
+ System.Type profileType)
+ {
+ Behavior = behavior;
+ Property = property;
+ SubstitutionReference = substitutionReference;
+ ProfileType = profileType;
+ TargetFolder = null;
+
+ CloneName = (SubstitutionReference != null) ? "New " + SubstitutionReference.name : "New " + profileType.Name;
+ }
+
+ public ProfileCloneBehavior Behavior;
+ public SerializedProperty Property;
+ public string CloneName;
+ public Object SubstitutionReference;
+ public System.Type ProfileType;
+ internal Object TargetFolder;
+ }
+
+ private const string AdvancedModeKey = "MRTK_ProfileCloneWindow_AdvancedMode_Key";
+ private static bool AdvancedMode = false;
+ protected static string DefaultCustomProfileFolder => Path.Combine(MixedRealityToolkitFiles.MapModulePath(MixedRealityToolkitModuleType.Generated), "CustomProfiles");
+ private const string IsCustomProfileProperty = "isCustomProfile";
+ private static readonly Vector2 MinWindowSizeBasic = new Vector2(500, 180);
+ private const float SubProfileSizeMultiplier = 95f;
+ private static MixedRealityProfileCloneWindow cloneWindow;
+
+ private BaseMixedRealityProfile parentProfile;
+ private BaseMixedRealityProfile childProfile;
+ private SerializedProperty childProperty;
+ private SerializedObject childSerializedObject;
+ private Object targetFolder;
+ private Object selectionTarget;
+ private string childProfileTypeName;
+ private string childProfileAssetName;
+ private List subProfileActions = new List();
+
+ public static void OpenWindow(BaseMixedRealityProfile parentProfile, BaseMixedRealityProfile childProfile, SerializedProperty childProperty, Object selectionTarget = null)
+ {
+ if (cloneWindow != null)
+ {
+ cloneWindow.Close();
+ }
+
+ cloneWindow = GetWindow(true, "Clone Profile", true);
+ cloneWindow.Initialize(parentProfile, childProfile, childProperty, selectionTarget);
+ cloneWindow.Show(true);
+ }
+
+ private void Initialize(BaseMixedRealityProfile parentProfile, BaseMixedRealityProfile childProfile, SerializedProperty childProperty, Object selectionTarget)
+ {
+ this.childProperty = childProperty;
+ this.parentProfile = parentProfile;
+ this.childProfile = childProfile;
+ this.selectionTarget = selectionTarget;
+
+ childSerializedObject = new SerializedObject(childProfile);
+ childProfileTypeName = childProfile.GetType().Name;
+ childProfileAssetName = "New " + childProfileTypeName;
+
+ // Find all the serialized properties for sub-profiles
+ SerializedProperty iterator = childSerializedObject.GetIterator();
+ System.Type basePropertyType = typeof(BaseMixedRealityProfile);
+
+ while (iterator.Next(true))
+ {
+ SerializedProperty subProfileProperty = childSerializedObject.FindProperty(iterator.name);
+
+ if (subProfileProperty == null)
+ {
+ continue;
+ }
+
+ if (!subProfileProperty.type.Contains("PPtr<$")) // Not an object reference type
+ {
+ continue;
+ }
+
+ string subProfileTypeName = subProfileProperty.type.Replace("PPtr<$", string.Empty).Replace(">", string.Empty).Trim();
+ System.Type subProfileType = FindProfileType(subProfileTypeName);
+ if (subProfileType == null)
+ {
+ continue;
+ }
+
+ if (!basePropertyType.IsAssignableFrom(subProfileType))
+ {
+ continue;
+ }
+
+ subProfileActions.Add(new SubProfileAction(
+ ProfileCloneBehavior.UseExisting,
+ subProfileProperty,
+ subProfileProperty.objectReferenceValue,
+ subProfileType));
+ }
+
+ cloneWindow.maxSize = MinWindowSizeBasic;
+
+ targetFolder = EnsureTargetFolder(targetFolder, false);
+ }
+
+ private void OnGUI()
+ {
+ if (cloneWindow == null || childProfile == null)
+ {
+ Close();
+ return;
+ }
+
+ EditorGUILayout.BeginVertical(EditorStyles.helpBox);
+ EditorGUILayout.ObjectField("Cloning profile", childProfile, typeof(BaseMixedRealityProfile), false);
+ if (parentProfile != null)
+ { // Only show this if we're initiating this from a parent profile
+ EditorGUILayout.ObjectField("from parent profile", parentProfile, typeof(BaseMixedRealityProfile), false);
+ }
+ EditorGUILayout.EndVertical();
+ EditorGUILayout.Space();
+
+ if (subProfileActions.Count > 0)
+ {
+ AdvancedMode = EditorGUILayout.Foldout(SessionState.GetBool(AdvancedModeKey, false), "Advanced Options", true, MixedRealityStylesUtility.BoldFoldoutStyle);
+ SessionState.SetBool(AdvancedModeKey, AdvancedMode);
+
+ if (AdvancedMode)
+ {
+ EditorGUILayout.HelpBox("This profile has sub-profiles. By default your clone will reference the existing profiles. If you want to specify a different profile, or if you want to clone the sub-profile, use the options below.", MessageType.Info);
+
+ EditorGUILayout.BeginVertical(EditorStyles.helpBox);
+
+ for (int i = 0; i < subProfileActions.Count; i++)
+ {
+ GUI.color = Color.white;
+ EditorGUILayout.Space();
+
+ SubProfileAction action = subProfileActions[i];
+
+ action.Behavior = (ProfileCloneBehavior)EditorGUILayout.EnumPopup(action.Property.displayName, action.Behavior);
+
+ switch (action.Behavior)
+ {
+ case ProfileCloneBehavior.UseExisting:
+ GUI.color = Color.Lerp(Color.white, Color.clear, 0.5f);
+ EditorGUILayout.ObjectField("Existing", action.Property.objectReferenceValue, action.ProfileType, false);
+ break;
+
+ case ProfileCloneBehavior.UseSubstitution:
+ action.SubstitutionReference = EditorGUILayout.ObjectField("Substitution", action.SubstitutionReference, action.ProfileType, false);
+ break;
+
+ case ProfileCloneBehavior.CloneExisting:
+ if (action.Property.objectReferenceValue == null)
+ {
+ EditorGUILayout.LabelField("Can't clone profile - none is set.");
+ }
+ else
+ {
+ action.CloneName = EditorGUILayout.TextField("Clone name", action.CloneName);
+ }
+ using (new EditorGUILayout.HorizontalScope())
+ {
+ if (action.TargetFolder == null)
+ {
+ action.TargetFolder = targetFolder;
+ }
+ action.TargetFolder = EditorGUILayout.ObjectField("Target Folder", action.TargetFolder, typeof(DefaultAsset), false);
+ if (GUILayout.Button("Put in original folder", EditorStyles.miniButton, GUILayout.MaxWidth(120)))
+ {
+ string profilePath = AssetDatabase.GetAssetPath(action.Property.objectReferenceValue);
+ action.TargetFolder = AssetDatabase.LoadAssetAtPath