-
Notifications
You must be signed in to change notification settings - Fork 40
/
tranform.js
110 lines (98 loc) · 3.97 KB
/
tranform.js
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
/**
* Transform class for mapping
* joints data from video stream space
* into Babylon 3D space
*/
export default class Transform{
/**
* the class constructor
* @param {Joints} _joints
*/
constructor(_joints){
this.joints = _joints;
}
/**
* Updates joints data
* @param {array} _keypoints raw joints data from posenet
* @param {float} treshHoldScore for determining whether to update or no
*/
updateKeypoints(_keypoints, treshHoldScore){
this.keypoints = {};
_keypoints.forEach(({ score, part, position }) => {
if (score > treshHoldScore) this.keypoints[part] = position;
});
this.distance = null;
this.headCenter = null;
this.shoulderCenter = null;
this.calibrate();
};
/**
* Makes the system invariant to scale and translations,
* given joints data
*/
calibrate(){
if (this.keypoints['leftEye'] && this.keypoints['rightEye']){
const left_x = this.keypoints['leftEye'].x;
const left_y = this.keypoints['leftEye'].y;
const right_x = this.keypoints['rightEye'].x;
const right_y = this.keypoints['rightEye'].y;
this.distance = Math.sqrt(Math.pow(left_x - right_x, 2) + Math.pow(left_y - right_y, 2));
this.headCenter = {'x':(left_x + right_x) / 2.0, 'y':(left_y + right_y) / 2.0};
}
if (this.keypoints['leftShoulder'] && this.keypoints['rightShoulder']) {
const left_x = this.keypoints['leftShoulder'].x;
const left_y = this.keypoints['leftShoulder'].y;
const right_x = this.keypoints['rightShoulder'].x;
const right_y = this.keypoints['rightShoulder'].y;
this.shoulderCenter = { 'x': (left_x + right_x) / 2.0, 'y': (left_y + right_y) / 2.0 };
}
}
/** Updates head joint data */
head(){
if(this.keypoints['nose'] && this.headCenter && this.shoulderCenter){
var x = this.keypoints['nose'].x;
var y = this.keypoints['nose'].y;
// get nose relative points from origin
x = (this.headCenter.x - x)/(this.distance/15);
y = this.shoulderCenter.y - y;
// normalize (i.e. scale it)
y = this.map(y,this.distance*1.5,this.distance*2.8,-2,2);
// console.log(140/this.distance,260/this.distance);
this.joints.update('head', { x, y });
return { x, y };
}
}
/**
* Updates joints data and returns angle between three joints
* @param {integer} jointA index of a joint
* @param {intger} jointB index of a joint
* @param {intger} jointC index of a joint
* @returns {float} angle
*/
rotateJoint(jointA, jointB, jointC){
if (this.keypoints[jointA] && this.keypoints[jointB] && this.keypoints[jointC]){
const angle = this.findAngle(this.keypoints[jointA], this.keypoints[jointB], this.keypoints[jointC]);
const sign = (this.keypoints[jointC].y > this.keypoints[jointB].y) ? 1 : -1;
this.joints.update(jointB, sign * angle);
return angle;
}
}
/** Maps from one linear interpolation into another one */
map(original, in_min, in_max, out_min, out_max) {
return (original - in_min) * (out_max - out_min) / (in_max - in_min) + out_min;
}
/**
* Returns angle in radians given three points p1, p2, p3
* @param {integer} p1
* @param {integer} p2
* @param {integer} p3
* @returns {float}
*/
findAngle(p1,p2,p3){
const p12 = Math.sqrt(Math.pow((p1.x - p2.x), 2) + Math.pow((p1.y - p2.y), 2));
const p13 = Math.sqrt(Math.pow((p1.x - p3.x), 2) + Math.pow((p1.y - p3.y), 2));
const p23 = Math.sqrt(Math.pow((p2.x - p3.x), 2) + Math.pow((p2.y - p3.y), 2));
const resultRadian = Math.acos(((Math.pow(p12, 2)) + (Math.pow(p13, 2)) - (Math.pow(p23, 2))) / (2 * p12 * p13));
return resultRadian;
}
}