rj-action-library/Runtime/VirtualCameras/VirtualCamera3DManager.cs

162 lines
3.5 KiB
C#
Raw Normal View History

2024-05-19 15:59:41 +00:00
using System.Diagnostics;
using System.Collections;
using System.Collections.Generic;
using System;
using Godot;
namespace Rokojori
{
[Tool]
[GlobalClass]
2025-01-08 18:46:17 +00:00
public partial class VirtualCamera3DManager:NetworkNode
2024-05-19 15:59:41 +00:00
{
[Export]
public Camera3D camera;
[Export]
public bool refreshSlots = false;
2024-06-16 16:02:34 +00:00
[Export]
public bool active = false;
2024-05-19 15:59:41 +00:00
public override void _Process( double delta )
{
2024-06-16 16:02:34 +00:00
if ( ! active )
{
return;
}
2024-05-19 15:59:41 +00:00
LerpCameras( delta );
}
2025-01-08 18:46:17 +00:00
[Export]
public float CameraPrioritySmoothingCoefficient = 0.1f;
[Export]
public float CameraPrioritySmoothingStepFPS = 120;
2024-05-19 15:59:41 +00:00
public float smoothStepDelta => 1f / CameraPrioritySmoothingStepFPS;
public float safeSmoothing => Mathf.Max( 0, CameraPrioritySmoothingCoefficient );
List<VirtualCamera3DSlot> _cameraSlots = new List<VirtualCamera3DSlot>();
2025-01-03 12:09:23 +00:00
public VirtualCamera3DSlot GetSlot( int index )
{
return _cameraSlots[ index ];
}
2024-05-19 15:59:41 +00:00
public void SetActiveSlot( VirtualCamera3DSlot slot )
{
_cameraSlots.ForEach( c => c.priority = c == slot ? 1 : 0 );
}
void LerpCameras( double delta )
{
if ( refreshSlots || _cameraSlots == null || _cameraSlots.Count == 0 )
{
refreshSlots = false;
_cameraSlots = Nodes.GetDirectChildren<VirtualCamera3DSlot>( this );
}
var sumPriority = 0f;
_cameraSlots.ForEach(
c =>
{
c.Update( delta, this );
sumPriority += MathF.Max( 0, c.smoothedPriority );
}
);
if ( sumPriority == 0 )
{
return;
}
var position = new Vector3();
var up = new Vector3();
var forward = new Vector3();
2024-07-25 05:40:31 +00:00
var fov = 0f;
2024-05-19 15:59:41 +00:00
_cameraSlots.ForEach(
c =>
{
var priority = MathF.Max( 0, c.smoothedPriority );
2025-01-03 12:09:23 +00:00
var rotation = c.GetCameraRotation();
2024-06-16 16:02:34 +00:00
if ( ! rotation.IsFinite() || rotation.Length() == 0 )
{
rotation = new Quaternion();
rotation.X = 0;
rotation.Y = 0;
rotation.Z = 0;
rotation.W = 1;
rotation = rotation.Normalized();
}
2024-05-19 15:59:41 +00:00
var vUp = rotation * Vector3.Up;
var vForward = rotation * Vector3.Forward;
2025-01-03 12:09:23 +00:00
position += priority * c.GetCameraPosition();
2024-05-19 15:59:41 +00:00
up += priority * vUp;
forward += priority * vForward;
2025-01-03 12:09:23 +00:00
fov += priority * c.GetCameraFOV();
2024-05-19 15:59:41 +00:00
}
);
position /= sumPriority;
2024-07-25 05:40:31 +00:00
fov /= sumPriority;
2024-05-19 15:59:41 +00:00
if ( forward.LengthSquared() == 0 )
{
forward = camera.Basis.Z;
}
else
{
forward = forward.Normalized();
}
if ( up.LengthSquared() == 0 )
{
up = camera.Basis.Y;
}
else
{
up = up.Normalized();
}
2024-12-01 17:07:41 +00:00
// RJLog.Log( "Set Cam", position );
2024-05-19 15:59:41 +00:00
camera.GlobalPosition = position;
camera.LookAt( position - forward, up );
2024-07-25 05:40:31 +00:00
camera.Fov = fov;
2024-05-19 15:59:41 +00:00
}
2025-01-08 18:46:17 +00:00
public VirtualCamera3D GetCamera( int index )
2024-05-19 15:59:41 +00:00
{
return _cameraSlots[ index ].camera;
}
2025-01-08 18:46:17 +00:00
public int GetCameraIndex( VirtualCamera3D camera3D )
2024-05-19 15:59:41 +00:00
{
return _cameraSlots.FindIndex( c => c.camera == camera3D );
}
2025-01-08 18:46:17 +00:00
public float GetCameraPriority( int index )
2024-05-19 15:59:41 +00:00
{
return _cameraSlots[ index ].priority;
}
2025-01-08 18:46:17 +00:00
public void SetCameraPriority( int index, float priority )
2024-05-19 15:59:41 +00:00
{
_cameraSlots[ index ].priority = priority;
}
}
}