using UnityEngine;
using UnityEditor;
using System;
using System.Collections.Generic;
using System.IO;
namespace Cinemachine.Editor
{
///
/// Collection of tools and helpers for drawing inspectors
///
public class InspectorUtility
{
/// Put multiple properties on a single inspector line, with
/// optional label overrides. Passing null as a label (or sublabel) override will
/// cause the property's displayName to be used as a label. For no label at all,
/// pass GUIContent.none.
/// Rect in which to draw
/// Main label
/// Properties to place on the line
/// Sublabels for the properties
public static void MultiPropertyOnLine(
Rect rect,
GUIContent label,
SerializedProperty[] props, GUIContent[] subLabels)
{
if (props == null || props.Length == 0)
return;
const int hSpace = 2;
int indentLevel = EditorGUI.indentLevel;
float labelWidth = EditorGUIUtility.labelWidth;
float totalSubLabelWidth = 0;
int numBoolColumns = 0;
List actualLabels = new List();
for (int i = 0; i < props.Length; ++i)
{
GUIContent sublabel = new GUIContent(props[i].displayName, props[i].tooltip);
if (subLabels != null && subLabels.Length > i && subLabels[i] != null)
sublabel = subLabels[i];
actualLabels.Add(sublabel);
totalSubLabelWidth += GUI.skin.label.CalcSize(sublabel).x;
if (i > 0)
totalSubLabelWidth += hSpace;
// Special handling for toggles, or it looks stupid
if (props[i].propertyType == SerializedPropertyType.Boolean)
{
totalSubLabelWidth += rect.height;
++numBoolColumns;
}
}
float subFieldWidth = rect.width - labelWidth - totalSubLabelWidth;
float numCols = props.Length - numBoolColumns;
float colWidth = numCols == 0 ? 0 : subFieldWidth / numCols;
// Main label. If no first sublabel, then main label must take on that
// role, for mouse dragging value-scrolling support
int subfieldStartIndex = 0;
if (label == null)
label = new GUIContent(props[0].displayName, props[0].tooltip);
if (actualLabels[0] != GUIContent.none)
rect = EditorGUI.PrefixLabel(rect, label);
else
{
rect.width = labelWidth + colWidth;
EditorGUI.PropertyField(rect, props[0], label);
rect.x += rect.width + hSpace;
subfieldStartIndex = 1;
}
for (int i = subfieldStartIndex; i < props.Length; ++i)
{
EditorGUI.indentLevel = 0;
EditorGUIUtility.labelWidth = GUI.skin.label.CalcSize(actualLabels[i]).x;
if (props[i].propertyType == SerializedPropertyType.Boolean)
{
rect.width = EditorGUIUtility.labelWidth + rect.height;
props[i].boolValue = EditorGUI.ToggleLeft(rect, actualLabels[i], props[i].boolValue);
}
else
{
rect.width = EditorGUIUtility.labelWidth + colWidth;
EditorGUI.PropertyField(rect, props[i], actualLabels[i]);
}
rect.x += rect.width + hSpace;
}
EditorGUIUtility.labelWidth = labelWidth;
EditorGUI.indentLevel = indentLevel;
}
///
/// Normalize a curve so that each of X and Y axes ranges from 0 to 1
///
/// Curve to normalize
/// The normalized curve
public static AnimationCurve NormalizeCurve(AnimationCurve curve)
{
return RuntimeUtility.NormalizeCurve(curve, true, true);
}
///
/// Remove the "Cinemachine" prefix, then call the standard Unity Nicify.
///
/// The name to nicify
/// The nicified name
public static string NicifyClassName(string name)
{
if (name.StartsWith("Cinemachine"))
name = name.Substring(11); // Trim the prefix
return ObjectNames.NicifyVariableName(name);
}
///
/// Add to a list all assets of a given type found in a given location
///
/// The asset type to look for
/// The list to add found assets to
/// The location in which to look. Path is relative to package root.
public static void AddAssetsFromPackageSubDirectory(
Type type, List assets, string path)
{
try
{
path = "/" + path;
var info = new DirectoryInfo(ScriptableObjectUtility.CinemachineInstallPath + path);
path = ScriptableObjectUtility.kPackageRoot + path + "/";
var fileInfo = info.GetFiles();
foreach (var file in fileInfo)
{
if (file.Extension != ".asset")
continue;
string name = path + file.Name;
ScriptableObject a = AssetDatabase.LoadAssetAtPath(name, type) as ScriptableObject;
if (a != null)
assets.Add(a);
}
}
catch
{
}
}
// Temporarily here
///
/// Creates a new GameObject.
///
/// Name to give the object.
/// Optional components to add.
/// The GameObject that was created.
[Obsolete("Use ObjectFactory.CreateGameObject(string name, params Type[] types) instead.")]
public static GameObject CreateGameObject(string name, params Type[] types)
{
return ObjectFactory.CreateGameObject(name, types);
}
private static int m_lastRepaintFrame;
///
/// Force a repaint of the Game View
///
/// Like it says
public static void RepaintGameView(UnityEngine.Object unused = null)
{
if (m_lastRepaintFrame == Time.frameCount)
return;
m_lastRepaintFrame = Time.frameCount;
EditorApplication.QueuePlayerLoopUpdate();
UnityEditorInternal.InternalEditorUtility.RepaintAllViews();
}
///
/// Try to get the name of the owning virtual camera oibject. If none then use
/// the object's name
///
///
///
internal static string GetVirtualCameraObjectName(SerializedProperty property)
{
// A little hacky here, as we favour virtual cameras...
var obj = property.serializedObject.targetObject;
GameObject go = obj as GameObject;
if (go == null)
{
var component = obj as Component;
if (component != null)
go = component.gameObject;
}
if (go != null)
{
var vcam = go.GetComponentInParent();
if (vcam != null)
return vcam.Name;
return go.name;
}
return obj.name;
}
}
}