66 lines
		
	
	
	
		
			2.9 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			66 lines
		
	
	
	
		
			2.9 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
| // This shader is used with a cloned GPUParticles3D system that is rendered in an offscreen SubViewport 
 | ||
| // (sized 8×2 pixels) on a dedicated layer (e.g. layer 20). Its purpose is to check each particle’s full 3D 
 | ||
| // world-space position against up to eight static target areas. Each target area is defined by a center (vec3), 
 | ||
| // extents (vec3, representing half-sizes), and a precomputed inverse basis (mat3) that converts world space into 
 | ||
| // the target’s local space. If a particle is within a target, the shader maps its geometry to a quad occupying the 
 | ||
| // horizontal slice corresponding to that target (i.e. one pixel column in the 8×1 output). The fragment outputs red 
 | ||
| // with a configurable alpha value. With additive blending, the red intensity in each slice roughly indicates the number 
 | ||
| // of particles hitting that target.
 | ||
| // 
 | ||
| // Targets with zero extents are considered disabled.
 | ||
| 
 | ||
| shader_type spatial;
 | ||
| render_mode unshaded, depth_draw_never, cull_disabled, blend_add;
 | ||
| 
 | ||
| const int MAX_TARGETS = 8;
 | ||
| 
 | ||
| uniform vec3 target_center[MAX_TARGETS];       // 3D centers for each target.
 | ||
| uniform vec3 target_extents[MAX_TARGETS];        // 3D half-sizes for each target.
 | ||
| uniform mat3 target_inv_basis[MAX_TARGETS];      // Precomputed inverse bases (inverts the target's rotation).
 | ||
| 
 | ||
| void vertex() {
 | ||
|     // Compute the particle's 3D world-space center (assumes local origin is the center).
 | ||
|     vec3 world_pos = (MODEL_MATRIX * vec4(0.0, 0.0, 0.0, 1.0)).xyz;
 | ||
|     
 | ||
|     int hit_target = -1;
 | ||
|     // Loop through targets.
 | ||
|     for (int i = 0; i < MAX_TARGETS; i++) {
 | ||
|         // Skip disabled targets.
 | ||
|         if (target_extents[i] == vec3(0.0))
 | ||
|             continue;
 | ||
|         // Transform the world position into the target's local space.
 | ||
|         vec3 local_pos = target_inv_basis[i] * (world_pos - target_center[i]);
 | ||
|         // Check if within the target's extents on all three axes.
 | ||
|         if (abs(local_pos.x) <= target_extents[i].x &&
 | ||
|             abs(local_pos.y) <= target_extents[i].y &&
 | ||
|             abs(local_pos.z) <= target_extents[i].z) {
 | ||
|             hit_target = i;
 | ||
|             break;
 | ||
|         }
 | ||
|     }
 | ||
|     
 | ||
|     if (hit_target == -1) {
 | ||
|         // Particle is not inside any target: send its geometry offscreen.
 | ||
|         POSITION = vec4(2.0, 2.0, 2.0, 1.0);
 | ||
|     } else {
 | ||
|         // Map the target index to a horizontal slice of clip space.
 | ||
|         float slice_width = 2.0 / float(MAX_TARGETS);
 | ||
|         float x_min = -1.0 + slice_width * float(hit_target);
 | ||
|         float x_max = x_min + slice_width;
 | ||
|         // Output a quad covering the full vertical range.
 | ||
|         if (VERTEX_ID == 0) {
 | ||
|             POSITION = vec4(x_min, -1.0, 0.0, 1.0);
 | ||
|         } else if (VERTEX_ID == 1) {
 | ||
|             POSITION = vec4(x_max, -1.0, 0.0, 1.0);
 | ||
|         } else if (VERTEX_ID == 2) {
 | ||
|             POSITION = vec4(x_min, 1.0, 0.0, 1.0);
 | ||
|         } else {
 | ||
|             POSITION = vec4(x_max, 1.0, 0.0, 1.0);
 | ||
|         }
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| void fragment() {
 | ||
|     ALBEDO = vec3(1.0, 0.0, 0.0);
 | ||
|     ALPHA = 0.05f;
 | ||
| }
 |