-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathcreate_dataset.py
300 lines (264 loc) · 11.7 KB
/
create_dataset.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
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
import os, sys, random
import matplotlib.pyplot as plt
from shutil import copyfile
import numpy as np
from mtcnn.mtcnn import MTCNN
from scipy import misc
import cv2
import time
import shutil
def last_8chars(x):
"""Function that aids at list sorting.
Args:
x: String name of files.
Returns:
A string of the last 8 characters of x.
"""
return (x[-8:])
def last_3chars(x):
"""Function that aids at list sorting.
Args:
x: String name of files.
Returns:
A string of the last 8 characters of x.
"""
return (x[-3:])
def get_vid_name(file):
"""Creates the new video name.
Args:
file: String name of the original image name.
Returns:
A string name defined by the Subject and Session
numbers.
"""
return file.split("_")[0] + "_" + file.split("_")[1]
def create_moving_batch(images_list, depth):
"""Creates the list of neutral and emotion-labeled images to be processed.
From the whole list of images of a given session, create an 2 dimensional
array with neutral and emotion-labeled image names. The quantity of
images per index in batch (same for neutral and emotion) is defined by
the selected clip depth. From the sequence, the first depth number of
images are placed at batch[1], and the last depth number of images at
batch[0].
Args:
images_list: String array with paths to all images from a given session.
depth: Integer number of images to be appended in each batch index. It
must be at most half of the total session quantity of images.
Returns:
A 2D string array containing the path to the images to be processed later
to create 3D arrays:
batch[0] = emotion-labeled images
batch[1] = neutral-labeled images
"""
batch = [[], []]
for index, image in enumerate(sorted(images_list, key=last_8chars)):
if (index < depth):
batch[1] = np.append(batch[1], image)
if (index > len(images_list) - depth - 1):
batch[0] = np.append(batch[0], image)
return batch
def move_images(images_list, src, dest, neu_dest, min_seq_len, depth, t_height,
t_width, crop_faces_flag, detector):
"""Creates the Numpy binary files with a 3D array of a sequence of images.
Args:
images_list: String array with paths to all images from a given Session.
src: String path to a selected images Session.
dest: String path to where the emotion-labeled Numpy binary files must
be saved.
neu_dest: String path to where the neutral-labeled Numpy binary files
must be saved.
min_seq_len: Integer number of minimum images in a given Session for them
to be processed. Only Sessions with more than this value will be
considered.
depth: Integer number of images to be appended to create a 3D array.
This is the temporal depth of the input instances.
t_height: Integer number of the target height to which the images will
be resized.
t_width: Integer number of the target width to which the images will
be resized.
crop_faces_flag: Boolean value that indicates if facial cropping will be
done before resizing the original images.
detector: MTCNN object.
"""
# Create neutral and emotion-labeled lists to be processed.
batch = create_moving_batch(images_list, depth)
# Emotion batch
vid_name = get_vid_name(batch[0][0])
if not os.path.exists(dest + '/' + vid_name):
im_size = cv2.imread(src + batch[0][0]).shape
im_shape = im_size
vid = np.zeros(0)
for index, image in enumerate(batch[0]):
if crop_faces_flag:
im = crop_face(src + image, detector)
else:
im = cv2.imread(src + image)
im_resized = cv2.resize(im, (t_height, t_width), cv2.INTER_LINEAR)
if index == 0:
temp = np.zeros((0, im_resized.shape[0], im_resized.shape[1],
im_resized.shape[2]))
temp = np.append(temp, [im_resized], axis=0)
vid = temp
# Save to Numpy binary file
try:
np.save(dest + '/' + vid_name, vid)
except Exception as e:
print("Unable to save instance at:",
dest + str(vid_name[0]) + "_" + str(vid_name[1]))
raise
# Neutral batch
vid_name = get_vid_name(batch[1][0])
if not os.path.exists(neu_dest + '/' + vid_name):
im_size = cv2.imread(src + batch[1][0]).shape
vid = np.zeros(0)
for index, image in enumerate(batch[1]):
if crop_faces_flag:
im = crop_face(src + image, detector)
else:
im = cv2.imread(src + image)
im_resized = cv2.resize(im, (t_height, t_width), cv2.INTER_LINEAR)
if index == 0:
temp = np.zeros((0, im_resized.shape[0], im_resized.shape[1],
im_resized.shape[2]))
temp = np.append(temp, [im_resized], axis=0)
vid = temp
# Save to Numpy binary file
try:
np.save(neu_dest + '/' + vid_name, vid)
except Exception as e:
print("Unable to save instance at:",
dest + str(vid_name[0]) + "_" + str(vid_name[1]))
raise
def get_label(filepath):
"""Returns the label for a given Session by reading its
corresponding CSV label file.
Args:
filepath: String path to the CSV file.
Returns:
String label name if found, else a -1.
"""
if os.path.exists(filepath) and os.listdir(filepath):
g = open(filepath + str(os.listdir(filepath)[0]), 'r')
label = g.readline().split('.')[0].replace(" ", "")
return label
else:
return -1
def crop_face(image_path, detector):
"""Returns a facial image if face is detected.
If the MTCNN face detector finds a face in the image, the returned
image size will be defined by the bounding box returned by the face
detector. Otherwise, the imagewill be centered cropped by empirically
found parameters.
Args:
image_path: String path to the image file.
detector: MTCNN object
Returns:
OpenCV image object of the processed image.
"""
image = cv2.imread(image_path)
if detector:
result = detector.detect_faces(image)
if result:
bbox = result[0]['box']
image = image[bbox[1]:bbox[1] + bbox[3], bbox[0]:
bbox[0] + bbox[2], :]
else:
print("Face could not be detected in image",
image_path + ", " + "proceeding with center cropping.")
image = image[:448, 96:-96, :]
return image
def gen_emotionsFolders(main_dir, label_main_dir, image_main_dir, emotions_dir,
crop_faces, neutral_label, min_seq_len, depth,
t_height, t_width):
"""Generates a folder of Numpy binary files with 3D arrays of images for each
label in CK+ database.
Check each Session if the number of images is more than min_seq_len and
create a 3D array by appending a depth number of images together. Each
image is resized to (t_height, t_width) from its original size or from
a face bounding box if the flag crop_faces is true. Numpy binary files
are saved in emotions_dir.
Args:
main_dir: String path to where the labels and images folders are placed.
label_main_dir: String path to CK+ labels folder
image_main_dir: String path to CK+ images folder
emotions_dir: String path where to Numpy binary files will be saved.
crop_faces_flag: Boolean value that indicates if facial cropping will be
done before resizing the original images.
neutral_label: String name of the neutral label. This value has to be
defined by the user because it will not appear in the CSV files.
min_seq_len: Integer number of minimum images in a given Session for them
to be processed. Only Sessions with more than this value will be
considered.
depth: Integer number of images to be appended to create a 3D array.
This is the temporal depth of the input instances.
t_height: Integer number of the target height to which the images will
be resized.
t_width: Integer number of the target width to which the images will
be resized.
"""
try:
assert (min_seq_len >= depth * 2)
except AssertionError as e:
print('The depth should be at most equal to half of min_seq_len.')
raise
start_time = time.time()
print("Getting images...")
if crop_faces=="True" or crop_faces=="1":
detector = MTCNN()
else:
detector = 0
if os.path.exists(emotions_dir):
shutil.rmtree(emotions_dir, ignore_errors=True)
list_labels = np.array([])
im_shape = []
list_labels = np.append(list_labels, neutral_label)
if not os.path.exists(emotions_dir + neutral_label):
os.makedirs(emotions_dir + str(neutral_label))
for subject in sorted(os.listdir(image_main_dir), key=last_3chars):
for session in sorted(
os.listdir(image_main_dir + str(subject)), key=last_3chars):
if session != ".DS_Store":
images_path = image_main_dir + str(subject) + '/' + str(
session) + '/'
images_list = [
x
for x in sorted(os.listdir(images_path), key=last_8chars)
if x.split(".")[1] == "png"
]
if (images_list and len(images_list) >= min_seq_len):
label_path = label_main_dir + str(subject) + '/' + str(
session) + '/'
label = get_label(label_path)
if label != -1:
if label not in list_labels:
list_labels = np.append(list_labels, label)
if not os.path.exists(emotions_dir + str(label)):
os.makedirs(emotions_dir + str(label))
im_shape = move_images(
images_list, images_path,
emotions_dir + str(label),
emotions_dir + str(neutral_label), min_seq_len,
depth, t_height, t_width, crop_faces, detector)
duration = time.time() - start_time
print("\nDone! Total time %.1f seconds." % (duration))
test_vid = np.load(emotions_dir + neutral_label + "/" +
sorted(os.listdir(emotions_dir + neutral_label))[0])
print("Clip size:", test_vid.shape)
print("Neutral label example:")
plt.imshow(test_vid[0][:, :, 0], cmap="gray", interpolation="nearest")
plt.axis("off")
plt.show()
def main(argv):
gen_emotionsFolders(
main_dir=str(argv[0]),
label_main_dir=str(argv[1]),
image_main_dir=str(argv[2]),
emotions_dir=str(argv[3]),
crop_faces=str(argv[4]),
neutral_label=str(argv[5]),
min_seq_len=int(argv[6]),
depth=int(argv[7]),
t_height=int(argv[8]),
t_width=int(argv[9]))
if __name__ == "__main__":
main(sys.argv[1:])