-
Notifications
You must be signed in to change notification settings - Fork 1
/
Social_Distance.py
220 lines (190 loc) · 11.3 KB
/
Social_Distance.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
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
#================================================================
from ctypes import *
import math
import random
import os
import cv2
import numpy as np
import time
import darknet
from itertools import combinations
import pafy
import youtube_dl
def is_close(p1, p2):
"""
#================================================================
# Purpose : Calculate Euclidean Distance between two points
#================================================================
:param:
p1, p2 = two points for calculating Euclidean Distance
:return:
dst = Euclidean Distance between two 2d points
"""
dst = math.sqrt(p1**2 + p2**2)
#=================================================================#
return dst
def convertBack(x, y, w, h):
#================================================================
# Purpose : Converts center coordinates to rectangle coordinates
#================================================================
"""
:param:
x, y = midpoint of bbox
w, h = width, height of the bbox
:return:
xmin, ymin, xmax, ymax
"""
xmin = int(round(x - (w / 2)))
xmax = int(round(x + (w / 2)))
ymin = int(round(y - (h / 2)))
ymax = int(round(y + (h / 2)))
return xmin, ymin, xmax, ymax
def cvDrawBoxes(detections, img):
"""
:param:
detections = total detections in one frame
img = image from detect_image method of darknet
:return:
img with bbox
"""
#================================================================
# Purpose : Filter out Persons class from detections and get
# bounding box centroid for each person detection.
#================================================================
if len(detections) > 0: # At least 1 detection in the image and check detection presence in a frame
centroid_dict = dict() # Function creates a dictionary and calls it centroid_dict
objectId = 0 # We inialize a variable called ObjectId and set it to 0
for detection in detections: # In this if statement, we filter all the detections for persons only
# Check for the only person name tag
name_tag = str(detection[0].decode()) # Coco file has string of all the names
if name_tag == 'person':
x, y, w, h = detection[2][0],\
detection[2][1],\
detection[2][2],\
detection[2][3] # Store the center points of the detections
xmin, ymin, xmax, ymax = convertBack(float(x), float(y), float(w), float(h)) # Convert from center coordinates to rectangular coordinates, We use floats to ensure the precision of the BBox
# Append center point of bbox for persons detected.
centroid_dict[objectId] = (int(x), int(y), xmin, ymin, xmax, ymax) # Create dictionary of tuple with 'objectId' as the index center points and bbox
objectId += 1 #Increment the index for each detection
#=================================================================#
#=================================================================
# Purpose : Determine which person bbox are close to each other
#=================================================================
red_zone_list = [] # List containing which Object id is in under threshold distance condition.
red_line_list = []
for (id1, p1), (id2, p2) in combinations(centroid_dict.items(), 2): # Get all the combinations of close detections, #List of multiple items - id1 1, points 2, 1,3
dx, dy = p1[0] - p2[0], p1[1] - p2[1] # Check the difference between centroid x: 0, y :1
distance = is_close(dx, dy) # Calculates the Euclidean distance
if distance < 75.0: # Set our social distance threshold - If they meet this condition then..
if id1 not in red_zone_list:
red_zone_list.append(id1) # Add Id to a list
red_line_list.append(p1[0:2]) # Add points to the list
if id2 not in red_zone_list:
red_zone_list.append(id2) # Same for the second id
red_line_list.append(p2[0:2])
for idx, box in centroid_dict.items(): # dict (1(key):red(value), 2 blue) idx - key box - value
if idx in red_zone_list: # if id is in red zone list
cv2.rectangle(img, (box[2], box[3]), (box[4], box[5]), (255, 0, 0), 2) # Create Red bounding boxes #starting point, ending point size of 2
else:
cv2.rectangle(img, (box[2], box[3]), (box[4], box[5]), (0, 255, 0), 2) # Create Green bounding boxes
#=================================================================#
#=================================================================
# Purpose : Display Risk Analytics and Show Risk Indicators
#=================================================================
text = "People at Risk: %s" % str(len(red_zone_list)) # Count People at Risk
location = (10,25) # Set the location of the displayed text
cv2.putText(img, text, location, cv2.FONT_HERSHEY_SIMPLEX, 1, (246,86,86), 2, cv2.LINE_AA) # Display Text
for check in range(0, len(red_line_list)-1): # Draw line between nearby bboxes iterate through redlist items
start_point = red_line_list[check]
end_point = red_line_list[check+1]
check_line_x = abs(end_point[0] - start_point[0]) # Calculate the line coordinates for x
check_line_y = abs(end_point[1] - start_point[1]) # Calculate the line coordinates for y
if (check_line_x < 75) and (check_line_y < 25): # If both are We check that the lines are below our threshold distance.
cv2.line(img, start_point, end_point, (255, 0, 0), 2) # Only above the threshold lines are displayed.
#=================================================================#
return img
netMain = None
metaMain = None
altNames = None
def YOLO():
"""
Perform Object detection
"""
global metaMain, netMain, altNames
configPath = "./cfg/yolov4-tiny.cfg" # Path to cfg
weightPath = "./yolov4-tiny.weights" # Path to weights
metaPath = "./cfg/coco.data" # Path to meta data
if not os.path.exists(configPath): # Checks whether file exists otherwise return ValueError
raise ValueError("Invalid config path `" +
os.path.abspath(configPath)+"`")
if not os.path.exists(weightPath):
raise ValueError("Invalid weight path `" +
os.path.abspath(weightPath)+"`")
if not os.path.exists(metaPath):
raise ValueError("Invalid data file path `" +
os.path.abspath(metaPath)+"`")
if netMain is None: # Checks the metaMain, NetMain and altNames. Loads it in script
netMain = darknet.load_net_custom(configPath.encode(
"ascii"), weightPath.encode("ascii"), 0, 1) # batch size = 1
if metaMain is None:
metaMain = darknet.load_meta(metaPath.encode("ascii"))
if altNames is None:
try:
with open(metaPath) as metaFH:
metaContents = metaFH.read()
import re
match = re.search("names *= *(.*)$", metaContents,
re.IGNORECASE | re.MULTILINE)
if match:
result = match.group(1)
else:
result = None
try:
if os.path.exists(result):
with open(result) as namesFH:
namesList = namesFH.read().strip().split("\n")
altNames = [x.strip() for x in namesList]
except TypeError:
pass
except Exception:
pass
#cap = cv2.VideoCapture(0) # Uncomment to use Webcam
#cap = cv2.VideoCapture("Video_for_Testing.mp4") # Uncomment for Local Stored video detection - Set input video
#url = "https://www.youtube.com/watch?v=isveXCH4NcM" # Uncomment these lines for video from youtube
#video = pafy.new(url)
#best = video.getbest(preftype="mp4")
#cap = cv2.VideoCapture()
#cap.open(best.url)
#cap = cv2.VideoCapture('http://192.168.0.106:4747/mjpegfeed') # Uncomment for Video from Mobile Camera (DroidCam Hosted Camera)
frame_width = int(cap.get(3)) # Returns the width and height of capture video
frame_height = int(cap.get(4))
new_height, new_width = frame_height // 2, frame_width // 2
#print("Video Reolution: ",(width, height))
#out = cv2.VideoWriter("output.avi", cv2.VideoWriter_fourcc(*"MJPG"), 10.0, # Uncomment to save the output video # Set the Output path for video writer
#(new_width, new_height))
# print("Starting the YOLO loop...")
# Create an image we reuse for each detect
darknet_image = darknet.make_image(new_width, new_height, 3) # Create image according darknet for compatibility of network
while True: # Load the input frame and write output frame.
prev_time = time.time()
ret, frame_read = cap.read()
# Check if frame present :: 'ret' returns True if frame present, otherwise break the loop.
if not ret:
break
frame_rgb = cv2.cvtColor(frame_read, cv2.COLOR_BGR2RGB) # Convert frame into RGB from BGR and resize accordingly
frame_resized = cv2.resize(frame_rgb,
(new_width, new_height),
interpolation=cv2.INTER_LINEAR)
darknet.copy_image_from_bytes(darknet_image,frame_resized.tobytes()) # Copy that frame bytes to darknet_image
detections = darknet.detect_image(netMain, metaMain, darknet_image, thresh=0.25) # Detection occurs at this line and return detections, for customize we can change
image = cvDrawBoxes(detections, frame_resized) # Call the function cvDrawBoxes() for colored bounding box per class
image = cv2.cvtColor(image, cv2.COLOR_BGR2RGB)
print(1/(time.time()-prev_time)) # Prints frames per second
cv2.imshow('Demo', image) # Display Image window
cv2.waitKey(3)
#out.write(image) # Write that frame into output video
cap.release() # For releasing cap and out.
#out.release() # Uncomment to save the output video
print(":::Video Write Completed")
if __name__ == "__main__":
YOLO() # Calls the main function YOLO()