ClientServer/Client/Assets/GPUInstancer/Scripts/GPUInstancerTreeManager.cs
TG9six 03a642d635 first push
first push
2025-09-06 17:17:39 +04:00

289 lines
12 KiB
C#

using System.Collections.Generic;
using UnityEngine;
using System.Collections;
#if UNITY_EDITOR
using UnityEditor;
#endif
namespace GPUInstancer
{
/// <summary>
/// Add this to a Unity terrain for GPU Instancing terrain trees at runtime.
/// </summary>
[ExecuteInEditMode]
public class GPUInstancerTreeManager : GPUInstancerTerrainManager
{
private static ComputeShader _treeInstantiationComputeShader;
public bool initializeWithCoroutine = true;
private bool _isCoroutineActive;
#region MonoBehaviour Methods
public override void Awake()
{
base.Awake();
if (_treeInstantiationComputeShader == null)
_treeInstantiationComputeShader = Resources.Load<ComputeShader>(GPUInstancerConstants.TREE_INSTANTIATION_RESOURCE_PATH);
}
public override void Update()
{
base.Update();
if (Application.isPlaying && _requiresTerrainUpdate && !_isCoroutineActive)
{
StartCoroutine(ReplaceUnityTrees());
_requiresTerrainUpdate = false;
}
}
#endregion MonoBehaviour Methods
#region Override Methods
public override void ClearInstancingData()
{
base.ClearInstancingData();
if (_terrains != null)
{
foreach (Terrain terrain in _terrains)
{
if (terrain != null && terrain.treeDistance == 0)
terrain.treeDistance = terrainSettings.maxTreeDistance;
}
}
}
public override void GeneratePrototypes(bool forceNew = false)
{
base.GeneratePrototypes(forceNew);
if (terrainSettings != null && terrain != null && terrain.terrainData != null)
{
GPUInstancerUtility.SetTreeInstancePrototypes(gameObject, prototypeList, terrain.terrainData.treePrototypes, terrainSettings, forceNew);
}
}
#if UNITY_EDITOR
public override void CheckPrototypeChanges()
{
base.CheckPrototypeChanges();
if (!Application.isPlaying && terrainSettings != null && terrain != null && terrain.terrainData != null)
{
if (prototypeList.Count != terrain.terrainData.treePrototypes.Length)
{
GeneratePrototypes();
}
int index = 0;
foreach (GPUInstancerTreePrototype prototype in prototypeList)
{
prototype.prototypeIndex = index;
index++;
}
}
}
#endif
public override void InitializeRuntimeDataAndBuffers(bool forceNew = true)
{
base.InitializeRuntimeDataAndBuffers(forceNew);
if (!forceNew && isInitialized)
return;
if (terrainSettings == null)
return;
if (prototypeList != null && prototypeList.Count > 0)
{
GPUInstancerUtility.AddTreeInstanceRuntimeDataToList(runtimeDataList, prototypeList, terrainSettings);
}
StartCoroutine(ReplaceUnityTrees());
isInitialized = true;
}
public override void DeletePrototype(GPUInstancerPrototype prototype, bool removeSO = true)
{
if (terrainSettings != null && terrain != null && terrain.terrainData != null)
{
int treePrototypeIndex = prototypeList.IndexOf(prototype);
TreePrototype[] treePrototypes = terrain.terrainData.treePrototypes;
List<TreePrototype> newTreePrototypes = new List<TreePrototype>(treePrototypes);
List<TreeInstance> newTreeInstanceList = new List<TreeInstance>();
TreeInstance treeInstance;
for (int i = 0; i < terrain.terrainData.treeInstances.Length; i++)
{
treeInstance = terrain.terrainData.treeInstances[i];
if (treeInstance.prototypeIndex < treePrototypeIndex)
{
newTreeInstanceList.Add(treeInstance);
}
else if (treeInstance.prototypeIndex > treePrototypeIndex)
{
treeInstance.prototypeIndex = treeInstance.prototypeIndex - 1;
newTreeInstanceList.Add(treeInstance);
}
}
if (newTreePrototypes.Count > treePrototypeIndex)
newTreePrototypes.RemoveAt(treePrototypeIndex);
terrain.terrainData.treeInstances = newTreeInstanceList.ToArray();
terrain.terrainData.treePrototypes = newTreePrototypes.ToArray();
terrain.terrainData.RefreshPrototypes();
if (removeSO)
base.DeletePrototype(prototype, removeSO);
GeneratePrototypes(false);
if (!removeSO)
base.DeletePrototype(prototype, removeSO);
}
else
base.DeletePrototype(prototype, removeSO);
}
#endregion Override Methods
public IEnumerator ReplaceUnityTrees()
{
_isCoroutineActive = true;
if (prototypeList.Count > 0)
{
Vector4[] treeScales = new Vector4[prototypeList.Count];
int count = 0;
foreach (GPUInstancerTreePrototype tp in prototypeList)
{
treeScales[count] = tp.isApplyPrefabScale ? tp.prefabObject.transform.localScale : Vector3.one;
count++;
}
int[] instanceCounts = new int[prototypeList.Count];
List<Vector4> treeDataList = new List<Vector4>(); // prototypeIndex - positionx3 - rotation - scalex2
int instanceTotal = 0;
foreach (Terrain terrain in _terrains)
{
if (terrain == null)
continue;
if (terrain.terrainData.treePrototypes.Length > prototypeList.Count)
{
Debug.LogError("Additional Terrain has more Tree prototypes than defined prototypes on the Tree Manager. Tree Manager requires every Terrain to have the same Tree prototypes defined.", terrain);
continue;
}
terrain.treeDistance = 0f; // will not persist if called at runtime.
Vector3 terrainSize = terrain.terrainData.size;
Vector3 terrainPosition = terrain.GetPosition();
TreeInstance[] treeInstances = terrain.terrainData.treeInstances;
instanceTotal += treeInstances.Length;
foreach (TreeInstance treeInstance in treeInstances)
{
treeDataList.Add(new Vector4(
treeInstance.prototypeIndex,
treeInstance.position.x * terrainSize.x + terrainPosition.x,
treeInstance.position.y * terrainSize.y + terrainPosition.y,
treeInstance.position.z * terrainSize.z + terrainPosition.z
));
treeDataList.Add(new Vector4(
treeInstance.rotation,
treeInstance.widthScale,
treeInstance.heightScale,
0
));
instanceCounts[treeInstance.prototypeIndex]++;
}
}
if (instanceTotal > 0)
{
if (initializeWithCoroutine && !isInitialized)
yield return null;
ComputeBuffer treeDataBuffer = new ComputeBuffer(treeDataList.Count, GPUInstancerConstants.STRIDE_SIZE_FLOAT4);
#if UNITY_2019_1_OR_NEWER
treeDataBuffer.SetData(treeDataList);
#else
treeDataBuffer.SetData(treeDataList.ToArray());
#endif
ComputeBuffer treeScalesBuffer = new ComputeBuffer(treeScales.Length, GPUInstancerConstants.STRIDE_SIZE_FLOAT4);
treeScalesBuffer.SetData(treeScales);
ComputeBuffer counterBuffer = new ComputeBuffer(1, GPUInstancerConstants.STRIDE_SIZE_INT);
uint[] emptyCounterData = new uint[1];
treeDataList = null;
treeScales = null;
GPUInstancerRuntimeData runtimeData;
for (int i = 0; i < runtimeDataList.Count; i++)
{
runtimeData = runtimeDataList[i];
int instanceCount = instanceCounts[i];
runtimeData.bufferSize = instanceCount;
runtimeData.instanceCount = instanceCount;
if (instanceCount == 0)
{
GPUInstancerUtility.ReleaseInstanceBuffers(runtimeData);
continue;
}
counterBuffer.SetData(emptyCounterData);
if (runtimeData.transformationMatrixVisibilityBuffer != null)
runtimeData.transformationMatrixVisibilityBuffer.Release();
runtimeData.transformationMatrixVisibilityBuffer = new ComputeBuffer(instanceCount, GPUInstancerConstants.STRIDE_SIZE_MATRIX4X4);
_treeInstantiationComputeShader.SetBuffer(0,
GPUInstancerConstants.VisibilityKernelPoperties.INSTANCE_DATA_BUFFER, runtimeData.transformationMatrixVisibilityBuffer);
_treeInstantiationComputeShader.SetBuffer(0,
GPUInstancerConstants.TreeKernelProperties.TREE_DATA, treeDataBuffer);
_treeInstantiationComputeShader.SetBuffer(0,
GPUInstancerConstants.TreeKernelProperties.TREE_SCALES, treeScalesBuffer);
_treeInstantiationComputeShader.SetBuffer(0,
GPUInstancerConstants.GrassKernelProperties.COUNTER_BUFFER, counterBuffer);
_treeInstantiationComputeShader.SetInt(
GPUInstancerConstants.VisibilityKernelPoperties.BUFFER_PARAMETER_BUFFER_SIZE, instanceTotal);
//_treeInstantiationComputeShader.SetVector(
// GPUInstancerConstants.GrassKernelProperties.TERRAIN_SIZE_DATA, terrain.terrainData.size);
//_treeInstantiationComputeShader.SetVector(
// GPUInstancerConstants.TreeKernelProperties.TERRAIN_POSITION, terrain.GetPosition());
_treeInstantiationComputeShader.SetBool(
GPUInstancerConstants.TreeKernelProperties.IS_APPLY_ROTATION, ((GPUInstancerTreePrototype)runtimeData.prototype).isApplyRotation);
_treeInstantiationComputeShader.SetBool(
GPUInstancerConstants.TreeKernelProperties.IS_APPLY_TERRAIN_HEIGHT, ((GPUInstancerTreePrototype)runtimeData.prototype).isApplyTerrainHeight);
_treeInstantiationComputeShader.SetInt(
GPUInstancerConstants.TreeKernelProperties.PROTOTYPE_INDEX, i);
_treeInstantiationComputeShader.Dispatch(0,
Mathf.CeilToInt(instanceTotal / GPUInstancerConstants.COMPUTE_SHADER_THREAD_COUNT), 1, 1);
GPUInstancerUtility.InitializeGPUBuffer(runtimeData);
if (initializeWithCoroutine && !isInitialized)
yield return null;
}
treeDataBuffer.Release();
treeScalesBuffer.Release();
counterBuffer.Release();
}
else
{
GPUInstancerUtility.ReleaseInstanceBuffers(runtimeDataList);
}
}
isInitial = true;
if (!isInitialized)
GPUInstancerUtility.TriggerEvent(GPUInstancerEventType.TreeInitializationFinished);
_isCoroutineActive = false;
}
}
}