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

281 lines
6.5 KiB
C#

using System.Diagnostics;
using System.Collections;
using System.Collections.Generic;
using System;
using Godot;
namespace Rokojori
{
[Tool]
[GlobalClass]
public partial class StrategyTopDownCamera:VirtualCamera3D
{
public enum ConstrainMode
{
Unconstrained,
Axis_Aligned_Box,
Circle_XZ_Range_Y
}
[Export]
public Vector3 target;
[Export]
public float yaw = 0;
[Export]
public float pitch = 0;
[Export]
public float distance = 10;
float smoothDistance = 10;
[ExportCategory("Move")]
[Export]
public ConstrainMode constrainMode = ConstrainMode.Unconstrained;
[Export]
public Node3D constrainMin;
[Export]
public Node3D constrainMax;
[Export]
public Sensor moveUpButton;
[Export]
public Sensor moveDownButton;
[Export]
public Sensor moveRightButton;
[Export]
public Sensor moveLeftButton;
[Export]
public float buttonMoveSpeed = 1f;
[Export]
public Sensor mouseMoveButton;
[Export]
public float mouseMoveSpeed = 0.001f;
[Export]
public bool flipHorizontal = false;
[Export]
public bool flipVertical = false;
[Export]
public bool moveAtBorders = true;
[Export]
public float moveAtBorderSpeed = 0.001f;
[Export]
public float borderSizePercentage = 5;
[ExportCategory("Orbit")]
[Export]
public Sensor orbitRightButton;
[Export]
public Sensor orbitLeftButton;
[Export]
public Sensor orbitMouseButton;
[Export]
public float mouseOrbitSpeedScale = -0.5f;
[Export]
public float orbitSpeed = 0.001f;
[ExportCategory("Zoom")]
[Export]
public float zoomStepInPercentage = 10;
[Export]
public float minDistance = 0.001f;
[Export]
public float maxDistance = 200f;
[Export]
public Sensor zoomInButton;
[Export]
public Sensor[] zoomInModifierButtons;
[Export]
public Sensor zoomOutButton;
[Export]
public Sensor[] zoomOutModifierButtons;
[Export]
public float zoomSmoothingCoefficient = 0.1f;
Smoother smoother = new Smoother();
public override void _Process( double delta )
{
Move();
Orbit( (float) delta );
Zoom();
Apply( (float) delta );
if ( ! hasMotionDelta )
{
motionDelta.X = 0;
motionDelta.Y = 0;
}
hasMotionDelta = false;
}
bool hasMotionDelta = false;
Vector2 motionDelta = Vector2.Zero;
float borderMoveHorizontal = 0;
float borderMoveVertical = 0;
float GetBorderSize()
{
var size = GetViewport().GetVisibleRect().Size;
var borderX = ( size.X * borderSizePercentage ) / 100f;
var borderY = ( size.Y * borderSizePercentage ) / 100f;
return Mathf.Min( borderX, borderY );
}
public override void _Input( InputEvent inputEvent )
{
if ( inputEvent is InputEventMouseMotion )
{
var eventMouseMotion = inputEvent as InputEventMouseMotion;
motionDelta = eventMouseMotion.ScreenRelative;
if ( moveAtBorders )
{
var position = eventMouseMotion.Position;
var screenSize = GetViewport().GetVisibleRect().Size;
var borderSize = GetBorderSize();
borderMoveHorizontal = - MathX.PolarAxis( position.X < borderSize, position.X > screenSize.X - borderSize );
borderMoveVertical = - MathX.PolarAxis( position.Y < borderSize, position.Y > screenSize.Y - borderSize );
}
hasMotionDelta = true;
}
}
void Move()
{
var deltaX = 0f;
var deltaY = 0f;
if ( Sensors.IsActive( mouseMoveButton ) )
{
deltaX = motionDelta.X * mouseMoveSpeed;
deltaY = motionDelta.Y * mouseMoveSpeed;
}
else
{
deltaX = Sensors.PolarAxis( moveLeftButton, moveRightButton ) + borderMoveHorizontal * moveAtBorderSpeed;
deltaY = Sensors.PolarAxis( moveUpButton, moveDownButton ) + borderMoveVertical * moveAtBorderSpeed;
}
var forward = Math3D.GetYPlaneForward( this );
var right = Math3D.GetYPlaneRight( this );
var flipH = flipHorizontal ? -1 : 1;
var flipV = flipVertical ? -1 : 1;
var xAmount = deltaX * smoothDistance * right * flipH;
var zAmount = deltaY * smoothDistance * forward * flipV;
target += ( xAmount + zAmount );
ConstrainTarget();
}
void ConstrainTarget()
{
if ( ConstrainMode.Unconstrained == constrainMode )
{
return;
}
if ( ConstrainMode.Axis_Aligned_Box == constrainMode )
{
target = Box3.Constrain( target, constrainMin.GlobalPosition, constrainMax.GlobalPosition );
return;
}
if ( ConstrainMode.Circle_XZ_Range_Y == constrainMode )
{
var direction = ( constrainMax.GlobalPosition - constrainMin.GlobalPosition );
var directionXZ = direction; directionXZ.Y = 0;
var distanceXZ = directionXZ.Length();
var targetDirection = target - constrainMin.GlobalPosition;
var targetDirectionXZ = targetDirection; targetDirectionXZ.Y = 0;
target.Y = Mathf.Clamp( target.Y, constrainMin.GlobalPosition.Y, constrainMax.GlobalPosition.Y );
if ( targetDirectionXZ.Length() > distanceXZ )
{
targetDirectionXZ = targetDirectionXZ.Normalized() * distanceXZ;
var constrainedPosition = constrainMin.GlobalPosition + targetDirectionXZ;
target = new Vector3( constrainedPosition.X, target.Y, constrainedPosition.Z );
}
}
}
void Orbit( float delta )
{
var direction = Sensors.PolarAxis( orbitLeftButton, orbitRightButton );
if ( Sensors.IsActive( orbitMouseButton ) )
{
yaw += motionDelta.X * mouseOrbitSpeedScale;
}
yaw += direction * orbitSpeed * delta;
}
void Zoom()
{
var zoomButtonAxis = Sensors.PolarAxis( zoomOutButton, zoomInButton );
distance *= Mathf.Pow( 1 + zoomStepInPercentage / 100f, zoomButtonAxis );
distance = Mathf.Clamp( distance, minDistance, maxDistance );
}
void Apply( float delta )
{
smoothDistance = smoother.SmoothWithCoefficient( smoothDistance, distance, zoomSmoothingCoefficient, delta );
GlobalRotation = new Vector3( Mathf.DegToRad( pitch ), Mathf.DegToRad( yaw ), 0 );
var forward = Math3D.GetGlobalForward( this ) * smoothDistance;
GlobalPosition = target + forward;
}
}
}