rj-action-library/Runtime/Procedural/Assets/Grass/GrassPatch.cs

484 lines
13 KiB
C#
Raw Normal View History

2024-11-12 08:03:36 +00:00
using System.Collections;
using System.Collections.Generic;
using Godot;
using System;
namespace Rokojori
{
[Tool]
2024-11-13 11:57:10 +00:00
[GlobalClass, Icon("res://addons/rokojori_action_library/Icons/Scatterer.svg") ]
2024-11-12 08:03:36 +00:00
public partial class GrassPatch:Node3D
{
[Export]
public MeshInstance3D output;
[Export]
public int seed = 1984;
[Export]
public bool update;
[Export]
public bool updateAlways;
2025-01-03 12:09:23 +00:00
[ExportGroup( "Blades")]
2024-11-12 08:03:36 +00:00
[Export( PropertyHint.Range, "0,100")]
public int blades = 20;
[Export( PropertyHint.Range, "0,100")]
public int bladesX = 0;
[Export( PropertyHint.Range, "0,100")]
public int bladesZ = 0;
[Export]
public int X_numBlades;
2025-01-03 12:09:23 +00:00
[ExportGroup( "Patch")]
2024-11-12 08:03:36 +00:00
[Export]
public float patchSize = 2;
[Export]
public float patchSizeX = 0;
[Export]
public float patchSizeZ = 0;
[Export]
public Vector2 patchOffsetPosition = new Vector2( 0.5f, 0.5f );
2025-01-03 12:09:23 +00:00
[ExportGroup( "Blade Triangles")]
2024-11-12 08:03:36 +00:00
[Export( PropertyHint.Range, "1,20")]
public int bladeSegments = 3;
[Export]
public bool createBackFaces = true;
2025-01-03 12:09:23 +00:00
[Export]
public bool allowTrianglesOnEnds = true;
2024-11-12 08:03:36 +00:00
[Export]
public int X_numTriangles;
[Export]
public Curve bladeSegmentMapping = MathX.Curve( 0, 1 );
2025-01-03 12:09:23 +00:00
[ExportGroup( "Blade Segmentation")]
2024-11-12 08:03:36 +00:00
[Export]
public int uvSegmentColumns = 1;
[Export]
public int uvSegmentRows = 1;
[Export]
public Curve uvSegmentWeightsClose = MathX.Curve( 1f );
[Export]
public Curve uvSegmentWeightsFar = null;
2025-01-03 12:09:23 +00:00
[ExportGroup( "Blade Shape")]
2024-11-12 08:03:36 +00:00
[Export]
public Curve bladeScale = MathX.Curve( 1f );
2024-12-01 17:07:41 +00:00
[Export]
public Curve bladeHeight = MathX.Curve( 0.4f );
[Export]
public Curve bladeInGround = MathX.Curve( 0.05f );
2024-11-12 08:03:36 +00:00
[Export]
public Curve bladeWidth = MathX.Curve( 0.05f );
[Export]
public Curve bladeWidth2 = null;
[Export]
public Curve bladeBending = MathX.Curve( 0f );
[Export]
public Curve bladeBending2 = null;
2025-01-03 12:09:23 +00:00
[ExportGroup( "Blade Offset & Scale")]
2024-11-12 08:03:36 +00:00
[Export]
2024-12-01 17:07:41 +00:00
public Curve positionJitter = MathX.Curve( 0.05f );
2024-11-12 08:03:36 +00:00
[Export]
2025-01-03 12:09:23 +00:00
public Curve positionJitterX = MathX.Curve( 0.05f );
2024-11-12 08:03:36 +00:00
[Export]
2025-01-03 12:09:23 +00:00
public Curve positionJitterZ = MathX.Curve( 0.05f );
2024-11-12 08:03:36 +00:00
[Export]
2025-01-03 12:09:23 +00:00
public Curve scaleByDistanceX = MathX.Curve( 1f );
2024-11-12 08:03:36 +00:00
[Export]
2025-01-03 12:09:23 +00:00
public Curve scaleByDistanceZ = MathX.Curve( 1f );
[ExportGroup( "Blade Rotation")]
2024-11-12 08:03:36 +00:00
[Export]
public Curve yawRotation = MathX.Curve( 0f, 1f );
[Export]
public Curve randomRotation = MathX.Curve( 0f, 20f );
2025-01-03 12:09:23 +00:00
[ExportGroup( "Blade Normals")]
[Export]
public Curve normalBlending = MathX.Curve( 0.5f );
[Export]
public Vector3 normalBlendingDirection = Vector3.Up;
[ExportGroup( "Blade Filter")]
2024-11-12 08:03:36 +00:00
[Export (PropertyHint.Range, "0,1")]
public float filterTreshold = 1;
[Export]
public Vector2 filterScale = new Vector2( 1, 1 );
[Export]
public Vector2 filterOffset = new Vector2( 0, 0 );
[Export]
public Vector2 positionToFilter = new Vector2( 0, 0 );
2025-01-03 12:09:23 +00:00
[ExportGroup("LOD Levels")]
[Export]
public GrassPatchLODLevel[] lodLevels;
[Export]
public int currentLODLevel = -1;
2024-11-13 19:33:08 +00:00
SerializedGodotObject _cached;
2024-11-12 08:03:36 +00:00
public override void _Process( double delta )
{
if ( ! ( update || updateAlways ) )
{
return;
}
2024-12-01 17:07:41 +00:00
update = false;
2024-11-13 19:33:08 +00:00
var current = SerializedGodotObject.Create( this );
var isEquals = _cached != null && _cached.Equals( current );
if ( _cached != null && _cached.Equals( current ) )
{
return;
}
2024-12-01 17:07:41 +00:00
2024-11-12 08:03:36 +00:00
CreatePatch();
2024-11-13 19:33:08 +00:00
_cached = current;
2024-11-12 08:03:36 +00:00
}
float _maxWidth = 0;
Vector3 _patchOffset = Vector3.Zero;
public void CreatePatch()
{
2025-01-03 12:09:23 +00:00
if ( blades == 0 && bladesX == 0 && bladesZ == 0)
{
return;
}
if ( blades == 0 )
{
if ( bladesX < bladesZ )
{
bladesX = Mathf.Max( 1, bladesX );
}
else
{
bladesZ = Mathf.Max( 1, bladesZ );
}
}
if ( lodLevels == null )
{
lodLevels = new GrassPatchLODLevel[]{};
}
if ( output == null )
{
this.output = this.CreateChild<MeshInstance3D>( "Grass Patch Mesh" );
}
2024-11-12 08:03:36 +00:00
var mg = new MeshGeometry();
2025-01-03 12:09:23 +00:00
2024-11-12 08:03:36 +00:00
var cellSizeX = ( patchSizeX + patchSize ) / ( bladesX + blades );
var cellSizeZ = ( patchSizeZ + patchSize ) / ( bladesZ + blades );
var random = new LCG();
random.SetSeed( 1712 + seed );
_patchOffset = new Vector3( - ( patchSizeX + patchSize ) * patchOffsetPosition.X, 0, - ( patchSizeZ + patchSize ) * patchOffsetPosition.Y );
_maxWidth = MathX.CurveMaximum( bladeWidth );
X_numBlades = 0;
var allBladesX = bladesX + blades;
var allBladesZ = bladesZ + blades;
for ( int i = 0; i < allBladesX; i++ )
{
var x = ( i + 0.5f ) * cellSizeX;
for ( int j = 0; j < allBladesZ; j++ )
{
var z = ( j + 0.5f ) * cellSizeZ;
random.SetSeed( i * 11223 + j *12895 + seed );
var position = new Vector3( x, 0, z );
var yaw = random.Sample( yawRotation );
var rotationY = yaw * Mathf.Pi * 2f;
var maxRotation = random.Sample( randomRotation );
2024-12-01 17:07:41 +00:00
var normalBlendingAmount = random.Sample( normalBlending );
2024-11-12 08:03:36 +00:00
var rotationOther = random.Next() * Mathf.DegToRad( maxRotation );
var rotationX = random.Next() * rotationOther;
var rotationZ = rotationOther - rotationX;
var positionOffset = Math3D.OnCircleXZ( random.Next() * Mathf.Pi * 2f ) * random.Sample( positionJitter );
2025-01-03 12:09:23 +00:00
positionOffset.X += random.Sample( positionJitterX );
positionOffset.Z += random.Sample( positionJitterZ );
2024-11-12 08:03:36 +00:00
var worldPosition = position + _patchOffset + positionOffset;
var filterValue = Noise.PerlinXZ( Math3D.XYasXZ( filterScale ) * worldPosition + Math3D.XYasXZ( filterOffset ) + -GlobalPosition * Math3D.XYasXZ( positionToFilter ));
var trsf = new Transform3D(
new Basis( Math3D.RotateY( rotationY ) * Math3D.RotateX( rotationX ) * Math3D.RotateZ( rotationZ )),
worldPosition
);
var bladeMG = CreateBlade( random, worldPosition );
if ( filterValue > filterTreshold )
{
continue;
}
2025-01-03 12:09:23 +00:00
var cL = Mathf.Clamp( currentLODLevel, -1, lodLevels.Length - 1 );
if ( cL != -1 )
{
var lodFilterTreshold = lodLevels[ cL ].filter;
if ( random.Next() >= lodFilterTreshold )
{
continue;
}
}
2024-11-12 08:03:36 +00:00
bladeMG.ApplyTransform( trsf );
2024-12-01 17:07:41 +00:00
MeshGeometry clonedBladeMG = null;
if ( createBackFaces )
{
clonedBladeMG = bladeMG.Clone();
}
bladeMG.BlendNormals( normalBlendingDirection, normalBlendingAmount );
2024-11-12 08:03:36 +00:00
mg.Add( bladeMG );
if ( createBackFaces )
{
2024-12-01 17:07:41 +00:00
clonedBladeMG.FlipNormalDirection();
clonedBladeMG.BlendNormals( normalBlendingDirection, normalBlendingAmount );
mg.Add( clonedBladeMG );
2024-11-12 08:03:36 +00:00
}
X_numBlades ++;
}
}
X_numTriangles = mg.indices.Count / 3;
output.Mesh = mg.GenerateMesh();
}
2025-01-03 12:09:23 +00:00
2024-11-12 08:03:36 +00:00
MeshGeometry CreateBlade( RandomEngine random, Vector3 position )
{
2025-01-03 12:09:23 +00:00
var bladeSegments = this.bladeSegments;
var currentLodLevel = Mathf.Clamp( currentLODLevel, -1, lodLevels.Length - 1 );
var bladeScaleMultiplier = 1f;
var bladeWidthMultiplier = 1f;
if ( currentLodLevel != -1 )
{
bladeSegments = lodLevels[ currentLodLevel ].bladeSegments;
bladeScaleMultiplier = lodLevels[ currentLodLevel ].bladeScaleMultiplier;
bladeWidthMultiplier = lodLevels[ currentLodLevel].bladeWidthMultiplier;
}
2024-11-12 08:03:36 +00:00
var bmg = new MeshGeometry();
var inGround = random.Sample( bladeInGround );
var height = random.Sample( bladeHeight );
var size = height + inGround;
var distancesToCenter = new Vector2( position.X / ( patchSizeX + patchSize ), position.Z / ( patchSizeZ + patchSize ) );
distancesToCenter = distancesToCenter.Abs();
var minDistance = distancesToCenter.Length();
var scaling = scaleByDistanceX.Sample( distancesToCenter.X ) * scaleByDistanceZ.Sample( distancesToCenter.Y );
scaling *= random.Sample( bladeScale );
2025-01-03 12:09:23 +00:00
scaling *= bladeScaleMultiplier;
2024-11-12 08:03:36 +00:00
size *= scaling;
var firstIsTriangle = false;
var lastIsTriangle = false;
var bladeWidthLerp = random.Next();
var bladeBendLerp = random.Next();
var uvMin = new Vector2( 0, 0 );
var uvMax = new Vector2( 1, 1 );
var uvSegments = uvSegmentRows * uvSegmentColumns;
if ( uvSegments > 1 )
{
var index = random.IntegerExclusive( uvSegments );
if ( uvSegmentWeightsClose != null )
2024-11-13 11:57:10 +00:00
{
2024-11-12 08:03:36 +00:00
var weightsClose = MathX.GetCurveWeights( uvSegmentWeightsClose, uvSegments );
if ( uvSegmentWeightsFar != null )
{
var weightsFar = MathX.GetCurveWeights( uvSegmentWeightsFar, uvSegments );
var distanceFade = minDistance;
for ( int i = 0; i < uvSegments; i++ )
{
weightsClose[ i ] += distanceFade * ( weightsFar[ i ] - weightsClose[ i ] );
}
}
index = random.IndexFromUnnormalizedWeights( weightsClose );
}
var x = index % uvSegmentColumns;
2024-11-13 11:57:10 +00:00
var y = Mathf.FloorToInt( index / uvSegmentColumns );
2024-11-12 08:03:36 +00:00
2024-11-13 11:57:10 +00:00
var xSize = 1f / uvSegmentColumns;
var ySize = 1f / uvSegmentRows;
2024-11-12 08:03:36 +00:00
uvMin.X = x * xSize;
uvMin.Y = y * ySize;
uvMax = uvMin + new Vector2( xSize, ySize );
2025-01-03 12:09:23 +00:00
}
2024-11-13 11:57:10 +00:00
2025-01-03 12:09:23 +00:00
var endingTriangles = allowTrianglesOnEnds;
if ( currentLodLevel != -1 )
{
var useTris = lodLevels[ currentLodLevel ].allowTrianglesOnEnd;
endingTriangles = TrilleanLogic.ToBool( useTris, endingTriangles );
2024-11-12 08:03:36 +00:00
}
2025-01-03 12:09:23 +00:00
2024-11-12 08:03:36 +00:00
for ( int i = 0; i <= bladeSegments; i++ )
{
var t = (float)i / bladeSegments;
t = bladeSegmentMapping.Sample( t );
var v = 1f - t;
var y = size * t - inGround;
var bw2 = bladeWidth2 == null ? bladeWidth : bladeWidth2;
var bb2 = bladeBending2 == null ? bladeBending : bladeBending2;
var width = Mathf.Lerp( bladeWidth.Sample( v ), bw2.Sample( v ), bladeWidthLerp );
2025-01-03 12:09:23 +00:00
width *= scaling * bladeWidthMultiplier;
2024-11-12 08:03:36 +00:00
var bending = Mathf.Lerp( bladeBending.Sample( v ), bb2.Sample( v ), bladeBendLerp ) * Vector3.Forward.Z * scaling;
var bendingNormalAngle = Mathf.LerpAngle( MathX.CurveAngle( bladeBending, v ), MathX.CurveAngle( bb2, v ), bladeBendLerp );
var bendingNormal = Math3D.NormalAngle( bendingNormalAngle );
2025-01-03 12:09:23 +00:00
2024-11-12 08:03:36 +00:00
2025-01-03 12:09:23 +00:00
if ( endingTriangles && width < 0.005f && ( i == 0 || ( i == bladeSegments - 1 )) )
2024-11-12 08:03:36 +00:00
{
bmg.vertices.Add( new Vector3( 0, y, bending ) );
bmg.normals.Add( bendingNormal );
bmg.uvs.Add( MapUV( new Vector2( 0.5f, v ), uvMin, uvMax ) );
if ( i == 0 ){ firstIsTriangle = true; }
if ( i == bladeSegments - 1 ){ lastIsTriangle = true; }
}
else
{
bmg.vertices.Add( new Vector3( -width, y, bending ) );
bmg.vertices.Add( new Vector3( width, y, bending ) );
bmg.normals.Add( bendingNormal );
bmg.normals.Add( bendingNormal );
bmg.uvs.Add( MapUV( new Vector2( -width / _maxWidth, v ), uvMin, uvMax ) );
bmg.uvs.Add( MapUV( new Vector2( width / _maxWidth, v ), uvMin, uvMax ) );
}
}
for ( int i = 0; i < bladeSegments; i++ )
{
var index = i * 2;
if ( firstIsTriangle && index != 0 )
{
index --;
}
if ( i == 0 && firstIsTriangle )
{
bmg.AddTriangle( index + 0, index + 1, index + 2 );
}
else if ( i == ( bladeSegments - 1 ) && lastIsTriangle )
{
bmg.AddTriangle( index + 1, index + 0, index + 2 );
}
else
{
bmg.AddQuad( index + 0, index + 1, index + 2, index + 3 );
}
}
return bmg;
}
Vector2 MapUV( Vector2 uv, Vector2 min, Vector2 max )
{
uv = Math2D.Clamp01( uv );
return Math2D.Map( uv, Vector2.Zero, Vector2.One, min, max );
}
}
}