implementation of drecon in unity 2022 lts
forked from:
https://github.com/joanllobera/marathon-envs
114 lines
3.8 KiB
114 lines
3.8 KiB
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using UnityEngine;
|
|
|
|
public class TrackBodyStatesInWorldSpace : MonoBehaviour
|
|
{
|
|
[System.Serializable]
|
|
public class Stat
|
|
{
|
|
public string Name;
|
|
public Vector3 Position;
|
|
public Quaternion Rotation;
|
|
public Vector3 Velocity;
|
|
public Vector3 AngualrVelocity;
|
|
[HideInInspector]
|
|
public Vector3 LastPosition;
|
|
[HideInInspector]
|
|
public Quaternion LastRotation;
|
|
[HideInInspector]
|
|
public bool LastIsSet;
|
|
}
|
|
public List<TrackBodyStatesInWorldSpace.Stat> Stats;
|
|
|
|
internal List<Rigidbody> _rigidbodies;
|
|
|
|
// Start is called before the first frame update
|
|
public void OnAgentInitialize()
|
|
{
|
|
_rigidbodies = GetComponentsInChildren<Rigidbody>().ToList();
|
|
Stats = _rigidbodies
|
|
.Select(x=> new TrackBodyStatesInWorldSpace.Stat{Name = x.name})
|
|
.ToList();
|
|
}
|
|
|
|
void FixedUpdate()
|
|
{
|
|
if (_rigidbodies == null)
|
|
OnAgentInitialize();
|
|
|
|
float timeDelta = Time.fixedDeltaTime;
|
|
|
|
foreach (var rb in _rigidbodies)
|
|
{
|
|
Stat stat = Stats.First(x=>x.Name == rb.name);
|
|
if (!stat.LastIsSet)
|
|
{
|
|
stat.LastPosition = rb.transform.position;
|
|
stat.LastRotation = rb.transform.rotation;
|
|
}
|
|
stat.Position = rb.transform.position;
|
|
stat.Rotation = rb.transform.rotation;
|
|
stat.Velocity = rb.transform.position - stat.LastPosition;
|
|
stat.Velocity /= timeDelta;
|
|
stat.AngualrVelocity = DReConObservationStats.GetAngularVelocity(stat.LastRotation, rb.transform.rotation, timeDelta);
|
|
stat.LastPosition = rb.transform.position;
|
|
stat.LastRotation = rb.transform.rotation;
|
|
stat.LastIsSet = true;
|
|
}
|
|
}
|
|
|
|
public void LinkStatsToRigidBodies()
|
|
{
|
|
foreach (var rb in _rigidbodies)
|
|
{
|
|
Stat stat = Stats.First(x=>x.Name == rb.name);
|
|
stat.LastPosition = rb.transform.position;
|
|
stat.LastRotation = rb.transform.rotation;
|
|
stat.Position = rb.transform.position;
|
|
stat.Rotation = rb.transform.rotation;
|
|
stat.Velocity = Vector3.zero;
|
|
stat.AngualrVelocity = Vector3.zero;
|
|
stat.LastPosition = rb.transform.position;
|
|
stat.LastRotation = rb.transform.rotation;
|
|
stat.LastIsSet = true;
|
|
}
|
|
|
|
}
|
|
|
|
public void CopyStatesTo(GameObject target)
|
|
{
|
|
var targets = target.GetComponentsInChildren<ArticulationBody>().ToList();
|
|
var root = targets.First(x=>x.isRoot);
|
|
root.gameObject.SetActive(false);
|
|
foreach (var stat in Stats)
|
|
{
|
|
var targetRb = targets.First(x=>x.name == stat.Name);
|
|
targetRb.transform.position = stat.Position;
|
|
targetRb.transform.rotation = stat.Rotation;
|
|
// targetRb.velocity = stat.Velocity;
|
|
// targetRb.angularVelocity = stat.AngualrVelocity;
|
|
|
|
// var drive = targetRb.yDrive;
|
|
// drive.targetVelocity = stat.AngualrVelocity.x;
|
|
// targetRb.yDrive = drive;
|
|
|
|
// drive = targetRb.zDrive;
|
|
// drive.targetVelocity = stat.AngualrVelocity.y;
|
|
// targetRb.zDrive = drive;
|
|
|
|
// drive = targetRb.xDrive;
|
|
// drive.targetVelocity = stat.AngualrVelocity.z;
|
|
// targetRb.xDrive = drive;
|
|
|
|
targetRb.inertiaTensor = stat.Velocity;
|
|
targetRb.inertiaTensorRotation = Quaternion.Euler(stat.AngualrVelocity);
|
|
if (targetRb.isRoot)
|
|
{
|
|
targetRb.TeleportRoot(stat.Position, stat.Rotation);
|
|
}
|
|
}
|
|
root.gameObject.SetActive(true);
|
|
}
|
|
}
|
|
|