using System.Diagnostics; using System.Collections; using System.Collections.Generic; using System; using Godot; namespace Rokojori { [Tool] [GlobalClass] public partial class MouseEditorCamera:VirtualCamera3D { [Export] public Vector3 target; [Export] public float yaw = 0; [Export] public float pitch = 0; [Export] public float distance = 10; float smoothDistance = 10; [ExportGroup("Orbit")] [Export] public float yawSpeed = 1; [Export] public float pitchSpeed = 1; [Export] public float minPitch = -89; [Export] public float maxPitch = 89; [Export] public Sensor orbitButton; [Export] public Sensor[] orbitModifierButtons; [ExportGroup("Pan")] [Export] public float panSpeedX = 0.01f; [Export] public float panSpeedY = 0.01f; [Export] public Sensor panButton; [Export] public Sensor[] panModifierButtons; [ExportGroup("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(); [ExportGroup("Move")] [Export] public Sensor forwardButton; [Export] public Sensor backwardsButton; [Export] public Sensor leftButton; [Export] public Sensor rightButton; [Export] public Sensor upButton; [Export] public Sensor downButton; [Export] public float moveSpeed = 1; public override void _Process( double delta ) { Orbit(); Pan(); Zoom(); Move(); Apply( (float) delta ); if ( ! hasMotionDelta ) { motionDelta.X = 0; motionDelta.Y = 0; } hasMotionDelta = false; } bool hasMotionDelta = false; Vector2 motionDelta = Vector2.Zero; public override void _Input( InputEvent inputEvent ) { var mouseMotionEvent = inputEvent as InputEventMouseMotion; if ( mouseMotionEvent == null ) { return; } motionDelta = mouseMotionEvent.ScreenRelative; hasMotionDelta = true; } void Orbit() { if ( ! Sensors.IsActive( orbitButton ) ) { return; } yaw += motionDelta.X * yawSpeed; pitch += motionDelta.Y * pitchSpeed; pitch = Mathf.Clamp( pitch, minPitch, maxPitch ); } void Pan() { if ( ! Sensors.IsActive( panButton ) ) { return; } var xAmount = motionDelta.X * smoothDistance * GlobalBasis.X * panSpeedX; var yAmount = motionDelta.Y * smoothDistance * GlobalBasis.Y * panSpeedY; target += xAmount + yAmount; } void Zoom() { if ( Sensors.IsActive( zoomInButton ) ) { distance *= Mathf.Pow( 1 + zoomStepInPercentage / 100f, 1 ); } if ( Sensors.IsActive( zoomOutButton ) ) { distance *= Mathf.Pow( 1 + zoomStepInPercentage / 100f, -1 ); } distance = Mathf.Clamp( distance, minDistance, maxDistance ); } Vector3 moveDirection = Vector3.Zero; void Move() { moveDirection = Vector3.Zero; if ( Sensors.IsActive( forwardButton ) || Sensors.IsActive( backwardsButton ) ) { moveDirection = ( Sensors.GetValue( forwardButton ) - Sensors.GetValue( backwardsButton ) ) * this.GlobalForward(); } if ( Sensors.IsActive( rightButton ) || Sensors.IsActive( leftButton ) ) { moveDirection = ( Sensors.GetValue( rightButton ) - Sensors.GetValue( leftButton ) ) * this.GlobalRight(); } if ( Sensors.IsActive( upButton ) || Sensors.IsActive( downButton ) ) { moveDirection = ( Sensors.GetValue( upButton ) - Sensors.GetValue( downButton ) ) * this.GlobalUp(); } moveDirection = moveDirection.Normalized() * moveSpeed; } 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; target -= moveDirection * delta; GlobalPosition = target + forward; } } }