// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using System; using UnityEditor; using UnityEngine; namespace Microsoft.MixedReality.WebRTC.Unity.Editor { /// /// Inspector editor for . /// [CustomEditor(typeof(WebcamSource))] [CanEditMultipleObjects] public class WebcamSourceEditor : UnityEditor.Editor { SerializedProperty _enableMixedRealityCapture; SerializedProperty _enableMrcRecordingIndicator; SerializedProperty _formatMode; SerializedProperty _videoProfileId; SerializedProperty _videoProfileKind; SerializedProperty _constraints; SerializedProperty _width; SerializedProperty _height; SerializedProperty _framerate; SerializedProperty _videoStreamStarted; SerializedProperty _videoStreamStopped; GUIContent _anyContent; float _anyWidth; float _unitWidth; int _prevWidth = 640; int _prevHeight = 480; double _prevFramerate = 30.0; VideoProfileKind _prevVideoProfileKind = VideoProfileKind.VideoConferencing; string _prevVideoProfileId = ""; /// /// Helper enumeration for commonly used video codecs. /// The enum names must match exactly the standard SDP naming. /// See https://en.wikipedia.org/wiki/RTP_audio_video_profile for reference. /// enum SdpVideoCodecs { /// /// Do not force any codec, let WebRTC decide. /// None, /// /// Try to use H.264 if available. /// H264, /// /// Try to use VP8 if available. /// VP8, /// /// Try to use VP9 if available. /// VP9, /// /// Try to use the given codec if available. /// Custom } void OnEnable() { _enableMixedRealityCapture = serializedObject.FindProperty("EnableMixedRealityCapture"); _enableMrcRecordingIndicator = serializedObject.FindProperty("EnableMRCRecordingIndicator"); _formatMode = serializedObject.FindProperty("FormatMode"); _videoProfileId = serializedObject.FindProperty("VideoProfileId"); _videoProfileKind = serializedObject.FindProperty("VideoProfileKind"); _constraints = serializedObject.FindProperty("Constraints"); _width = _constraints.FindPropertyRelative("width"); _height = _constraints.FindPropertyRelative("height"); _framerate = _constraints.FindPropertyRelative("framerate"); _videoStreamStarted = serializedObject.FindProperty("VideoStreamStarted"); _videoStreamStopped = serializedObject.FindProperty("VideoStreamStopped"); _anyContent = new GUIContent("(any)"); _anyWidth = -1f; // initialized later _unitWidth = -1f; // initialized later } /// /// Override implementation of Editor.OnInspectorGUI /// to draw the inspector GUI for the currently selected . /// public override void OnInspectorGUI() { // CalcSize() can only be called inside a GUI method if (_anyWidth < 0) _anyWidth = GUI.skin.label.CalcSize(_anyContent).x; if (_unitWidth < 0) _unitWidth = GUI.skin.label.CalcSize(new GUIContent("fps")).x; serializedObject.Update(); if (!PlayerSettings.WSA.GetCapability(PlayerSettings.WSACapability.WebCam)) { EditorGUILayout.HelpBox("The UWP player is missing the WebCam capability. The WebcamSource component will not function correctly." + " Add the WebCam capability in Project Settings > Player > UWP > Publishing Settings > Capabilities.", MessageType.Error); if (GUILayout.Button("Open Player Settings")) { SettingsService.OpenProjectSettings("Project/Player"); } if (GUILayout.Button("Add WebCam Capability")) { PlayerSettings.WSA.SetCapability(PlayerSettings.WSACapability.WebCam, true); } } GUILayout.Space(10); EditorGUILayout.LabelField("Video capture", EditorStyles.boldLabel); EditorGUILayout.PropertyField(_formatMode, new GUIContent("Capture format", "Decide how to obtain the constraints used to select the best capture format.")); if ((LocalVideoSourceFormatMode)_formatMode.intValue == LocalVideoSourceFormatMode.Manual) { using (new EditorGUI.IndentLevelScope()) { EditorGUILayout.LabelField("General constraints (all platforms)"); using (new EditorGUI.IndentLevelScope()) { OptionalIntField(_width, ref _prevWidth, new GUIContent("Width", "Only consider capture formats with the specified width."), new GUIContent("px", "Pixels")); OptionalIntField(_height, ref _prevHeight, new GUIContent("Height", "Only consider capture formats with the specified height."), new GUIContent("px", "Pixels")); OptionalDoubleField(_framerate, ref _prevFramerate, new GUIContent("Framerate", "Only consider capture formats with the specified framerate."), new GUIContent("fps", "Frames per second")); } EditorGUILayout.LabelField("UWP constraints"); using (new EditorGUI.IndentLevelScope()) { OptionalEnumField(_videoProfileKind, VideoProfileKind.Unspecified, ref _prevVideoProfileKind, new GUIContent("Video profile kind", "Only consider capture formats associated with the specified video profile kind.")); OptionalTextField(_videoProfileId, ref _prevVideoProfileId, new GUIContent("Video profile ID", "Only consider capture formats associated with the specified video profile.")); if ((_videoProfileKind.intValue != (int)VideoProfileKind.Unspecified) && (_videoProfileId.stringValue.Length > 0)) { EditorGUILayout.HelpBox("Video profile ID is already unique. Specifying also a video kind over-constrains the selection algorithm and can decrease the chances of finding a matching video profile. It is recommended to select either a video profile kind, or a video profile ID.", MessageType.Warning); } } } } _enableMixedRealityCapture.boolValue = EditorGUILayout.ToggleLeft("Enable Mixed Reality Capture (MRC)", _enableMixedRealityCapture.boolValue); if (_enableMixedRealityCapture.boolValue) { using (var scope = new EditorGUI.IndentLevelScope()) { _enableMrcRecordingIndicator.boolValue = EditorGUILayout.ToggleLeft("Show recording indicator in device", _enableMrcRecordingIndicator.boolValue); if (!PlayerSettings.virtualRealitySupported) { EditorGUILayout.HelpBox("Mixed Reality Capture can only work in exclusive-mode apps. XR support must be enabled in Project Settings > Player > XR Settings > Virtual Reality Supported, and the project then saved to disk.", MessageType.Error); if (GUILayout.Button("Enable XR support")) { PlayerSettings.virtualRealitySupported = true; } } } } GUILayout.Space(10); EditorGUILayout.PropertyField(_videoStreamStarted); EditorGUILayout.PropertyField(_videoStreamStopped); serializedObject.ApplyModifiedProperties(); } /// /// ToggleLeft control associated with a given SerializedProperty, to enable automatic GUI /// handlings like Prefab revert menu. /// /// The boolean property associated with the control. /// The label to display next to the toggle control. private void ToggleLeft(SerializedProperty property, GUIContent label) { var rect = EditorGUILayout.GetControlRect(); using (new EditorGUI.PropertyScope(rect, label, property)) { property.boolValue = EditorGUI.ToggleLeft(rect, label, property.boolValue); } } /// /// IntField with optional toggle associated with a given SerializedProperty, to enable /// automatic GUI handlings like Prefab revert menu. /// /// Valid integer values are any non-zero positive integer. Any negative or zero value /// is considered invalid, and means that the value is considered as not set, which shows /// up as an unchecked left toggle widget. /// /// To enforce a valid value when the toggle control is checked by the user, a default valid /// value is provided . For UI consistency, the last selected /// valid value is returned in , to allow toggling the field /// ON and OFF without losing the valid value it previously had. /// /// The integer property associated with the control. /// /// Default value if the property value is invalid (negative or zero). /// Assigned the new value on return if valid. /// /// The label to display next to the toggle control. /// The label indicating the unit of the value. private void OptionalIntField(SerializedProperty intProperty, ref int lastValidValue, GUIContent label, GUIContent unitLabel) { if (lastValidValue <= 0) { throw new ArgumentOutOfRangeException("Default value cannot be invalid."); } using (new EditorGUILayout.HorizontalScope()) { var rect = EditorGUILayout.GetControlRect(); using (new EditorGUI.PropertyScope(rect, label, intProperty)) { bool hadValidValue = (intProperty.intValue > 0); bool needsValidValue = EditorGUI.ToggleLeft(rect, label, hadValidValue); int newValue = intProperty.intValue; if (needsValidValue) { // Force a valid value, otherwise the edit field won't show up if (newValue <= 0) { newValue = lastValidValue; } // Make updating the value of the serialized property delayed to allow overriding the // value the user will input before it's assigned to the property, for validation. newValue = EditorGUILayout.DelayedIntField(newValue); if (newValue < 0) { newValue = 0; } } else { // Force invalid value for consistency, otherwise this breaks Prefab revert newValue = 0; } intProperty.intValue = newValue; if (newValue > 0) { GUILayout.Label(unitLabel, GUILayout.Width(_unitWidth)); // Save valid value as new default. This allows toggling the toggle widget ON and OFF // without losing the value previously input. This works only while the inspector is // alive, that is while the object is select, but is better than nothing. lastValidValue = newValue; } else { GUILayout.Label(_anyContent, GUILayout.Width(_anyWidth)); } } } } /// /// DoubleField with optional toggle associated with a given SerializedProperty, to enable /// automatic GUI handlings like Prefab revert menu. /// /// Valid doubles values are any non-zero positive doubles. Any negative or zero value /// is considered invalid, and means that the value is considered as not set, which shows /// up as an unchecked left toggle widget. /// /// To enforce a valid value when the toggle control is checked by the user, a default valid /// value is provided . For UI consistency, the last selected /// valid value is returned in , to allow toggling the field /// ON and OFF without losing the valid value it previously had. /// /// The double property associated with the control. /// /// Default value if the property value is invalid (negative or zero). /// Assigned the new value on return if valid. /// /// The label to display next to the toggle control. /// The label indicating the unit of the value. private void OptionalDoubleField(SerializedProperty doubleProperty, ref double lastValidValue, GUIContent label, GUIContent unitLabel) { if (lastValidValue <= 0.0) { throw new ArgumentOutOfRangeException("Default value cannot be invalid."); } using (new EditorGUILayout.HorizontalScope()) { var rect = EditorGUILayout.GetControlRect(); using (new EditorGUI.PropertyScope(rect, label, doubleProperty)) { bool hadValidValue = (doubleProperty.doubleValue > 0.0); bool needsValidValue = EditorGUI.ToggleLeft(rect, label, hadValidValue); double newValue = doubleProperty.doubleValue; if (needsValidValue) { // Force a valid value, otherwise the edit field won't show up if (newValue <= 0.0) { newValue = lastValidValue; } // Make updating the value of the serialized property delayed to allow overriding the // value the user will input before it's assigned to the property, for validation. newValue = EditorGUILayout.DelayedDoubleField(newValue); if (newValue < 0.0) { newValue = 0.0; } } else { // Force invalid value for consistency, otherwise this breaks Prefab revert newValue = 0.0; } doubleProperty.doubleValue = newValue; if (newValue > 0.0) { GUILayout.Label(unitLabel, GUILayout.Width(_unitWidth)); // Save valid value as new default. This allows toggling the toggle widget ON and OFF // without losing the value previously input. This works only while the inspector is // alive, that is while the object is select, but is better than nothing. lastValidValue = newValue; } else { GUILayout.Label(_anyContent, GUILayout.Width(_anyWidth)); } } } } /// /// Helper to convert an enum to its integer value. /// /// The enum type. /// The enum value. /// The integer value associated with . public static int EnumToInt(TValue value) where TValue : Enum => (int)(object)value; /// /// Helper to convert an integer to its enum value. /// /// The enum type. /// The integer value. /// The enum value whose integer value is . public static TValue IntToEnum(int value) where TValue : Enum => (TValue)(object)value; /// /// EnumPopup with optional toggle associated with a given SerializedProperty, to enable /// automatic GUI handlings like Prefab revert menu. /// /// Valid enum values are any value different from . A value of /// is considered invalid, and means that the value is considered as /// not set, which shows up as an unchecked left toggle widget. /// /// To enforce a valid value when the toggle control is checked by the user, a default valid value /// is provided which must be different from . /// For UI consistency, the last selected valid value is returned in , /// to allow toggling the field ON and OFF without losing the valid value it previously had. /// /// The enum property associated with the control. /// Value considered to be "invalid", which deselects the toggle control. /// /// Default value if the property value is not . /// Assigned the new value on return if not . /// /// The label to display next to the toggle control. private void OptionalEnumField(SerializedProperty enumProperty, T nilValue, ref T lastValidValue, GUIContent label) where T : Enum { if (nilValue.CompareTo(lastValidValue) == 0) { throw new ArgumentOutOfRangeException("Default value cannot be invalid."); } using (new EditorGUILayout.HorizontalScope()) { var rect = EditorGUILayout.GetControlRect(); using (new EditorGUI.PropertyScope(rect, label, enumProperty)) { bool hadValidValue = (enumProperty.intValue != EnumToInt(nilValue)); bool needsValidValue = EditorGUI.ToggleLeft(rect, label, hadValidValue); T newValue = IntToEnum(enumProperty.intValue); if (needsValidValue) { // Force a valid value, otherwise the popup control won't show up if (newValue.CompareTo(nilValue) == 0) { newValue = lastValidValue; } newValue = (T)EditorGUILayout.EnumPopup(newValue); } else { // Force invalid value for consistency, otherwise this breaks Prefab revert newValue = nilValue; } enumProperty.intValue = EnumToInt(newValue); if (newValue.CompareTo(nilValue) != 0) { // Save valid value as new default. This allows toggling the toggle widget ON and OFF // without losing the value previously input. This works only while the inspector is // alive, that is while the object is select, but is better than nothing. lastValidValue = newValue; } else { GUILayout.Label(_anyContent, GUILayout.Width(_anyWidth)); } } } } /// /// TextField with optional toggle associated with a given SerializedProperty, to enable /// automatic GUI handlings like Prefab revert menu. /// /// Valid string values are any non-empty non-space-only string. Any empty string or string /// made up of only spaces is considered invalid, and means that the value is considered as /// not set, which shows up as an unchecked left toggle widget. /// /// To enforce a valid value when the toggle control is checked by the user, a default valid /// value is provided . For UI consistency, the last selected /// valid value is returned in , to allow toggling the field /// ON and OFF without losing the valid value it previously had. /// /// The string property associated with the control. /// /// Default value if the property value null or whitespace. /// Assigned the new value on return if valid. /// /// The label to display next to the toggle control. private void OptionalTextField(SerializedProperty stringProperty, ref string lastValidValue, GUIContent label) { if (string.IsNullOrWhiteSpace(lastValidValue)) { throw new ArgumentOutOfRangeException("Default value cannot be invalid."); } using (new EditorGUILayout.HorizontalScope()) { var rect = EditorGUILayout.GetControlRect(); using (new EditorGUI.PropertyScope(rect, label, stringProperty)) { bool hadValidValue = !string.IsNullOrWhiteSpace(stringProperty.stringValue); bool needsValidValue = EditorGUI.ToggleLeft(rect, label, hadValidValue); string newValue = stringProperty.stringValue; if (needsValidValue) { // Force a valid value, otherwise the edit field won't show up if (string.IsNullOrWhiteSpace(newValue)) { newValue = lastValidValue; } // Make updating the value of the serialized property delayed to allow overriding the // value the user will input before it's assigned to the property, for validation. newValue = EditorGUILayout.DelayedTextField(newValue); if (string.IsNullOrWhiteSpace(newValue)) { newValue = string.Empty; } } else { // Force invalid value for consistency, otherwise this breaks Prefab revert newValue = string.Empty; } stringProperty.stringValue = newValue; if (!string.IsNullOrWhiteSpace(newValue)) { // Save valid value as new default. This allows toggling the toggle widget ON and OFF // without losing the value previously input. This works only while the inspector is // alive, that is while the object is select, but is better than nothing. lastValidValue = newValue; } else { GUILayout.Label(_anyContent, GUILayout.Width(_anyWidth)); } } } } } }