-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathhead_pose_from_image.py
103 lines (76 loc) · 3.17 KB
/
head_pose_from_image.py
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
#!/usr/bin/env python3
import os
import cv2
import sys
import dlib
import argparse
import numpy as np
from drawFace import draw
import reference_world as world
PREDICTOR_PATH = os.path.join("models", "shape_predictor_68_face_landmarks.dat")
if not os.path.isfile(PREDICTOR_PATH):
print("[ERROR] USE models/downloader.sh to download the predictor")
sys.exit()
parser = argparse.ArgumentParser()
parser.add_argument("-i", "--image",
type=str,
help="image location for pose estimation")
parser.add_argument("-f", "--focal",
type=float,
help="Callibrated Focal Length of the camera")
args = parser.parse_args()
def main(image):
detector = dlib.get_frontal_face_detector()
predictor = dlib.shape_predictor(PREDICTOR_PATH)
while True:
im = cv2.imread(image)
faces = detector(cv2.cvtColor(im, cv2.COLOR_BGR2RGB), 0)
face3Dmodel = world.ref3DModel()
for face in faces:
shape = predictor(cv2.cvtColor(im, cv2.COLOR_BGR2RGB), face)
draw(im, shape)
refImgPts = world.ref2dImagePoints(shape)
height, width, channel = im.shape
focalLength = args.focal * width
cameraMatrix = world.cameraMatrix(focalLength, (height / 2, width / 2))
mdists = np.zeros((4, 1), dtype=np.float64)
# calculate rotation and translation vector using solvePnP
success, rotationVector, translationVector = cv2.solvePnP(
face3Dmodel, refImgPts, cameraMatrix, mdists)
noseEndPoints3D = np.array([[0, 0, 1000.0]], dtype=np.float64)
noseEndPoint2D, jacobian = cv2.projectPoints(
noseEndPoints3D, rotationVector, translationVector, cameraMatrix, mdists)
# draw nose line
p1 = (int(refImgPts[0, 0]), int(refImgPts[0, 1]))
p2 = (int(noseEndPoint2D[0, 0, 0]), int(noseEndPoint2D[0, 0, 1]))
cv2.line(im, p1, p2, (110, 220, 0),
thickness=2, lineType=cv2.LINE_AA)
# calculating angle
rmat, jac = cv2.Rodrigues(rotationVector)
angles, mtxR, mtxQ, Qx, Qy, Qz = cv2.RQDecomp3x3(rmat)
print('*' * 80)
print("Angle: ", angles)
# print(f"Qx:{Qx}\tQy:{Qy}\tQz:{Qz}\t")
x = np.arctan2(Qx[2][1], Qx[2][2])
y = np.arctan2(-Qy[2][0], np.sqrt((Qy[2][1] * Qy[2][1] ) + (Qy[2][2] * Qy[2][2])))
z = np.arctan2(Qz[0][0], Qz[1][0])
print("AxisX: ", x)
print("AxisY: ", y)
print("AxisZ: ", z)
print('*' * 80)
gaze = "Looking: "
if angles[1] < -15:
gaze += "Left"
elif angles[1] > 15:
gaze += "Right"
else:
gaze += "Forward"
cv2.putText(im, gaze, (20, 20), cv2.FONT_HERSHEY_SIMPLEX, 1, (0, 255, 80), 2)
cv2.imshow("Head Pose", im)
key = cv2.waitKey(10) & 0xFF
if key == 27:
cv2.imwrite(f"joye-{gaze}.jpg", im)
break
cv2.destroyAllWindows()
if __name__ == "__main__":
main(args.image)