-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathJiggleBone.cs
126 lines (102 loc) · 4.4 KB
/
JiggleBone.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
// ============================================================
// Name: Jiggle Bone v.1.0
// Author: Michael Cook (Fishypants)
// Date: 9-25-2011
// License: Free to use. Any credit would be nice :)
//
// To Use:
// Drag this script onto a bone. (ideally bones at the end)
// Set the boneAxis to be the front facing axis of the bone.
// Done! Now you have bones with jiggle dynamics.
//
// ============================================================
using System.Collections;
namespace JiggleBone {
public class JiggleBone : MonoBehaviour {
Transform transform = new Transform();
public bool debugMode = true;
// Target and dynamic positions
public Vector3 targetPos = new Vector3();
public Vector3 dynamicPos = new Vector3();
// Bone settings
public Vector3 boneAxis = new Vector3(0,0,1);
public float targetDistance = 2.0f;
// Dynamics settings
public float bStiffness = 0.1f;
public float bMass = 0.9f;
public float bDamping = 0.75f;
public float bGravity = 0.75f;
// Dynamics variables
public Vector3 force = new Vector3();
public Vector3 acc = new Vector3();
public Vector3 vel = new Vector3();
// Squash and stretch variables
public bool SquashAndStretch = true;
public float sideStretch = 0.15f;
public float frontStretch = 0.2f;
public void Awake(){
// Set targetPos and dynamicPos at startup
Vector3 targetPos = transform.position + transform.TransformDirection(new Vector3((boneAxis.x * targetDistance),(boneAxis.y * targetDistance),(boneAxis.z * targetDistance)));
dynamicPos = targetPos;
}
public void LateUpdate(){
// Reset the bone rotation so we can recalculate the upVector and forwardVector
transform.rotation = new Quaternion();
// Update forwardVector and upVector
Vector3 forwardVector = transform.TransformDirection(new Vector3((boneAxis.x * targetDistance),(boneAxis.y * targetDistance),(boneAxis.z * targetDistance)));
Vector3 upVector = transform.TransformDirection(new Vector3(0,1,0));
// Calculate target position
Vector3 targetPos = transform.position + transform.TransformDirection(new Vector3((boneAxis.x * targetDistance),(boneAxis.y * targetDistance),(boneAxis.z * targetDistance)));
// Calculate force, acceleration, and velocity per X, Y and Z
force.x = (targetPos.x - dynamicPos.x) * bStiffness;
acc.x = force.x / bMass;
vel.x += acc.x * (1 - bDamping);
force.y = (targetPos.y - dynamicPos.y) * bStiffness;
force.y -= bGravity / 10; // Add some gravity
acc.y = force.y / bMass;
vel.y += acc.y * (1 - bDamping);
force.z = (targetPos.z - dynamicPos.z) * bStiffness;
acc.z = force.z / bMass;
vel.z += acc.z * (1 - bDamping);
// Update dynamic postion
dynamicPos += vel + force;
// Set bone rotation to look at dynamicPos
transform.LookAt(dynamicPos, upVector);
// ==================================================
// Squash and Stretch section
// ==================================================
if(SquashAndStretch){
// Create a vector from target position to dynamic position
// We will measure the magnitude of the vector to determine
// how much squash and stretch we will apply
Vector3 dynamicVec = dynamicPos - targetPos;
// Get the magnitude of the vector
float stretchMag = dynamicVec.magnitude;
// Here we determine the amount of squash and stretch based on stretchMag
// and the direction the Bone Axis is pointed in. Ideally there should be
// a vector with two values at 0 and one at 1. Like Vector3(0,0,1)
// for the 0 values, we assume those are the sides, and 1 is the direction
// the bone is facing
float xStretch;
if(boneAxis.x == 0) xStretch = 1 + (-stretchMag * sideStretch);
else xStretch = 1 + (stretchMag * frontStretch);
float yStretch;
if(boneAxis.y == 0) yStretch = 1 + (-stretchMag * sideStretch);
else yStretch = 1 + (stretchMag * frontStretch);
float zStretch;
if(boneAxis.z == 0) zStretch = 1 + (-stretchMag * sideStretch);
else zStretch = 1 + (stretchMag * frontStretch);
// Set the bone scale
transform.localScale = new Vector3(xStretch, yStretch, zStretch);
}
// ==================================================
// DEBUG VISUALIZATION
// ==================================================
// Green line is the bone's local up vector
// Blue line is the bone's local foward vector
// Yellow line is the target postion
// Red line is the dynamic postion
// ==================================================
}
}
}