2024-10-20 15:47:57 -06:00
|
|
|
# This script is an addon for HungryProton's Scatter https://github.com/HungryProton/scatter
|
|
|
|
# It provides a `Project on Terrain3D` modifier, which allows Scatter
|
|
|
|
# to detect the terrain height from Terrain3D without using collision.
|
|
|
|
# Copy this file into /addons/proton_scatter/src/modifiers
|
|
|
|
# Then uncomment everything below
|
|
|
|
# In the editor, add this modifier to Scatter, then set your Terrain3D node
|
|
|
|
|
|
|
|
# This script is an addon for HungryProton's Scatter https://github.com/HungryProton/scatter
|
|
|
|
# It allows Scatter to detect the terrain height from Terrain3D
|
|
|
|
# Copy this file into /addons/proton_scatter/src/modifiers
|
|
|
|
# Then uncomment everything below (select, press CTRL+K)
|
|
|
|
# In the editor, add this modifier, then set your Terrain3D node
|
|
|
|
|
|
|
|
#@tool
|
|
|
|
#extends "base_modifier.gd"
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#signal projection_completed
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#@export var terrain_node : NodePath
|
|
|
|
#@export var align_with_collision_normal := false
|
|
|
|
#
|
|
|
|
#var _terrain: Terrain3D
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#func _init() -> void:
|
|
|
|
#display_name = "Project On Terrain3D"
|
|
|
|
#category = "Edit"
|
|
|
|
#can_restrict_height = false
|
|
|
|
#global_reference_frame_available = true
|
|
|
|
#local_reference_frame_available = true
|
|
|
|
#individual_instances_reference_frame_available = true
|
|
|
|
#use_global_space_by_default()
|
|
|
|
#
|
|
|
|
#documentation.add_paragraph(
|
|
|
|
#"This is a duplicate of `Project on Colliders` that queries the terrain system
|
|
|
|
#for height and sets the transform height appropriately.
|
|
|
|
#
|
|
|
|
#This modifier must have terrain_node set to a Terrain3D node.")
|
|
|
|
#
|
|
|
|
#var p := documentation.add_parameter("Terrain Node")
|
|
|
|
#p.set_type("NodePath")
|
|
|
|
#p.set_description("Set your Terrain3D node.")
|
|
|
|
#
|
|
|
|
#p = documentation.add_parameter("Align with collision normal")
|
|
|
|
#p.set_type("bool")
|
|
|
|
#p.set_description(
|
|
|
|
#"Rotate the transform to align it with the collision normal in case
|
|
|
|
#the ray cast hit a collider.")
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#func _process_transforms(transforms, domain, _seed) -> void:
|
|
|
|
#if transforms.is_empty():
|
|
|
|
#return
|
|
|
|
#
|
|
|
|
#if terrain_node:
|
|
|
|
#_terrain = domain.get_root().get_node_or_null(terrain_node)
|
|
|
|
#
|
|
|
|
#if not _terrain:
|
|
|
|
#warning += """No Terrain3D node found"""
|
|
|
|
#return
|
|
|
|
#
|
2024-11-17 12:47:37 -07:00
|
|
|
#if not _terrain.data:
|
|
|
|
#warning += """Terrain3DData is not initialized"""
|
2024-10-20 15:47:57 -06:00
|
|
|
#return
|
|
|
|
#
|
|
|
|
## Get global transform
|
|
|
|
#var gt: Transform3D = domain.get_global_transform()
|
|
|
|
#var gt_inverse := gt.affine_inverse()
|
|
|
|
#for i in transforms.list.size():
|
|
|
|
#var location: Vector3 = (gt * transforms.list[i]).origin
|
2024-11-17 12:47:37 -07:00
|
|
|
#var height: float = _terrain.data.get_height(location)
|
|
|
|
#var normal: Vector3 = _terrain.data.get_normal(location)
|
2024-10-20 15:47:57 -06:00
|
|
|
#
|
|
|
|
#if align_with_collision_normal and not is_nan(normal.x):
|
|
|
|
#transforms.list[i].basis.y = normal
|
|
|
|
#transforms.list[i].basis.x = -transforms.list[i].basis.z.cross(normal)
|
|
|
|
#transforms.list[i].basis = transforms.list[i].basis.orthonormalized()
|
|
|
|
#
|
|
|
|
#transforms.list[i].origin.y = gt.origin.y if is_nan(height) else height - gt.origin.y
|
|
|
|
#
|
|
|
|
#if transforms.is_empty():
|
|
|
|
#warning += """Every point has been removed. Possible reasons include: \n
|
|
|
|
#+ No collider is close enough to the shapes.
|
|
|
|
#+ Ray length is too short.
|
|
|
|
#+ Ray direction is incorrect.
|
|
|
|
#+ Collision mask is not set properly.
|
|
|
|
#+ Max slope is too low.
|
|
|
|
#"""
|