-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlib_tracker.py
163 lines (133 loc) · 6.28 KB
/
lib_tracker.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
'''
This script implements a simple `Tracker` class.
'''
import numpy as np
import cv2
import math
import functools
class Tracker(object):
''' A simple tracker:
For previous skeletons(S1) and current skeletons(S2),
S1[i] and S2[j] are matched, if:
1. For S1[i], S2[j] is the most nearest skeleton in S2.
2. For S2[j], S1[i] is the most nearest skeleton in S1.
3. The distance between S1[i] and S2[j] are smaller than self._dist_thresh.
(Unit: The image width is 1.0, the image height is scale_h=rows/cols)
For unmatched skeletons in S2, they are considered
as new people appeared in the video.
'''
def __init__(self, dist_thresh=20, max_humans=5):
'''
Arguments:
dist_thresh {float}: 0.0~1.0. The distance between the joints
of the two matched people should be smaller than this.
The image width and height has a unit length of 1.0.
max_humans {int}: max humans to track.
If the number of humans exceeds this threshold, the new
skeletons will be abandoned instead of taken as new people.
'''
self._dist_thresh = dist_thresh
self._max_humans = max_humans
self._dict_id2skeleton = {}
self._cnt_humans = 0
def track(self, curr_skels):
''' Track the input skeletons by matching them with previous skeletons,
and then obtain their corresponding human id.
Arguments:
curr_skels {list of list}: each sub list is a person's skeleton.
Returns:
self._dict_id2skeleton {dict}: a dict mapping human id to his/her skeleton.
'''
curr_skels = self._sort_skeletons_by_dist_to_center(curr_skels)
N = len(curr_skels)
# Match skeletons between curr and prev
if len(self._dict_id2skeleton) > 0:
ids, prev_skels = map(list, zip(*self._dict_id2skeleton.items()))
good_matches = self._match_features(prev_skels, curr_skels)
self._dict_id2skeleton = {}
is_matched = [False]*N
for i2, i1 in good_matches.items():
human_id = ids[i1]
self._dict_id2skeleton[human_id] = np.array(curr_skels[i2])
is_matched[i2] = True
unmatched_idx = [i for i, matched in enumerate(
is_matched) if not matched]
else:
good_matches = []
unmatched_idx = range(N)
# Add unmatched skeletons (which are new skeletons) to the list
num_humans_to_add = min(len(unmatched_idx),
self._max_humans - len(good_matches))
for i in range(num_humans_to_add):
self._cnt_humans += 1
self._dict_id2skeleton[self._cnt_humans] = np.array(
curr_skels[unmatched_idx[i]])
return self._dict_id2skeleton
def _get_neck(self, skeleton):
x, y = skeleton[2], skeleton[3]
return x, y
def _sort_skeletons_by_dist_to_center(self, skeletons):
''' Skeletons are sorted based on the distance
between neck and image center, from small to large.
A skeleton near center will be processed first and be given a smaller human id.
Here the center is defined as (0.5, 0.5), although it's not accurate due to h_scale.
'''
def calc_dist(p1, p2): return (
(p1[0] - p2[0])**2 + (p1[1] - p2[1])**2)**0.5
def cost(skeleton):
x1, y1 = self._get_neck(skeleton)
return calc_dist((x1, y1), (0.5, 0.5)) # dist to center
def cmp(a, b): return (a > b)-(a < b)
def mycmp(sk1, sk2): return cmp(cost(sk1), cost(sk2))
sorted_skeletons = sorted(skeletons, key=functools.cmp_to_key(mycmp))
return sorted_skeletons
def _match_features(self, features1, features2):
''' Match the features. Output the matched indices.
Returns:
good_matches {dict}: a dict which matches the
`index of features2` to `index of features1`.
'''
features1, features2 = np.array(features1), np.array(features2)
#cost = lambda x1, x2: np.linalg.norm(x1-x2)
def calc_dist(p1, p2): return (
(p1[0] - p2[0])**2 + (p1[1] - p2[1])**2)**0.5
def cost(sk1, sk2):
# neck, shoulder, elbow, hip, knee
joints = np.array([2, 3, 4, 5, 6, 7, 10, 11, 12,
13, 16, 17, 18, 19, 22, 23, 24, 25])
sk1, sk2 = sk1[joints], sk2[joints]
valid_idx = np.logical_and(sk1 != 0, sk2 != 0)
sk1, sk2 = sk1[valid_idx], sk2[valid_idx]
sum_dist, num_points = 0, int(len(sk1)/2)
if num_points == 0:
return 99999
else:
for i in range(num_points): # compute distance between each pair of joint
idx = i * 2
sum_dist += calc_dist(sk1[idx:idx+2], sk2[idx:idx+2])
mean_dist = sum_dist / num_points
mean_dist /= (1.0 + 0.05*num_points) # more points, the better
return mean_dist
# If f1i is matched to f2j and vice versa, the match is good.
good_matches = {}
n1, n2 = len(features1), len(features2)
if n1 and n2:
# dist_matrix[i][j] is the distance between features[i] and features[j]
dist_matrix = [[cost(f1, f2) for f2 in features2]
for f1 in features1]
dist_matrix = np.array(dist_matrix)
# Find the match of features1[i]
matches_f1_to_f2 = [dist_matrix[row, :].argmin()
for row in range(n1)]
# Find the match of features2[i]
matches_f2_to_f1 = [dist_matrix[:, col].argmin()
for col in range(n2)]
for i1, i2 in enumerate(matches_f1_to_f2):
if matches_f2_to_f1[i2] == i1 and dist_matrix[i1, i2] < self._dist_thresh:
good_matches[i2] = i1
if 0:
print("distance matrix:", dist_matrix)
print("matches_f1_to_f2:", matches_f1_to_f2)
print("matches_f1_to_f2:", matches_f2_to_f1)
print("good_matches:", good_matches)
return good_matches