xrjamfeb2025/scenes/collisions/proxy_collision_detector.gd
2025-02-14 15:43:06 -07:00

124 lines
5.1 KiB
GDScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# This script is designed to drive gameplay logic based on the number of GPUParticles3D
# that "hit" various static target areas in 3D space. Multiple particle systems can be cloned
# and rendered in a dedicated SubViewport on layer 20. Each uses a special testing shader that,
# for each particle, checks its 3D world-space position against up to eight static target
# bounding boxes.
#
# The testing shader maps particles that hit a target into a designated horizontal slice of clip
# space so that the SubViewport (sized 8×1 pixels) accumulates red intensity in each column. Higher
# red intensity means more particles hit that target.
#
# In this version, the targets are static so their parameters (center, extents, and the inverse of
# their 3×3 rotation basis) are computed once and sent to the shader. The attractors are shared
# (set up in the main scene and used by the cloned particle systems) rather than cloned.
#
# The script looks for nodes in the "targets" group that are instances of the Target class,
# and directly calls their set_occupancy() method with the measured red intensity.
extends Node3D
# Optional debug mesh to visualize the viewport texture
@export var debug_mesh_path: NodePath
var debug_mesh: MeshInstance3D
# Exported node paths for particle systems to clone
@export var source_particles_paths: Array[NodePath] = [] # Array of GPUParticles3D nodes to clone
# Path to the testing shader resource.
const shader = preload("res://scenes/particle_test.gdshader")
# Maximum number of target areas supported.
const MAX_TARGETS: int = 8
# References.
var clone_particles: Array[GPUParticles3D] = []
@onready var sub_viewport: SubViewport = $SubViewport
var targets: Array[Target] = []
func _ready():
# Find all Target nodes in the "targets" group
for node in get_tree().get_nodes_in_group("targets"):
print("Found node in targets group: ", node.name, " of type ", node.get_class())
if node is Target:
print("Adding target: ", node.name)
targets.append(node)
else:
print("Skipping non-Target node: ", node.name)
print("Found ", targets.size(), " valid targets")
# Ensure we don't exceed MAX_TARGETS
if targets.size() > MAX_TARGETS:
push_warning("More than %d targets found, extras will be ignored" % MAX_TARGETS)
targets.resize(MAX_TARGETS)
# Set up debug mesh if path provided
if not debug_mesh_path.is_empty():
debug_mesh = get_node(debug_mesh_path)
if debug_mesh:
var debug_material = debug_mesh.get_surface_override_material(0)
debug_material.albedo_texture = sub_viewport.get_texture()
debug_mesh.set_surface_override_material(0, debug_material)
# Create shared shader material
var shader_material = ShaderMaterial.new()
shader_material.shader = shader
# Precompute static target parameters and set them in the shader once
var centers: Array = [] # Array of Vector3 centers
var extents: Array = [] # Array of Vector3 half-sizes
var inv_bases: Array = [] # Array of mat3 (precomputed inverse bases)
for i in range(MAX_TARGETS):
if i < targets.size():
var collision_shape = targets[i].get_node_or_null("CollisionShape3D") as CollisionShape3D
if collision_shape and collision_shape.shape is BoxShape3D:
var box: BoxShape3D = collision_shape.shape
var gxf: Transform3D = collision_shape.global_transform
print("Target %d:" % i)
print(" Box extents: ", box.extents)
print(" Global transform origin: ", gxf.origin)
print(" Global transform basis: ", gxf.basis)
centers.append(gxf.origin)
extents.append(box.extents)
inv_bases.append(gxf.basis.inverse()) # Precompute the inverse once
else:
centers.append(Vector3.ZERO)
extents.append(Vector3.ZERO) # Zero extents means this target is disabled
inv_bases.append(Basis.IDENTITY)
else:
centers.append(Vector3.ZERO)
extents.append(Vector3.ZERO)
inv_bases.append(Basis.IDENTITY)
shader_material.set_shader_parameter("target_center", centers)
shader_material.set_shader_parameter("target_extents", extents)
shader_material.set_shader_parameter("target_inv_basis", inv_bases)
# Create clones of each particle system
for path in source_particles_paths:
var source_particles = get_node(path) as GPUParticles3D
if source_particles:
# Clone the source particles
var clone = source_particles.duplicate()
# disable trails since they add extra meshes
clone.trail_enabled = false
# replace the draw pass with a simple quad mesh
clone.draw_pass_1 = QuadMesh.new()
# and set the shared material
clone.draw_pass_1.surface_set_material(0, shader_material)
# set visual layer to 20 so it doesn't draw on main camera
clone.layers = 0
clone.set_layer_mask_value(20, true)
# add clone as child of SubViewport
sub_viewport.add_child(clone)
clone_particles.append(clone)
func _process(_delta):
# Each frame, read back the 8 pixel wide image from the SubViewport
var viewport_tex = sub_viewport.get_texture()
if viewport_tex:
var img: Image = viewport_tex.get_image()
if img:
for x in range(targets.size()):
var col: Color = img.get_pixel(x, 0)
var occupancy: float = col.r # The red channel intensity
targets[x].set_occupancy(occupancy)