implementation of drecon in unity 2022 lts forked from: https://github.com/joanllobera/marathon-envs
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

239 lines
9.0 KiB

10 months ago
// A class that describes a body part of an Agent or an Animator. The object
// stores the rigid body associated with an Agent's body part. Along with some
// important data like root bone, i. e. 'butt' for a marathonMan. Implments methods
// to calculate distance to the characteristics of an animation being mimiced.
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using UnityEngine;
using Unity.MLAgents;
[System.Serializable]
public class BodyPart002
{
public string Name;
public BodyHelper002.BodyPartGroup Group;
public Vector3 ObsLocalPosition;
public Quaternion ObsRotation;
public Quaternion ObsRotationFromBase;
public Vector3 ObsRotationVelocity;
public Vector3 ObsVelocity;
public float ObsAngleDeltaFromAnimationRotation;
public Vector3 ObsDeltaFromAnimationPosition;
public Vector3 ObsDeltaFromAnimationVelocity;
public Vector3 ObsDeltaFromAnimationAngularVelocity;
public Vector3 ObsDeltaFromAnimationAngularVelocityWorld;
public Vector3 DebugMaxRotationVelocity;
public Vector3 DebugMaxVelocity;
public Quaternion DefaultLocalRotation;
public Quaternion ToJointSpaceInverse;
public Quaternion ToJointSpaceDefault;
public Rigidbody Rigidbody;
public Transform Transform;
public BodyPart002 Root;
public Quaternion InitialRootRotation;
public Vector3 InitialRootPosition;
// base = from where to measure rotation and position from
public Quaternion BaseRotation;
public Vector3 BasePosition;
//
public Quaternion ToFocalRoation;
Quaternion _lastObsRotation;
Quaternion _lastWorldRotation;
Vector3 _lastLocalPosition;
Vector3 _lastWorldPosition;
Vector3 _animationAngularVelocity;
Vector3 _animationAngularVelocityWorld;
Vector3 _animationVelocityWorld;
DecisionRequester _decisionRequester;
float _lastUpdateObsTime;
bool _firstRunComplete;
bool _hasRanVeryFirstInit;
private Vector3 _animationPositionWorld;
private Quaternion _animationRotation;
static Vector3 Vector3Max (Vector3 a, Vector3 b)
{
var answer = new Vector3(
Mathf.Max(Mathf.Abs(a.x), Mathf.Abs(b.x)),
Mathf.Max(Mathf.Abs(a.y), Mathf.Abs(b.y)),
Mathf.Max(Mathf.Abs(a.z), Mathf.Abs(b.z)));
return answer;
}
// Initialize
public void Init()
{
_decisionRequester = GameObject.Find("MarathonMan").GetComponent<DecisionRequester>();
_firstRunComplete = false;
if (Rigidbody != null){
Rigidbody.angularVelocity = Vector3.zero;
Rigidbody.velocity = Vector3.zero;
}
if (!_hasRanVeryFirstInit) {
InitialRootRotation = Root.Transform.transform.rotation;
InitialRootPosition = Root.Transform.transform.position;
BaseRotation = Root.Transform.transform.rotation;
BasePosition = Root.Transform.transform.position;
DefaultLocalRotation = LocalRotation;
Vector3 forward = this.Transform.forward;
Vector3 up = this.Transform.up;
Quaternion toJointSpace = Quaternion.LookRotation(forward, up);
ToJointSpaceInverse = Quaternion.Inverse(toJointSpace);
ToJointSpaceDefault = DefaultLocalRotation * toJointSpace;
// set body part direction
Vector3 focalOffset = new Vector3(10,0,0);
if (Rigidbody != null){
var focalPoint = Rigidbody.position + focalOffset;
ToFocalRoation = Rigidbody.rotation;
ToFocalRoation.SetLookRotation(focalPoint - Rigidbody.position);
}
_hasRanVeryFirstInit = true;
}
}
// Update values like ObsLocalPosition, ObsRotation, ... that will be accessed
// by the agent as observations for the neural network. Also calculates distances
// to an animation being mimicked
public void UpdateObservations()
{
Quaternion rotation;
Vector3 position;
if (this == Root) {
rotation = Quaternion.Inverse(InitialRootRotation) * Transform.rotation;
position = Transform.position - InitialRootPosition;
}
else {
rotation = Quaternion.Inverse(Root.Transform.rotation) * Transform.rotation;
position = Transform.position - Root.Transform.position;
}
if (_firstRunComplete == false){
_lastUpdateObsTime = Time.time;
_lastLocalPosition = position;
_lastWorldPosition = Transform.position;
_lastObsRotation = rotation;
_lastWorldRotation = Transform.rotation;
}
var dt = Time.fixedDeltaTime * _decisionRequester.DecisionPeriod;
var velocity = (position - _lastLocalPosition)/dt;
var velocityWorld = (Transform.position - _lastWorldPosition)/dt;
var angularVelocity = JointHelper002.CalcDeltaRotationNormalizedEuler(_lastObsRotation, rotation)/dt;
var angularVelocityWorld = JointHelper002.CalcDeltaRotationNormalizedEuler(_lastWorldRotation, Transform.rotation)/dt;
// old calulation for observation vector
//angularVelocity = NormalizedEulerAngles(rotationVelocity.eulerAngles);
//angularVelocity /= 128f;
// old calculation end
_lastUpdateObsTime = Time.time;
_lastLocalPosition = position;
_lastWorldPosition = Transform.position;
_lastObsRotation = rotation;
_lastWorldRotation = Transform.rotation;
//if (Name == "right_right_foot") {
// Debug.Log("^^^^^^^^^^^^");
// Debug.Log("body part name: " + Name);
// Debug.Log("animation angular velocity:" + _animationAngularVelocity);
// Debug.Log("angular velocity:" + angularVelocity);
// Debug.Log("animation angular velocity world:" + _animationAngularVelocityWorld);
// Debug.Log("angular velocity world:" + angularVelocityWorld);
// Debug.Log("rotation local:" + rotation.eulerAngles);
// Debug.Log("animation rotation local: " + _animationRotation.eulerAngles);
// Debug.Log("velocity world: " + velocityWorld);
// Debug.Log("animation velocity world:" + _animationVelocityWorld);
// Debug.Log("transform position:" + Transform.position);
// Debug.Log("animation position world: " + _animationPositionWorld);
// Debug.Log("dt:" + dt);
//}
ObsLocalPosition = position;
ObsRotation = rotation;
ObsRotationVelocity = angularVelocity;
ObsVelocity = velocity;
ObsDeltaFromAnimationPosition = _animationPositionWorld - Transform.position;
ObsAngleDeltaFromAnimationRotation = Quaternion.Angle(_animationRotation, rotation);
ObsAngleDeltaFromAnimationRotation = JointHelper002.NormalizedAngle(ObsAngleDeltaFromAnimationRotation);
ObsDeltaFromAnimationVelocity = _animationVelocityWorld - velocityWorld;
ObsDeltaFromAnimationAngularVelocity = (_animationAngularVelocity - angularVelocity);
ObsDeltaFromAnimationAngularVelocityWorld = (_animationAngularVelocityWorld - angularVelocityWorld);
DebugMaxRotationVelocity = Vector3Max(DebugMaxRotationVelocity, angularVelocity);
DebugMaxVelocity = Vector3Max(DebugMaxVelocity, velocity);
_firstRunComplete = true;
}
// returns local rotation
public Quaternion LocalRotation {
get {
return Quaternion.Inverse(RootRotation) * Transform.rotation;
}
}
// retuns root rotation
public Quaternion RootRotation{
get {
return InitialRootRotation;
}
}
// Set the position, rotation of a body part to match animation's position and
// rotation. Also sets the rigid body's position and velocity, which is used
// in angular momentum calculation
public void MoveToAnim(Vector3 animPosition, Quaternion animRotation, Vector3 angularVelocity, Vector3 velocity)
{
Transform.position = animPosition;
Transform.rotation = animRotation;
if (Rigidbody != null){
foreach (var childRb in Rigidbody.GetComponentsInChildren<Rigidbody>())
{
if (childRb == Rigidbody)
continue;
childRb.transform.localPosition = Vector3.zero;
childRb.transform.localEulerAngles = Vector3.zero;
childRb.angularVelocity = Vector3.zero;
childRb.velocity = Vector3.zero;
}
Rigidbody.angularVelocity = angularVelocity;
Rigidbody.velocity = velocity;
}
}
// Set the position of the animation being mimicked
public void SetAnimationPosition(Vector3 animPositionWorld, Quaternion animRotationLocal, Vector3 animVelocityWorld, Vector3 animAngularVelocityLocal, Vector3 animAngularVelocityWorld)
{
_animationPositionWorld = animPositionWorld;
_animationRotation = animRotationLocal;
_animationVelocityWorld = animVelocityWorld;
_animationAngularVelocity = animAngularVelocityLocal;
_animationAngularVelocityWorld = animAngularVelocityWorld;
}
}