-
Notifications
You must be signed in to change notification settings - Fork 80
/
Copy pathdemo.py
executable file
·188 lines (170 loc) · 8.73 KB
/
demo.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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
#!/usr/bin/env python3
import sys
sys.path.append("../..")
from HandTrackerRenderer import HandTrackerRenderer
from Filters import LandmarksSmoothingFilter
import argparse
import numpy as np
import cv2
from o3d_utils import Visu3D
LINES_HAND = [[0,1],[1,2],[2,3],[3,4],
[0,5],[5,6],[6,7],[7,8],
[5,9],[9,10],[10,11],[11,12],
[9,13],[13,14],[14,15],[15,16],
[13,17],[17,18],[18,19],[19,20],[0,17]]
class HandTracker3DRenderer:
def __init__(self, tracker, mode_3d="image", smoothing=True):
self.tracker = tracker
self.mode_3d = mode_3d
if self.mode_3d == "mixed" and not self.tracker.xyz:
print("'mixed' 3d visualization needs the tracker to be in 'xyz' mode !")
print("3d visualization falling back to 'world' mode.")
self.mode_3d = 'world'
if self.mode_3d == "image":
self.vis3d = Visu3D(zoom=0.7, segment_radius=10)
z = min(tracker.img_h, tracker.img_w)/3
self.vis3d.create_grid([0,tracker.img_h,-z],[tracker.img_w,tracker.img_h,-z],[tracker.img_w,tracker.img_h,z],[0,tracker.img_h,z],5,2) # Floor
self.vis3d.create_grid([0,0,z],[tracker.img_w,0,z],[tracker.img_w,tracker.img_h,z],[0,tracker.img_h,z],5,2) # Wall
self.vis3d.init_view()
elif "world" in self.mode_3d:
self.vis3d = Visu3D(bg_color=(0.2, 0.2, 0.2), zoom=1.1, segment_radius=0.01)
x_max = 0.2 if self.tracker.solo else 0.4
y_max = 0.2
z_max = 0.2
self.vis3d.create_grid([-x_max,y_max,-z_max],[x_max,y_max,-z_max],[x_max,y_max,z_max],[-x_max,y_max,z_max],1 if self.tracker.solo else 2,1) # Floor
self.vis3d.create_grid([-x_max,y_max,z_max],[x_max,y_max,z_max],[x_max,-y_max,z_max],[-x_max,-y_max,z_max],1 if self.tracker.solo else 2,1) # Wall
self.vis3d.init_view()
elif self.mode_3d == "mixed":
self.vis3d = Visu3D(bg_color=(0.4, 0.4, 0.4), zoom=0.8, segment_radius=0.01)
x_max = 0.9
y_max = 0.6
grid_depth = 2
self.vis3d.create_grid([-x_max,y_max,0],[x_max,y_max,0],[x_max,y_max,grid_depth],[-x_max,y_max,grid_depth],2,grid_depth) # Floor
self.vis3d.create_grid([-x_max,y_max,grid_depth],[x_max,y_max,grid_depth],[x_max,-y_max,grid_depth],[-x_max,-y_max,grid_depth],2,2) # Wall
self.vis3d.create_camera()
self.vis3d.init_view()
self.smoothing = smoothing
self.filter = None
if self.smoothing:
if tracker.solo:
if self.mode_3d == "image":
self.filter = [LandmarksSmoothingFilter(min_cutoff=0.01, beta=40, derivate_cutoff=1)]
else:
self.filter = [LandmarksSmoothingFilter(min_cutoff=1, beta=20, derivate_cutoff=10, disable_value_scaling=True)]
else:
if self.mode_3d == "image":
self.filter = [
LandmarksSmoothingFilter(min_cutoff=0.01,beta=40,derivate_cutoff=1),
LandmarksSmoothingFilter(min_cutoff=0.01,beta=40,derivate_cutoff=1)
]
else:
self.filter = [
LandmarksSmoothingFilter(min_cutoff=1, beta=20, derivate_cutoff=10, disable_value_scaling=True),
LandmarksSmoothingFilter(min_cutoff=1, beta=20, derivate_cutoff=10, disable_value_scaling=True)
]
self.nb_hands_in_previous_frame = 0
def draw_hand(self, hand, i):
if self.mode_3d == "image":
# Denormalize z-component of 'norm_landmarks'
lm_z = (hand.norm_landmarks[:,2:3] * hand.rect_w_a / 0.4).astype(np.int32)
# ... and concatenates with x and y components of 'landmarks'
points = np.hstack((hand.landmarks, lm_z))
radius = hand.rect_w_a / 30 # Thickness of segments depends on the hand size
elif "world" in self.mode_3d:
if self.mode_3d == "raw_world":
points = hand.world_landmarks
else: # "world"
points = hand.get_rotated_world_landmarks()
if not self.tracker.solo:
delta_x = -0.2 if hand.label == "right" else 0.2
points = points + np.array([delta_x,0,0])
radius = 0.01
elif self.mode_3d == "mixed":
wrist_xyz = hand.xyz / 1000.0
# Beware that y value of (x,y,z) coordinates given by depth sensor is negative
# in the lower part of the image and positive in the upper part.
wrist_xyz[1] = -wrist_xyz[1]
points = hand.get_rotated_world_landmarks()
points = points + wrist_xyz - points[0]
radius = 0.01
if self.smoothing:
points = self.filter[i].apply(points, object_scale=hand.rect_w_a)
for i,a_b in enumerate(LINES_HAND):
a, b = a_b
self.vis3d.add_segment(points[a], points[b], radius=radius, color=[1*(1-hand.handedness),hand.handedness,0]) # if hand.handedness<0.5 else [0,1,0])
def draw(self, hands):
if self.smoothing and len(hands) != self.nb_hands_in_previous_frame:
for f in self.filter: f.reset()
self.vis3d.clear()
self.vis3d.try_move()
self.vis3d.add_geometries()
for i, hand in enumerate(hands):
self.draw_hand(hand, i)
self.vis3d.render()
self.nb_hands_in_previous_frame = len(hands)
parser = argparse.ArgumentParser()
# parser.add_argument('-e', '--edge', action="store_true",
# help="Use Edge mode (postprocessing runs on the device)")
parser_tracker = parser.add_argument_group("Tracker arguments")
parser_tracker.add_argument('-i', '--input', type=str,
help="Path to video or image file to use as input (if not specified, use OAK color camera)")
parser_tracker.add_argument("--pd_model", type=str,
help="Path to a blob file for palm detection model")
parser_tracker.add_argument("--lm_model", type=str,
help="Path to a blob file for landmark model")
parser_tracker.add_argument('-s', '--solo', action="store_true",
help="Detect one hand max")
parser_tracker.add_argument('-f', '--internal_fps', type=int,
help="Fps of internal color camera. Too high value lower NN fps (default= depends on the model)")
parser_tracker.add_argument('--internal_frame_height', type=int,
help="Internal color camera frame height in pixels")
parser_tracker.add_argument('--single_hand_tolerance_thresh', type=int, default=0,
help="(Duo mode only) Number of frames after only one hand is detected before calling palm detection (default=%(default)s)")
parser_renderer3d = parser.add_argument_group("3D Renderer arguments")
parser_renderer3d.add_argument('-m', '--mode_3d', nargs='?',
choices=['image', 'world', 'raw_world', 'mixed'], const='image', default='image',
help="Specify the 3D coordinates used. See README for description (default=%(default)s)")
parser_renderer3d.add_argument('--no_smoothing', action="store_true",
help="Disable smoothing filter (smoothing works only in solo mode)")
args = parser.parse_args()
args.edge = True
if args.edge:
from HandTrackerEdge import HandTracker
else:
from HandTracker import HandTracker
dargs = vars(args)
tracker_args = {a:dargs[a] for a in ['pd_model', 'lm_model', 'internal_fps', 'internal_frame_height'] if dargs[a] is not None}
tracker = HandTracker(
input_src=args.input,
use_world_landmarks=args.mode_3d != "image",
solo=args.solo,
xyz= args.mode_3d == "mixed",
stats=True,
single_hand_tolerance_thresh=args.single_hand_tolerance_thresh,
lm_nb_threads=1,
**tracker_args
)
renderer3d = HandTracker3DRenderer(tracker, mode_3d=args.mode_3d, smoothing=not args.no_smoothing)
renderer2d = HandTrackerRenderer(tracker)
pause = False
hands = []
while True:
# Run hand tracker on next frame
if not pause:
frame, hands, bag = tracker.next_frame()
if frame is None: break
# Render 2d frame
frame = renderer2d.draw(frame, hands, bag)
cv2.imshow("HandTracker", frame)
key = cv2.waitKey(1)
# Draw hands on open3d canvas
renderer3d.draw(hands)
if key == 27 or key == ord('q'):
break
elif key == 32: # space
pause = not pause
elif key == ord('s'):
if renderer3d.filter:
renderer3d.smoothing = not renderer3d.smoothing
renderer2d.exit()
tracker.exit()