mirror of
https://github.com/Outer-Wilds-New-Horizons/new-horizons.git
synced 2025-12-11 20:15:44 +01:00
133 lines
5.3 KiB
C#
133 lines
5.3 KiB
C#
using NewHorizons.Utility.OuterWilds;
|
|
using System.Collections;
|
|
using UnityEngine;
|
|
|
|
namespace NewHorizons.Components;
|
|
|
|
/// <summary>
|
|
/// properly add physics to a detail
|
|
/// </summary>
|
|
[DisallowMultipleComponent]
|
|
public class AddPhysics : MonoBehaviour
|
|
{
|
|
[Tooltip("The sector that the rigidbody will be simulated in, or none for it to always be on.")]
|
|
public Sector Sector;
|
|
[Tooltip("The mass of the physics object.\n" +
|
|
"Most pushable props use the default value, which matches the player mass.")]
|
|
public float Mass = 0.001f;
|
|
[Tooltip("The radius that the added sphere collider will use for physics collision.\n" +
|
|
"If there's already good colliders on the detail, you can make this 0.")]
|
|
public float Radius = 1f;
|
|
|
|
private OWRigidbody _parentBody;
|
|
private OWRigidbody _body;
|
|
private ImpactSensor _impactSensor;
|
|
private bool _maintainVelocity;
|
|
private Vector3 _centerOfMass;
|
|
|
|
private IEnumerator Start()
|
|
{
|
|
// detectors dont detect unless we wait for some reason
|
|
yield return new WaitForSeconds(10f);
|
|
|
|
_parentBody = GetComponentInParent<OWRigidbody>();
|
|
|
|
// hack: make all mesh colliders convex
|
|
// triggers are already convex
|
|
// prints errors for non readable meshes but whatever
|
|
foreach (var meshCollider in GetComponentsInChildren<MeshCollider>(true))
|
|
meshCollider.convex = true;
|
|
|
|
var bodyGo = new GameObject($"{name}_Body");
|
|
bodyGo.SetActive(false);
|
|
bodyGo.transform.parent = transform.parent;
|
|
bodyGo.transform.position = transform.position;
|
|
bodyGo.transform.rotation = transform.rotation;
|
|
|
|
_body = bodyGo.AddComponent<OWRigidbody>();
|
|
_body._simulateInSector = Sector;
|
|
_body._autoGenerateCenterOfMass = false;
|
|
_body._centerOfMass = _body.transform.InverseTransformPoint(_parentBody.GetWorldCenterOfMass());
|
|
|
|
bodyGo.layer = Layer.PhysicalDetector;
|
|
bodyGo.tag = "DynamicPropDetector";
|
|
// this collider is not included in groups. oh well
|
|
bodyGo.AddComponent<SphereCollider>().radius = Radius;
|
|
var shape = bodyGo.AddComponent<SphereShape>();
|
|
shape._collisionMode = Shape.CollisionMode.Detector;
|
|
shape._layerMask = (int)(Shape.Layer.Default | Shape.Layer.Gravity);
|
|
shape._radius = Radius;
|
|
bodyGo.AddComponent<DynamicForceDetector>();
|
|
var fluidDetector = bodyGo.AddComponent<DynamicFluidDetector>();
|
|
fluidDetector._buoyancy = Locator.GetProbe().GetOWRigidbody()._attachedFluidDetector._buoyancy;
|
|
fluidDetector._splashEffects = Locator.GetProbe().GetOWRigidbody()._attachedFluidDetector._splashEffects;
|
|
|
|
_impactSensor = bodyGo.AddComponent<ImpactSensor>();
|
|
var audioSource = bodyGo.AddComponent<AudioSource>();
|
|
audioSource.maxDistance = 30;
|
|
audioSource.dopplerLevel = 0;
|
|
audioSource.rolloffMode = AudioRolloffMode.Custom;
|
|
audioSource.playOnAwake = false;
|
|
audioSource.spatialBlend = 1;
|
|
var owAudioSource = bodyGo.AddComponent<OWAudioSource>();
|
|
owAudioSource._audioSource = audioSource;
|
|
owAudioSource._track = OWAudioMixer.TrackName.Environment;
|
|
var objectImpactAudio = bodyGo.AddComponent<ObjectImpactAudio>();
|
|
objectImpactAudio._minPitch = 0.4f;
|
|
objectImpactAudio._maxPitch = 0.6f;
|
|
objectImpactAudio._impactSensor = _impactSensor;
|
|
|
|
bodyGo.SetActive(true);
|
|
|
|
transform.parent = bodyGo.transform;
|
|
_body.SetMass(Mass);
|
|
_body.SetVelocity(_parentBody.GetPointVelocity(_body.GetWorldCenterOfMass()));
|
|
_body.SetAngularVelocity(_parentBody.GetAngularVelocity());
|
|
// hack: make center of mass at the parent body so it doesn't drift on rotating body
|
|
// _centerOfMass = _body.GetCenterOfMass();
|
|
// _body.SetCenterOfMass(_body.transform.InverseTransformPoint(_parentBody.GetWorldCenterOfMass()));
|
|
// _impactSensor.OnImpact += OnImpact;
|
|
|
|
// #536 - Physics objects in bramble dimensions not disabled on load
|
|
// sectors wait 3 frames and then call OnSectorOccupantsUpdated
|
|
// however we wait .1 real seconds which is longer
|
|
// so we have to manually call this
|
|
if (_body._simulateInSector != null)
|
|
_body.OnSectorOccupantsUpdated();
|
|
|
|
// it drifts otherwise for some reason
|
|
_maintainVelocity = true;
|
|
yield return new WaitForSeconds(10f);
|
|
_maintainVelocity = false;
|
|
}
|
|
|
|
private void FixedUpdate()
|
|
{
|
|
if (_maintainVelocity)
|
|
{
|
|
_body.SetVelocity(_parentBody.GetPointVelocity(_body.GetWorldCenterOfMass()));
|
|
_body.SetAngularVelocity(_parentBody.GetAngularVelocity());
|
|
}
|
|
}
|
|
|
|
private void OnImpact(ImpactData impact)
|
|
{
|
|
// revert it back to normal
|
|
_body.SetCenterOfMass(_centerOfMass);
|
|
_impactSensor.OnImpact -= OnImpact;
|
|
Destroy(this);
|
|
}
|
|
|
|
private void OnDrawGizmosSelected()
|
|
{
|
|
Gizmos.DrawWireSphere(transform.position, Radius);
|
|
}
|
|
|
|
private void OnRenderObject()
|
|
{
|
|
Popcron.Gizmos.Sphere(transform.position, Radius);
|
|
if (_body) Popcron.Gizmos.Line(transform.position, _body.GetWorldCenterOfMass(), Color.red);
|
|
if (_parentBody) Popcron.Gizmos.Line(transform.position, _parentBody.GetWorldCenterOfMass(), Color.green);
|
|
}
|
|
}
|