shmodot/scripts/orbits/Barycenter.cs

144 lines
3.2 KiB
C#
Raw Normal View History

2022-08-31 22:18:47 +02:00
using Godot;
using System;
2022-09-07 01:49:14 +02:00
using System.Collections.Generic;
using Vim.Math3d;
2022-08-31 22:18:47 +02:00
[Tool]
2022-09-07 01:49:14 +02:00
public class Barycenter : Node, IPointMass
2022-08-31 22:18:47 +02:00
{
2022-09-07 01:49:14 +02:00
[Export]
private NodePath[] _pointMassPaths;
2022-08-31 22:18:47 +02:00
private double _semiMajorAxis;
2022-08-31 22:18:47 +02:00
[Export]
private double SemiMajorAxis
2022-08-31 22:18:47 +02:00
{
get => _semiMajorAxis;
set
2022-08-31 22:18:47 +02:00
{
_semiMajorAxis = value;
if (!Engine.EditorHint) return;
2022-08-31 22:18:47 +02:00
InvalidateGeometry();
2022-08-31 22:18:47 +02:00
}
}
private double _eccentricity;
[Export(PropertyHint.Range, "0,1")]
private double Eccentricity
2022-08-31 22:18:47 +02:00
{
get => _eccentricity;
set
{
_eccentricity = value;
2022-08-31 22:18:47 +02:00
if (!Engine.EditorHint) return;
InvalidateGeometry();
}
2022-08-31 22:18:47 +02:00
}
2022-09-07 01:49:14 +02:00
public double Mass
2022-08-31 22:18:47 +02:00
{
get
2022-08-31 22:18:47 +02:00
{
2022-09-07 01:49:14 +02:00
double mass = 0;
foreach (var orbiter in Orbiters)
2022-08-31 22:18:47 +02:00
{
2022-09-07 01:49:14 +02:00
mass += orbiter.PointMass.Mass;
}
2022-09-07 01:49:14 +02:00
return mass;
2022-08-31 22:18:47 +02:00
}
}
2022-09-07 01:49:14 +02:00
public DVector3 Position { get; set; }
private ImmediateGeometry _orbitGeometry = null;
public ImmediateGeometry OrbitGeometry
2022-08-31 22:18:47 +02:00
{
get
{
2022-09-07 01:49:14 +02:00
if (_orbitGeometry == null)
{
2022-09-07 01:49:14 +02:00
_orbitGeometry = new ImmediateGeometry();
var m = new SpatialMaterial();
m.VertexColorUseAsAlbedo = true;
m.FlagsUnshaded = true;
_orbitGeometry.MaterialOverride = m;
AddChild(_orbitGeometry);
}
2022-09-07 01:49:14 +02:00
return _orbitGeometry;
2022-08-31 22:18:47 +02:00
}
}
2022-09-07 01:49:14 +02:00
private struct Orbiter
{
2022-09-07 01:49:14 +02:00
public IPointMass PointMass { get; set; }
public IOrbit Orbit { get; set; }
}
2022-09-07 01:49:14 +02:00
private Orbiter[] _orbiters = null;
private Orbiter[] Orbiters
{
get
{
2022-09-07 01:49:14 +02:00
if (_orbiters == null)
{
2022-09-07 01:49:14 +02:00
var length = _pointMassPaths.Length;
_orbiters = new Orbiter[length];
for (int i = 0; i < length; i++)
{
var pointMass = GetNode<IPointMass>(_pointMassPaths[i]);
var orbit = new Orbit(this, pointMass);
var orbiter = new Orbiter
{
PointMass = pointMass,
Orbit = orbit
};
_orbiters[i] = orbiter;
}
}
2022-09-07 01:49:14 +02:00
return _orbiters;
}
}
2022-09-07 01:49:14 +02:00
private IOrbit[] _orbits;
private IOrbit[] Orbits
{
get
{
2022-09-07 01:49:14 +02:00
if (_orbits == null)
{
2022-09-07 01:49:14 +02:00
}
2022-09-07 01:49:14 +02:00
return _orbits;
}
}
private float _time = 0;
[Export]
private float _speed = 3f;
public override void _Process(float delta)
{
_time += delta * _speed;
InvalidateGeometry();
}
2022-08-31 22:18:47 +02:00
private void InvalidateGeometry()
2022-08-31 22:18:47 +02:00
{
2022-09-07 01:49:14 +02:00
for (int i = 0; i < Orbiters.Length; i++)
{
var orbiter = Orbiters[i];
var orbit = orbiter.Orbit;
orbit.Ellipse = new Ellipse(SemiMajorAxis, Eccentricity);
orbit.Draw(OrbitGeometry);
2022-08-31 22:18:47 +02:00
2022-09-07 01:49:14 +02:00
var t = _time + i * Math.PI;
orbiter.PointMass.Position = orbit.GetPosition(t);
}
2022-08-31 22:18:47 +02:00
}
}