我们创建了 Fontainebleau 演示来说明摄影photogrammetry流程和 LayeredLit 着色器的使用。
您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
 
 
 
 

226 行
9.4 KiB

// NOTE: If you are getting errors of the sort that say something like:
// "The type or namespace name `PostProcessing' does not exist in the namespace"
// it is because the PostProcessing v2 module has been removed from your project.
//
// To make the errors go away, you can either:
// 1 - Download PostProcessing V2 and install it into your project
// or
// 2 - Go into PlayerSettings and remove the define for UNITY_POST_PROCESSING_STACK_V2
//
using UnityEngine;
using UnityEditor;
using UnityEngine.Rendering.PostProcessing;
using UnityEditor.Rendering.PostProcessing;
using UnityEngine.SceneManagement;
using System.IO;
namespace Cinemachine.PostFX.Editor
{
[CustomEditor(typeof(CinemachinePostProcessing))]
public sealed class CinemachinePostProcessingEditor
: Cinemachine.Editor.BaseEditor<CinemachinePostProcessing>
{
SerializedProperty m_Profile;
SerializedProperty m_FocusTracksTarget;
SerializedProperty m_FocusOffset;
EffectListEditor m_EffectList;
GUIContent m_ProfileLabel;
void OnEnable()
{
Texture textue = Resources.Load("PostProcessLayer") as Texture;
m_ProfileLabel = new GUIContent("Profile", textue, "A reference to a profile asset");
m_FocusTracksTarget = FindProperty(x => x.m_FocusTracksTarget);
m_FocusOffset = FindProperty(x => x.m_FocusOffset);
m_Profile = FindProperty(x => x.m_Profile);
m_EffectList = new EffectListEditor(this);
RefreshEffectListEditor(Target.m_Profile);
}
void OnDisable()
{
if (m_EffectList != null)
m_EffectList.Clear();
}
void RefreshEffectListEditor(PostProcessProfile asset)
{
if (m_EffectList == null)
m_EffectList = new EffectListEditor(this);
m_EffectList.Clear();
if (asset != null)
m_EffectList.Init(asset, new SerializedObject(asset));
}
public override void OnInspectorGUI()
{
serializedObject.Update();
if (m_FocusTracksTarget.boolValue)
{
bool valid = false;
DepthOfField dof;
if (Target.m_Profile != null && Target.m_Profile.TryGetSettings(out dof))
valid = dof.enabled && dof.active && dof.focusDistance.overrideState;
if (!valid)
EditorGUILayout.HelpBox(
"Focus Tracking requires an active DepthOfField/FocusDistance effect in the profile",
MessageType.Warning);
else
{
if (!Target.VirtualCamera.State.HasLookAt)
EditorGUILayout.HelpBox(
"Focus Offset is relative to the Camera position",
MessageType.Info);
else
EditorGUILayout.HelpBox(
"Focus Offset is relative to the Target position",
MessageType.Info);
}
}
var rect = GUILayoutUtility.GetRect(1, EditorGUIUtility.singleLineHeight); rect.y += 2;
float checkboxWidth = rect.height + 5;
rect = EditorGUI.PrefixLabel(rect, new GUIContent(m_FocusTracksTarget.displayName));
EditorGUI.PropertyField(new Rect(rect.x, rect.y, checkboxWidth, rect.height), m_FocusTracksTarget, GUIContent.none);
rect.x += checkboxWidth; rect.width -= checkboxWidth;
if (m_FocusTracksTarget.boolValue)
{
GUIContent offsetText = new GUIContent("Offset ");
var textDimensions = GUI.skin.label.CalcSize(offsetText);
float oldWidth = EditorGUIUtility.labelWidth;
EditorGUIUtility.labelWidth = textDimensions.x;
EditorGUI.PropertyField(rect, m_FocusOffset, offsetText);
EditorGUIUtility.labelWidth = oldWidth;
}
DrawProfileInspectorGUI();
Target.InvalidateCachedProfile();
serializedObject.ApplyModifiedProperties();
}
public static PostProcessProfile CreatePostProcessProfile(Scene scene, string targetName)
{
var path = string.Empty;
if (string.IsNullOrEmpty(scene.path))
{
path = "Assets/";
}
else
{
var scenePath = Path.GetDirectoryName(scene.path);
var extPath = scene.name + "_Profiles";
var profilePath = scenePath + "/" + extPath;
if (!AssetDatabase.IsValidFolder(profilePath))
AssetDatabase.CreateFolder(scenePath, extPath);
path = profilePath + "/";
}
path += targetName + " Profile.asset";
path = AssetDatabase.GenerateUniqueAssetPath(path);
var profile = ScriptableObject.CreateInstance<PostProcessProfile>();
AssetDatabase.CreateAsset(profile, path);
AssetDatabase.SaveAssets();
AssetDatabase.Refresh();
return profile;
}
void DrawProfileInspectorGUI()
{
EditorGUILayout.Space();
bool assetHasChanged = false;
bool showCopy = m_Profile.objectReferenceValue != null;
// The layout system sort of break alignement when mixing inspector fields with custom
// layouted fields, do the layout manually instead
int buttonWidth = showCopy ? 45 : 60;
float indentOffset = EditorGUI.indentLevel * 15f;
var lineRect = GUILayoutUtility.GetRect(1, EditorGUIUtility.singleLineHeight);
var labelRect = new Rect(lineRect.x, lineRect.y, EditorGUIUtility.labelWidth - indentOffset, lineRect.height);
var fieldRect = new Rect(labelRect.xMax, lineRect.y, lineRect.width - labelRect.width - buttonWidth * (showCopy ? 2 : 1), lineRect.height);
var buttonNewRect = new Rect(fieldRect.xMax, lineRect.y, buttonWidth, lineRect.height);
var buttonCopyRect = new Rect(buttonNewRect.xMax, lineRect.y, buttonWidth, lineRect.height);
EditorGUI.PrefixLabel(labelRect, m_ProfileLabel);
using (var scope = new EditorGUI.ChangeCheckScope())
{
m_Profile.objectReferenceValue
= (PostProcessProfile)EditorGUI.ObjectField(
fieldRect, m_Profile.objectReferenceValue, typeof(PostProcessProfile), false);
assetHasChanged = scope.changed;
}
if (GUI.Button(
buttonNewRect,
EditorUtilities.GetContent("New|Create a new profile."),
showCopy ? EditorStyles.miniButtonLeft : EditorStyles.miniButton))
{
// By default, try to put assets in a folder next to the currently active
// scene file. If the user isn't a scene, put them in root instead.
var targetName = Target.name;
var scene = Target.gameObject.scene;
var asset = CreatePostProcessProfile(scene, targetName);
m_Profile.objectReferenceValue = asset;
assetHasChanged = true;
}
if (showCopy && GUI.Button(
buttonCopyRect,
EditorUtilities.GetContent("Clone|Create a new profile and copy the content of the currently assigned profile."),
EditorStyles.miniButtonRight))
{
// Duplicate the currently assigned profile and save it as a new profile
var origin = (PostProcessProfile)m_Profile.objectReferenceValue;
var path = AssetDatabase.GetAssetPath(origin);
path = AssetDatabase.GenerateUniqueAssetPath(path);
var asset = Instantiate(origin);
asset.settings.Clear();
AssetDatabase.CreateAsset(asset, path);
foreach (var item in origin.settings)
{
var itemCopy = Instantiate(item);
itemCopy.hideFlags = HideFlags.HideInInspector | HideFlags.HideInHierarchy;
itemCopy.name = item.name;
asset.settings.Add(itemCopy);
AssetDatabase.AddObjectToAsset(itemCopy, asset);
}
AssetDatabase.SaveAssets();
AssetDatabase.Refresh();
m_Profile.objectReferenceValue = asset;
assetHasChanged = true;
}
if (m_Profile.objectReferenceValue == null)
{
if (assetHasChanged && m_EffectList != null)
m_EffectList.Clear(); // Asset wasn't null before, do some cleanup
EditorGUILayout.HelpBox(
"Assign an existing Post-process Profile by choosing an asset, or create a new one by clicking the \"New\" button.\nNew assets are automatically put in a folder next to your scene file. If your scene hasn't been saved yet they will be created at the root of the Assets folder.",
MessageType.Info);
}
else
{
if (assetHasChanged)
RefreshEffectListEditor((PostProcessProfile)m_Profile.objectReferenceValue);
if (m_EffectList != null)
m_EffectList.OnGUI();
}
}
}
}