This document outlines the recommended coding guidelines for the Mixed Reality Toolkit. The majority of these suggestions follow the recommended standards from MSDN.
All scripts posted to the MRTK should have the standard License header attached, exactly as shown below:
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License. See LICENSE in the project root for license information.
Any script files submitted without the license header will be rejected.
All public classes, structs, enums, functions, properties, fields posted to the MRTK should be described as to it's purpose and use, exactly as shown below:
/// <summary>
/// The Controller definition defines the Controller as defined by the SDK / Unity.
/// </summary>
public struct Controller
{
/// <summary>
/// The ID assigned to the Controller
/// </summary>
public string ID;
}
This ensures documentation is properly generated and disseminated for all classes, methods, and properties.
Any script files submitted without proper summary tags will be rejected.
The vNext structure adheres to a strict namespace culture of mapping the namespace 1-1 with the folder structure of the project. This ensures that classes are easy to discover and maintain. It also ensures the dependencies of any class are laid out in the beginning usings of the file.
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License. See LICENSE in the project root for license information.
namespace HoloToolkit.Unity.InputSystem
{
/// <summary>
/// The ButtonAction defines the set of actions exposed by a controller.
/// Denoting the available buttons / interactions that a controller supports.
/// </summary>
public enum ButtonAction
{
}
}
Absolutely no class / struct / enum or other definition should be entered in to the project without the appropriate namespace definition.
Please be sure to use 4 spaces instead of tabs when contributing to this project.
Additionally, ensure that spaces are added for conditional / loop functions like if / while / for
private Foo()
{
if(Bar==null) // <- no space between if and ()
{
DoThing();
}
while(true) // <- no space between while and ()
{
Do();
}
}
private Foo()
{
if (Bar==null)
{
DoThing();
}
while (true)
{
Do();
}
}
Always use PascalCase
for public / protected / virtual properties, and camelCase
for private properties and fields.
The only exception to this is for data structures that require the fields to be serialized by the
JsonUtility
.
public string myProperty; // <- Starts with a lower case letter
private string MyProperty; // <- Starts with an uppercase case letter
public string MyProperty;
protected string MyProperty;
private string myProperty;
Always declare an access modifier for all fields, properties and methods.
All Unity API Methods should be
private
by default, unless you need to override them in a derived class. In this caseprotected
should be used.
// No public / private access modifiers
void Foo() { }
void Bar() { }
private void Foo() { }
public void Bar() { }
protected virtual void FooBar() { }
Always use braces after each statement block, and place them on the next line.
private Foo()
{
if (Bar==null) // <- missing braces surrounding if action
DoThing();
else
DoTheOtherThing();
}
private Foo() { // <- Open bracket on same line
if (Bar==null) DoThing(); // <- if action on same line with no surrounding brackets
else DoTheOtherThing();
}
private Foo()
{
if (Bar==true)
{
DoThing();
}
else
{
DoTheOtherThing();
}
}
If the class, struct, or enum can be made private then it's okay to be included in the same file. This avoids compilation issues with Unity and ensures that proper code abstraction occurs, it also reduces conflicts and breaking changes when code needs to change.
public class MyClass
{
public struct MyStruct() { }
public enum MyEnumType() { }
public class MyNestedClass() { }
}
// Private references for use inside the class only
public class MyClass
{
private struct MyStruct() { }
private enum MyEnumType() { }
private class MyNestedClass() { }
}
// Public Struct / Enum definitions for use in your class. Try to make them generic for reuse.
MyStruct.cs
public struct MyStruct
{
public string Var1;
public string Var2;
}
MyEnumType.cs
public enum MuEnumType
{
Value1,
Value2 // <- note, no "," on last value to denote end of list.
}
MyClass.cs
public class MyClass
{
private MyStruct myStructReference;
private MyEnumType myEnumReference;
}
To ensure all Enum's are initialized correctly starting at 0, .NET gives you a tidy shortcut to automatically initialize the enum by just adding the first (starter) value.
E.G. Value 1 = 0 (Remaining values are not required)
public enum MyEnum
{
Value1, // <- no initializer
Value2,
Value3
}
public enum MyEnum
{
Value1 = 0,
Value2,
Value3
}
It is critical that if an Enum is likely to be extended in the future, to order defaults at the top of the Enum, this ensures Enum indexes are not affected with new additions.
public enum AudioFormat
{
PCM,
MP3,
Ogg,
None, // <- default value not at start
Other // <- anonymous value left to end of enum
}
/// <summary>
/// AudioFormat lists the supported / known formats of audio data
/// </summary>
public enum AudioFormat
{
/// <summary>
/// No specified format
/// </summary>
None = 0,
/// <summary>
/// Undefined format.
/// </summary>
Other,
/// <summary>
/// PCM
/// </summary>
PCM,
/// <summary>
/// MP3.
/// </summary>
MP3,
/// <summary>
/// Ogg Vorbis
/// </summary>
Ogg
}
If there is a possibility for an enum to require multiple states as a value, e.g. Handedness = Left & Right. Then the Enum needs to be decorated correctly with BitFlags to enable it to be used correctly
The Handedness.cs file has a concrete implementation for this
public enum MyEnum
{
None,
Left,
Right
}
[Flags]
public enum MyEnum
{
None = 1 << 0,
Left = 1 << 1,
Right = 1 << 2,
Both = Left | Right
}
Some of the target platforms of this project require us to take performance into consideration. With this in mind we should always be careful of allocating memory in frequently called code in tight update loops or algorithms.
Always use private fields and public properties if access to the field is needed from outside the class or struct. Be sure to co-locate the private field and the public property. This makes it easier to see, at a glance, what backs the property and that the field is modifiable by script.
If you need to have the ability to edit your field in the inspector, it's best practice to follow the rules for Encapsulation and serialize your backing field.
The only exception to this is for data structures that require the fields to be serialized by the
JsonUtility
, where a data class is required to have all public fields for the serialization to work.
public float MyValue;
/// <summary>
/// A value that is accessible within script only (it is not serialized in Unity for visibility in the Inspector)
/// </summary>
private float myValue;
/// <summary>
/// A value that is serialized in Unity for setting via the Inspector (not modifiable from other scripts)
/// </summary>
[Tooltip("A value that is serialized in Unity for setting via the Inspector (not modifiable from other scripts)")]
[SerializeField]
private float myValue;
private float myValue1;
private float myValue2;
public float MyValue1
{
get{ return myValue1; }
set{ myValue1 = value }
}
public float MyValue2
{
get{ return myValue2; }
set{ myValue2 = value }
}
/// <summary>
/// A value that is serialized in Unity for setting via the Inspector and is modifiable from other scripts
/// </summary>
[Tooltip("A value that is serialized in Unity for setting via the Inspector and is modifiable from other scripts")]
[SerializeField]
private float myValue; // <- Notice we co-located the backing field above our corresponding property.
public float MyValue
{
get{ return myValue; }
set{ myValue = value }
}
In some cases a foreach is required, e.g. when looping over an IEnumerable. But for performance benefit, avoid foreach when you can.
foreach(var item in items)
int length = items.length; // cache reference to list/array length
for (int i=0; i < length; i++)
With the HoloLens in mind, it's best to optimize for performance and cache references in the scene or prefab to limit runtime memory allocations.
void Update()
{
gameObject.GetComponent<Renderer>().Foo(Bar);
}
[SerializeField] // To enable setting the reference in the inspector.
private Renderer myRenderer;
private void Awake()
{
// If you didn't set it in the inspector, then we cache it on awake.
if (myRenderer == null)
{
myRenderer = gameObject.GetComponent<Renderer>();
}
}
private void Update()
{
myRenderer.Foo(Bar);
}
Unity will create a new material each time you use ".material", which will cause a memory leak if not cleaned up properly.
public class MyClass
{
void Update()
{
Material myMaterial = GetComponent<Renderer>().material;
myMaterial.SetColor("_Color", Color.White);
}
}
// Private references for use inside the class only
public class MyClass
{
private Material cachedMaterial;
private void Awake()
{
cachedMaterial = GetComponent<Renderer>().material;
}
void Update()
{
cachedMaterial.SetColor("_Color", Color.White);
}
private void OnDestroy()
{
Destroy(cachedMaterial);
}
}
Alternatively, use Unity's "SharedMaterial" property which does not create a new material each time it is referenced.
Use platform dependent compilation to ensure the Toolkit won't break the build on another platform
- Use
WINDOWS_UWP
in order to use UWP-specific, non-Unity APIs. This will prevent them from trying to run in the Editor or on unsupported platforms. This is equivalent toUNITY_WSA && !UNITY_EDITOR
and should be used in favor of. - Use
UNITY_WSA
to use UWP-specific Unity APIs, such as theUnityEngine.XR.WSA
namespace. This will run in the Editor when the platform is set to UWP, as well as in built UWP apps.
This chart can help you decide which #if
to use, depending on your use cases and the build settings you expect.
UWP IL2CPP | UWP .NET | Editor | |
---|---|---|---|
UNITY_EDITOR |
False | False | True |
UNITY_WSA |
True | True | True |
WINDOWS_UWP |
True | True | False |
UNITY_WSA && !UNITY_EDITOR |
True | True | False |
ENABLE_WINMD_SUPPORT |
True | True | False |
NETFX_CORE |
False | True | False |